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" }),
})
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 }
}
)
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 }
}
)
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,
}
}
)
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
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 }
}
)
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
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:
- Discount annual plans, not monthly — annual commitment protects your revenue
- Cap discounts at 30-35% — higher discounts devalue the product
- Create urgency — limited-time offers convert better than permanent discounts
- Track everything — UTM, coupon codes, conversion analytics, retention rates
- 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.
Top comments (0)