Hoppa till huvudinnehåll
Upsells och downsells låter dig erbjuda ytterligare produkter eller abonnemangsförändringar till kunder med deras sparade betalningsmetoder. Detta möjliggör köp med ett klick som hoppar över betalningsinsamling, vilket dramatiskt förbättrar konverteringsgraden.

Post-Purchase Upsells

Erbjud kompletterande produkter direkt efter kassan med ett klick för att köpa.

Subscription Upgrades

Flytta kunder till högre nivåer med automatisk proration och omedelbar fakturering.

Cross-Sells

Lägg till relaterade produkter till befintliga kunder utan att återigen ange betalningsuppgifter.

Översikt

Uppersell och neddersell är kraftfulla strategier för intäktsoptimering:
  • Uppersell: Erbjud en produkt eller uppgradering med högre värde (t.ex. Pro-plan istället för Basic)
  • Neddersell: Erbjud ett alternativ med lägre pris när en kund avböjer eller nedgraderas
  • Korsförsäljning: Föreslå kompletterande produkter (t.ex. tillägg, relaterade artiklar)
Dodo Payments möjliggör dessa flöden genom payment_method_id-parametern, som låter dig debitera en kunds sparade betalningsmetod utan att kräva att de anger kortuppgifter igen.

Nyckelfördelar

FördelEffekt
Köp med ett klickHoppa över betalningsformuläret helt för återkommande kunder
Högre konverteringMinska friktionen i beslutsmomentet
Omedelbar behandlingDebiteringar sker omedelbart med confirm: true
Sömlös användarupplevelseKunder stannar kvar i din app under hela flödet

Hur det fungerar

Förutsättningar

Innan du implementerar uppersell och neddersell, se till att du har:
1

Customer with Saved Payment Method

Kunder måste ha genomfört minst ett köp. Betalningsmetoder sparas automatiskt när kunder genomför kassan.
2

Products Configured

Skapa dina upsell-produkter i Dodo Payments-instrumentpanelen. Dessa kan vara engångsbetalningar, prenumerationer eller tillägg.
3

Webhook Endpoint

Ställ in webhookar för att hantera payment.succeeded, payment.failed och subscription.plan_changed-händelser.

Hämta kundbetalningsmetoder

Innan du erbjuder en uppersell, hämta kundens sparade betalningsmetoder:
import DodoPayments from 'dodopayments';

const client = new DodoPayments({
  bearerToken: process.env.DODO_PAYMENTS_API_KEY,
  environment: 'live_mode',
});

async function getPaymentMethods(customerId: string) {
  const paymentMethods = await client.customers.listPaymentMethods(customerId);
  
  // Returns array of saved payment methods
  // Each has: payment_method_id, type, card (last4, brand, exp_month, exp_year)
  return paymentMethods;
}

// Example usage
const methods = await getPaymentMethods('cus_123');
console.log('Available payment methods:', methods);

// Use the first available method for upsell
const primaryMethod = methods[0]?.payment_method_id;
Betalningsmetoder sparas automatiskt när kunder slutför köpet. Du behöver inte spara dem uttryckligen.

Engångsupersell efter köp

Erbjud ytterligare produkter omedelbart efter ett lyckat köp. Kunden kan godkänna med ett enda klick eftersom deras betalningsmetod redan är sparad.

Implementering

import DodoPayments from 'dodopayments';

const client = new DodoPayments({
  bearerToken: process.env.DODO_PAYMENTS_API_KEY,
  environment: 'live_mode',
});

async function createOneClickUpsell(
  customerId: string,
  paymentMethodId: string,
  upsellProductId: string
) {
  // Create checkout session with saved payment method
  // confirm: true processes the payment immediately
  const session = await client.checkoutSessions.create({
    product_cart: [
      {
        product_id: upsellProductId,
        quantity: 1
      }
    ],
    customer: {
      customer_id: customerId
    },
    payment_method_id: paymentMethodId,
    confirm: true,  // Required when using payment_method_id
    return_url: 'https://yourapp.com/upsell-success',
    feature_flags: {
      redirect_immediately: true  // Skip success page
    },
    metadata: {
      upsell_source: 'post_purchase',
      original_order_id: 'order_123'
    }
  });

  return session;
}

// Example: Offer premium add-on after initial purchase
async function handlePostPurchaseUpsell(customerId: string) {
  // Get customer's payment methods
  const methods = await client.customers.listPaymentMethods(customerId);
  
  if (methods.length === 0) {
    console.log('No saved payment methods available');
    return null;
  }

  // Create the upsell with one-click checkout
  const upsell = await createOneClickUpsell(
    customerId,
    methods[0].payment_method_id,
    'prod_premium_addon'
  );

  console.log('Upsell processed:', upsell.session_id);
  return upsell;
}
När du använder payment_method_id måste du ange confirm: true och tillhandahålla en befintlig customer_id. Betalningsmetoden måste tillhöra den kunden.

Uppgraderingar av abonnemang

Flytta kunder till abonnemangsplaner av högre nivå med automatisk proration.

Förhandsgranska innan du åtar dig

Förhandsgranska alltid planändringar för att visa kunder exakt vad de kommer att debiteras:
async function previewUpgrade(
  subscriptionId: string,
  newProductId: string
) {
  const preview = await client.subscriptions.previewChangePlan(subscriptionId, {
    product_id: newProductId,
    quantity: 1,
    proration_billing_mode: 'difference_immediately'
  });

  return {
    immediateCharge: preview.immediate_charge?.summary,
    newPlan: preview.new_plan,
    effectiveDate: preview.effective_date
  };
}

// Show customer the charge before confirming
const preview = await previewUpgrade('sub_123', 'prod_pro_plan');
console.log(`Upgrade will charge: ${preview.immediateCharge}`);

Genomföra uppgraderingen

async function upgradeSubscription(
  subscriptionId: string,
  newProductId: string,
  prorationMode: 'prorated_immediately' | 'difference_immediately' | 'full_immediately' = 'difference_immediately'
) {
  const result = await client.subscriptions.changePlan(subscriptionId, {
    product_id: newProductId,
    quantity: 1,
    proration_billing_mode: prorationMode
  });

  return {
    status: result.status,
    subscriptionId: result.subscription_id,
    paymentId: result.payment_id,
    invoiceId: result.invoice_id
  };
}

// Upgrade from Basic ($30) to Pro ($80)
// With difference_immediately: charges $50 instantly
const upgrade = await upgradeSubscription('sub_123', 'prod_pro_plan');
console.log('Upgrade status:', upgrade.status);

Prorateringslägen

Välj hur kunder debiteras vid uppgradering:
LägeBeteendeBäst för
difference_immediatelyDebiterar prisdifferensen omedelbart (3030→80 = $50)Enkla uppgraderingar
prorated_immediatelyDebiterar baserat på återstående tid i faktureringscykelnRättvis tidsbaserad fakturering
full_immediatelyDebiterar hela priset för det nya abonnemanget, ignorerar återstående tidFaktureringscykeln återställs
Använd difference_immediately för tydliga uppgraderingsflöden. Använd prorated_immediately när du vill ta hänsyn till oanvänd tid på det nuvarande abonnemanget.

Korsförsäljning

Lägg till kompletterande produkter för befintliga kunder utan att behöva ange betalningsuppgifter igen.

Implementering

async function createCrossSell(
  customerId: string,
  paymentMethodId: string,
  productId: string,
  quantity: number = 1
) {
  // Create a one-time payment using saved payment method
  const payment = await client.payments.create({
    product_cart: [
      {
        product_id: productId,
        quantity: quantity
      }
    ],
    customer_id: customerId,
    payment_method_id: paymentMethodId,
    return_url: 'https://yourapp.com/purchase-complete',
    metadata: {
      purchase_type: 'cross_sell',
      source: 'product_recommendation'
    }
  });

  return payment;
}

// Example: Customer bought a course, offer related ebook
async function offerRelatedProduct(customerId: string, relatedProductId: string) {
  const methods = await client.customers.listPaymentMethods(customerId);
  
  if (methods.length === 0) {
    // Fall back to standard checkout
    return client.checkoutSessions.create({
      product_cart: [{ product_id: relatedProductId, quantity: 1 }],
      customer: { customer_id: customerId },
      return_url: 'https://yourapp.com/purchase-complete'
    });
  }

  // One-click purchase
  return createCrossSell(customerId, methods[0].payment_method_id, relatedProductId);
}

Nedgradering av abonnemang

När kunder vill flytta till en plan av lägre nivå, hantera övergången smidigt med automatisk kredit.

Hur nedgraderingar fungerar

  1. Kunder begär nedgradering (Pro → Basic)
  2. Systemet beräknar återstående värde på aktuell plan
  3. Kredit läggs till abonnemanget för framtida förnyelser
  4. Kunden flyttas omedelbart till den nya planen
async function downgradeSubscription(
  subscriptionId: string,
  newProductId: string
) {
  // Preview the downgrade first
  const preview = await client.subscriptions.previewChangePlan(subscriptionId, {
    product_id: newProductId,
    quantity: 1,
    proration_billing_mode: 'difference_immediately'
  });

  console.log('Credit to be applied:', preview.credit_amount);

  // Execute the downgrade
  const result = await client.subscriptions.changePlan(subscriptionId, {
    product_id: newProductId,
    quantity: 1,
    proration_billing_mode: 'difference_immediately'
  });

  // Credits are automatically applied to future renewals
  return result;
}

// Downgrade from Pro ($80) to Basic ($30)
// $50 credit added to subscription, auto-applied on next renewal
const downgrade = await downgradeSubscription('sub_123', 'prod_basic_plan');
Krediter från nedgraderingar som använder difference_immediately är prenumerationsbundna och tillämpas automatiskt på framtida förnyelser. De skiljer sig från Credit-Based Billing-rättigheter.

Fullständigt exempel: Uppersell-flöde efter köp

Här är en komplett implementering som visar hur man erbjuder en uppersell efter ett lyckat köp:
import DodoPayments from 'dodopayments';
import express from 'express';

const client = new DodoPayments({
  bearerToken: process.env.DODO_PAYMENTS_API_KEY,
  environment: 'live_mode',
});

const app = express();

// Store for tracking upsell eligibility (use your database in production)
const eligibleUpsells = new Map<string, { customerId: string; productId: string }>();

// Webhook handler for initial purchase success
app.post('/webhooks/dodo', express.raw({ type: 'application/json' }), async (req, res) => {
  const event = JSON.parse(req.body.toString());
  
  switch (event.type) {
    case 'payment.succeeded':
      // Check if customer is eligible for upsell
      const customerId = event.data.customer_id;
      const productId = event.data.product_id;
      
      // Define upsell rules (e.g., bought Basic, offer Pro)
      const upsellProduct = getUpsellProduct(productId);
      
      if (upsellProduct) {
        eligibleUpsells.set(customerId, {
          customerId,
          productId: upsellProduct
        });
      }
      break;
      
    case 'payment.failed':
      console.log('Payment failed:', event.data.payment_id);
      // Handle failed upsell payment
      break;
  }
  
  res.json({ received: true });
});

// API endpoint to check upsell eligibility
app.get('/api/upsell/:customerId', async (req, res) => {
  const { customerId } = req.params;
  const upsell = eligibleUpsells.get(customerId);
  
  if (!upsell) {
    return res.json({ eligible: false });
  }
  
  // Get payment methods
  const methods = await client.customers.listPaymentMethods(customerId);
  
  if (methods.length === 0) {
    return res.json({ eligible: false, reason: 'no_payment_method' });
  }
  
  // Get product details for display
  const product = await client.products.retrieve(upsell.productId);
  
  res.json({
    eligible: true,
    product: {
      id: product.product_id,
      name: product.name,
      price: product.price,
      currency: product.currency
    },
    paymentMethodId: methods[0].payment_method_id
  });
});

// API endpoint to accept upsell
app.post('/api/upsell/:customerId/accept', async (req, res) => {
  const { customerId } = req.params;
  const upsell = eligibleUpsells.get(customerId);
  
  if (!upsell) {
    return res.status(400).json({ error: 'No upsell available' });
  }
  
  try {
    const methods = await client.customers.listPaymentMethods(customerId);
    
    // Create one-click purchase
    const session = await client.checkoutSessions.create({
      product_cart: [{ product_id: upsell.productId, quantity: 1 }],
      customer: { customer_id: customerId },
      payment_method_id: methods[0].payment_method_id,
      confirm: true,
      return_url: `${process.env.APP_URL}/upsell-success`,
      feature_flags: { redirect_immediately: true },
      metadata: { upsell: 'true', source: 'post_purchase' }
    });
    
    // Clear the upsell offer
    eligibleUpsells.delete(customerId);
    
    res.json({ success: true, sessionId: session.session_id });
  } catch (error) {
    console.error('Upsell failed:', error);
    res.status(500).json({ error: 'Upsell processing failed' });
  }
});

// Helper function to determine upsell product
function getUpsellProduct(purchasedProductId: string): string | null {
  const upsellMap: Record<string, string> = {
    'prod_basic_plan': 'prod_pro_plan',
    'prod_starter_course': 'prod_complete_bundle',
    'prod_single_license': 'prod_team_license'
  };
  
  return upsellMap[purchasedProductId] || null;
}

app.listen(3000);

Bästa metoder

Den bästa tiden att erbjuda en upsell är omedelbart efter ett lyckat köp när kunderna är i köpläge. Andra effektiva tillfällen:
  • Efter användningsmilstolpar för funktioner
  • När abonnemangsgränser närmar sig
  • Vid avslutning av onboarding
Innan du försöker ta ut en avgift med ett klick, verifiera betalningsmetoden:
  • Är kompatibel med produktens valuta
  • Har inte gått ut
  • Tillhör kunden
API:t kommer att validera detta, men att kontrollera proaktivt förbättrar användarupplevelsen.
När avgifter med ett klick misslyckas:
  1. Gå tillbaka till standardkassaflödet
  2. Meddela kunden med tydlig kommunikation
  3. Erbjud att uppdatera betalningsmetoden
  4. Försök inte upprepade gånger misslyckade avgifter
Upsells konverterar bättre när kunder förstår värdet:
  • Visa vad de får jämfört med nuvarande abonnemang
  • Lyft fram prisdifferensen, inte totalpriset
  • Använd socialt bevis (t.ex. “Mest populära uppgradering”)
  • Ge alltid ett enkelt sätt att tacka nej
  • Visa inte samma upsell upprepade gånger efter att den avvisats
  • Spåra och analysera vilka upsells som konverterar för att optimera erbjudandena

Webhooks för övervakning

Spåra dessa webhook-händelser för uppersell och nedgradering flöden:
HändelseUtlösareÅtgärd
payment.succeededBetalning för upsell/cross-sell slutfördLeverera produkt, uppdatera åtkomst
payment.failedAvgift med ett klick misslyckadesVisa fel, erbjud försök igen eller fallback
subscription.plan_changedUppgradering/nedgradering slutfördUppdatera funktioner, skicka bekräftelse
subscription.activePrenumeration återaktiverad efter abonnemangsändringGe åtkomst till ny nivå

Webhook Integration Guide

Lär dig hur du ställer in och verifierar webhookslutpunkter.

Relaterade resurser