DEV Community

sweet
sweet

Posted on

Holiday and Seasonal Promotions: A SaaS Discount Strategy That Works

Discounting is a double-edged sword for SaaS. Done well, holiday promotions can accelerate annual plan adoption, reactivate churned users, and drive seasonal spikes. Done poorly, they train users to wait for discounts and devalue your product. This guide covers a structured approach to holiday promotions — which holidays to target, what discount structures work, how to manage campaign timing, and the database infrastructure you need to run them. See production campaign management at tanstackship.com.


The SaaS Discounting Dilemma

Discount Strategy Risk Reward Best For
20-30% off annual plans Low (annual commitment offsets discount) High Black Friday, New Year
Free month Medium (user may cancel after) Medium Trial conversion push
Lifetime discount (early adopter) High (permanent revenue reduction) Very high Launch
2-for-1 annual Medium High End-of-year push
First 3 months 50% off Medium (churn after discount) Medium Q1 slumps

Key principle: Discount annual plans, not monthly. An annual discount locks in revenue for 12 months. A monthly discount creates churn risk when the promotional period ends.


Holiday Calendar by Market

Major SaaS Promotion Windows

Holiday Date Market SaaS Opportunity
New Year / Q1 Push Jan 1-31 Global Annual plan adoption
Valentine's Day Feb 14 US, EU "Love your business" campaign
Spring Sale March-April Global New year, new stack
Tax Season April US Spending mindset
Summer Slump June-August Global "Get ahead before fall"
Back to School Sept US, EU Productivity push
Singles' Day (11.11) Nov 11 China Biggest shopping day globally
Black Friday / Cyber Monday Nov-Dec Global The biggest SaaS promo window
Christmas / Year-End Dec 15-31 Global Last-chance annual deals

Campaign Structure

Campaign Database Schema

// D1 schema for campaign management
export const campaigns = sqliteTable("campaigns", {
  id: text("id").primaryKey(),
  name: text("name").notNull(),
  slug: text("slug").notNull().unique(),
  type: text("type", {
    enum: ["holiday", "seasonal", "launch", "retention", "reactivation"],
  }).notNull(),
  discountPercent: real("discount_percent").notNull(),
  discountType: text("discount_type", {
    enum: ["percentage", "fixed", "free_month", "tiered"],
  }).notNull(),
  appliesTo: text("applies_to", {
    enum: ["annual", "monthly", "all_plans", "specific_plans"],
  }).notNull(),
  specificPlans: text("specific_plans"), // JSON array of plan IDs
  startDate: integer("start_date", { mode: "timestamp" }).notNull(),
  endDate: integer("end_date", { mode: "timestamp" }).notNull(),
  maxRedemptions: integer("max_redemptions"),
  currentRedemptions: integer("current_redemptions").default(0),
  isActive: integer("is_active", { mode: "boolean" }).default(true),
  couponCode: text("coupon_code"),
  metadata: text("metadata"), // JSON for campaign assets
  createdAt: integer("created_at", { mode: "timestamp" }),
})
Enter fullscreen mode Exit fullscreen mode

Creating a Campaign

// server/campaigns.ts
export const createCampaign = createServerFn({ method: "POST" }).handler(
  async ({ data, context }: { data: CreateCampaignInput }) => {
    const id = crypto.randomUUID()

    // Create Stripe coupon
    const coupon = await stripe.coupons.create({
      percent_off: data.discountPercent,
      duration: "once",
      max_redemptions: data.maxRedemptions,
      applies_to: {
        products: data.specificPlans.length > 0
          ? data.specificPlans
          : undefined,
      },
    })

    await context.env.DB.prepare(`
      INSERT INTO campaigns
        (id, name, slug, type, discount_percent, discount_type,
         applies_to, start_date, end_date, max_redemptions, coupon_code)
      VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
    `).bind(
      id, data.name, data.slug, data.type,
      data.discountPercent, data.discountType,
      data.appliesTo, data.startDate.getTime(),
      data.endDate.getTime(), data.maxRedemptions,
      coupon.id
    ).run()

    return { id, couponId: coupon.id }
  }
)
Enter fullscreen mode Exit fullscreen mode

Black Friday Campaign Example

Pre-Launch (2 Weeks Before)

// Prepare campaign assets and notifications
export const prepareCampaign = createServerFn({ method: "GET" }).handler(
  async ({}, { context }) => {
    const upcoming = await context.env.DB.prepare(`
      SELECT * FROM campaigns
      WHERE start_date > ? AND start_date < ?
      AND is_active = 1
    `).bind(Date.now(), Date.now() + 14 * 24 * 60 * 60 * 1000).all()

    for (const campaign of upcoming.results) {
      // Send announcement emails to subscribers
      await sendCampaignAnnouncement(campaign)

      // Update pricing page to show upcoming sale
      // Update in KV for zero-downtime
      await context.env.KV.put("campaign:upcoming", JSON.stringify({
        name: campaign.name,
        discountPercent: campaign.discount_percent,
        startDate: campaign.start_date,
        slug: campaign.slug,
      }))
    }

    return { prepared: true, count: upcoming.results.length }
  }
)
Enter fullscreen mode Exit fullscreen mode

Live Campaign

// Apply discount during checkout
export const applyCampaignDiscount = createServerFn({ method: "POST" }).handler(
  async ({ data, context }: {
    data: { campaignSlug: string; priceId: string }
  }) => {
    const campaign = await context.env.DB.prepare(`
      SELECT * FROM campaigns
      WHERE slug = ? AND is_active = 1
      AND start_date <= ? AND end_date >= ?
    `).bind(data.campaignSlug, Date.now(), Date.now()).first()

    if (!campaign) {
      return { valid: false, error: "Campaign not active" }
    }

    // Check redemption limit
    if (campaign.max_redemptions && campaign.current_redemptions >= campaign.max_redemptions) {
      return { valid: false, error: "Campaign fully redeemed" }
    }

    // Increment counter
    await context.env.DB.prepare(`
      UPDATE campaigns SET current_redemptions = current_redemptions + 1
      WHERE id = ?
    `).bind(campaign.id).run()

    return {
      valid: true,
      couponCode: campaign.coupon_code,
      discountPercent: campaign.discount_percent,
    }
  }
)
Enter fullscreen mode Exit fullscreen mode

Promotion Types: Which to Use

Type Conversion Rate Revenue Impact Best Timing
% off annual +30-50% High (annual lock-in) Black Friday, New Year
Free months +20-30% Medium Launch, reactivation
Tiered discount +25-40% Medium-High Extended periods
Limited time +40-60% High (urgent) 24-48 hour flash sales
BOGO (buy org, get 1 free) +15-20% Medium Team/Org plans
Bundle discount +20-25% Medium Feature add-ons

Percentage Off vs Free Months

Percentage off annual plan:
  Normal: $29/mo × 12 = $348/year
  With 30% off: $243.60/year (saves $104.40)
  ✅ User commits for 12 months
  ✅ You get cash upfront
  ✅ Lower churn risk

Free months on monthly plan:
  Normal: $29/mo × 12 = $348/year
  With 2 months free: $29/mo × 10 = $290/year (saves $58)
  ❌ User can cancel any time
  ❌ Revenue is deferred
  ❌ Higher churn risk

Winner: Percentage off annual plan
Enter fullscreen mode Exit fullscreen mode

Analytics and Attribution

// Track campaign performance
export const getCampaignAnalytics = createServerFn({ method: "GET" }).handler(
  async ({ data, context }: { data: { campaignId: string } }) => {
    const [campaign, conversions, revenue] = await Promise.all([
      context.env.DB.prepare("SELECT * FROM campaigns WHERE id = ?")
        .bind(data.campaignId).first(),
      context.env.DB.prepare(`
        SELECT COUNT(*) as conversions, SUM(mrr) as total_mrr
        FROM subscriptions
        WHERE coupon_id = (SELECT coupon_code FROM campaigns WHERE id = ?)
        AND created_at BETWEEN
          (SELECT start_date FROM campaigns WHERE id = ?) AND
          (SELECT end_date FROM campaigns WHERE id = ?)
      `).bind(data.campaignId, data.campaignId, data.campaignId).first(),
      context.env.DB.prepare(`
        SELECT SUM(amount) as total_revenue
        FROM invoices
        WHERE subscription_id IN (
          SELECT id FROM subscriptions
          WHERE coupon_id = (SELECT coupon_code FROM campaigns WHERE id = ?)
        )
      `).bind(data.campaignId).first(),
    ])

    return { campaign, conversions, revenue }
  }
)
Enter fullscreen mode Exit fullscreen mode

Anti-Discounting Patterns

Behavior Problem Solution
Always running a promotion Users wait for discounts Create clear promotion windows
Discounting monthly plans High churn after promotion Discount annual only
50%+ discounts Devalues the product Cap at 30-40%
No expiration No urgency Limited time only
Discounting core features Trains users to value features less Discount commitment, not features

Campaign Calendar Template

Q1 (Jan-Mar): New Year Annual Plan Push
  - "New Year, New Stack" — 25% off annual
  - Target: Free users on monthly, upgrade to annual

Q2 (Apr-Jun): Spring Refresh
  - "Spring Cleaning" — 20% off annual
  - Target: Inactive users reactivation

Q3 (Jul-Sep): Back to Business
  - "Back to School for Your Business" — 20% off team plans
  - Target: Team/org plan upgrades

Q4 (Oct-Dec): Black Friday + Year-End
  - Pre-BF: "Early Black Friday" — 30% off annual (limited quantity)
  - BF/CM: "Black Friday Deal" — 35% off annual
  - Year-End: "Last Chance 2026" — 25% off annual
  - Target: All segments, maximum conversion
Enter fullscreen mode Exit fullscreen mode

Promotion Management Checklist

  • [ ] Campaign start/end dates set with proper timezone handling (UTC)
  • [ ] Stripe coupon created with redemption limits
  • [ ] Pricing page dynamically shows/hides promotional pricing
  • [ ] Email sequences ready: announcement, reminder, last chance, expired
  • [ ] UTM parameters set for all campaign links
  • [ ] Analytics dashboard tracking: impressions, clicks, conversions, revenue
  • [ ] Post-campaign retention analysis (do promo users retain at normal rates?)
  • [ ] Expired campaigns automatically hidden from checkout
  • [ ] Customer support briefed on campaign details
  • [ ] Refund policy adjusted for promotional purchases

Conclusion

Holiday promotions are a powerful growth lever for SaaS — when executed strategically. The key principles are:

  1. Discount annual plans, not monthly — annual commitment protects your revenue
  2. Cap discounts at 30-35% — higher discounts devalue the product
  3. Create urgency — limited-time offers convert better than permanent discounts
  4. Track everything — UTM, coupon codes, conversion analytics, retention rates
  5. Plan a calendar — know your promotion windows months in advance

The campaign management infrastructure — database schema, Stripe integration, analytics tracking — should be built once and reused for every promotion. With the right infrastructure, a holiday campaign is a matter of configuration, not engineering.

For a SaaS with built-in campaign management, UTM tracking, and promotion infrastructure, see tanstackship.com.

Related Resources

Top comments (0)