fluxer/packages/api/src/stripe/services/StripeSubscriptionService.tsx
2026-02-17 12:22:36 +00:00

208 lines
7.1 KiB
TypeScript

/*
* Copyright (C) 2026 Fluxer Contributors
*
* This file is part of Fluxer.
*
* Fluxer is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Fluxer is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with Fluxer. If not, see <https://www.gnu.org/licenses/>.
*/
import type {UserID} from '@fluxer/api/src/BrandedTypes';
import type {IGatewayService} from '@fluxer/api/src/infrastructure/IGatewayService';
import {Logger} from '@fluxer/api/src/Logger';
import type {User} from '@fluxer/api/src/models/User';
import {
getPrimarySubscriptionItem,
getSubscriptionItemPeriodEndUnix,
} from '@fluxer/api/src/stripe/StripeSubscriptionPeriod';
import {addMonthsClamp} from '@fluxer/api/src/stripe/StripeUtils';
import type {IUserRepository} from '@fluxer/api/src/user/IUserRepository';
import {mapUserToPrivateResponse} from '@fluxer/api/src/user/UserMappers';
import type {ICacheService} from '@fluxer/cache/src/ICacheService';
import {NoActiveSubscriptionError} from '@fluxer/errors/src/domains/payment/NoActiveSubscriptionError';
import {StripeError} from '@fluxer/errors/src/domains/payment/StripeError';
import {StripeNoActiveSubscriptionError} from '@fluxer/errors/src/domains/payment/StripeNoActiveSubscriptionError';
import {StripeNoSubscriptionError} from '@fluxer/errors/src/domains/payment/StripeNoSubscriptionError';
import {StripePaymentNotAvailableError} from '@fluxer/errors/src/domains/payment/StripePaymentNotAvailableError';
import {StripeSubscriptionAlreadyCancelingError} from '@fluxer/errors/src/domains/payment/StripeSubscriptionAlreadyCancelingError';
import {StripeSubscriptionNotCancelingError} from '@fluxer/errors/src/domains/payment/StripeSubscriptionNotCancelingError';
import {UnknownUserError} from '@fluxer/errors/src/domains/user/UnknownUserError';
import {seconds} from 'itty-time';
import type Stripe from 'stripe';
export class StripeSubscriptionService {
constructor(
private stripe: Stripe | null,
private userRepository: IUserRepository,
private cacheService: ICacheService,
private gatewayService: IGatewayService,
) {}
async cancelSubscriptionAtPeriodEnd(userId: UserID): Promise<void> {
if (!this.stripe) {
throw new StripePaymentNotAvailableError();
}
const user = await this.userRepository.findUnique(userId);
if (!user) {
throw new UnknownUserError();
}
if (!user.stripeSubscriptionId) {
throw new StripeNoActiveSubscriptionError();
}
if (user.premiumWillCancel) {
throw new StripeSubscriptionAlreadyCancelingError();
}
try {
await this.stripe.subscriptions.update(user.stripeSubscriptionId, {
cancel_at_period_end: true,
});
const updatedUser = await this.userRepository.patchUpsert(
userId,
{
premium_will_cancel: true,
},
user.toRow(),
);
await this.dispatchUser(updatedUser);
Logger.debug({userId, subscriptionId: user.stripeSubscriptionId}, 'Subscription set to cancel at period end');
} catch (error: unknown) {
Logger.error(
{error, userId, subscriptionId: user.stripeSubscriptionId},
'Failed to cancel subscription at period end',
);
const message = error instanceof Error ? error.message : 'Failed to cancel subscription';
throw new StripeError(message);
}
}
async reactivateSubscription(userId: UserID): Promise<void> {
if (!this.stripe) {
throw new StripePaymentNotAvailableError();
}
const user = await this.userRepository.findUnique(userId);
if (!user) {
throw new UnknownUserError();
}
if (!user.stripeSubscriptionId) {
throw new StripeNoSubscriptionError();
}
if (!user.premiumWillCancel) {
throw new StripeSubscriptionNotCancelingError();
}
try {
await this.stripe.subscriptions.update(user.stripeSubscriptionId, {
cancel_at_period_end: false,
});
const updatedUser = await this.userRepository.patchUpsert(
userId,
{
premium_will_cancel: false,
},
user.toRow(),
);
await this.dispatchUser(updatedUser);
Logger.debug({userId, subscriptionId: user.stripeSubscriptionId}, 'Subscription reactivated');
} catch (error: unknown) {
Logger.error({error, userId, subscriptionId: user.stripeSubscriptionId}, 'Failed to reactivate subscription');
const message = error instanceof Error ? error.message : 'Failed to reactivate subscription';
throw new StripeError(message);
}
}
async extendSubscriptionWithGiftTrial(user: User, durationMonths: number, idempotencyKey: string): Promise<void> {
if (!this.stripe || !user.stripeSubscriptionId) {
throw new NoActiveSubscriptionError();
}
const appliedKey = `gift_trial_applied:${user.id}:${idempotencyKey}`;
const inflightKey = `gift_trial_inflight:${user.id}:${idempotencyKey}`;
if (await this.cacheService.get<boolean>(appliedKey)) {
Logger.debug({userId: user.id, idempotencyKey}, 'Gift trial extension already applied (idempotent hit)');
return;
}
if (await this.cacheService.get<boolean>(inflightKey)) {
Logger.debug({userId: user.id, idempotencyKey}, 'Gift trial extension in-flight; skipping duplicate');
return;
}
await this.cacheService.set(inflightKey, seconds('1 minute'));
try {
const subscription = await this.stripe.subscriptions.retrieve(user.stripeSubscriptionId);
const currentTrialEnd = subscription.trial_end;
const item = getPrimarySubscriptionItem(subscription);
const currentPeriodEnd = getSubscriptionItemPeriodEndUnix(item);
const baseUnix = currentTrialEnd ?? currentPeriodEnd;
if (!baseUnix) {
throw new StripeError('Subscription has no trial_end or current_period_end');
}
const baseDate = new Date(baseUnix * 1000);
const newTrialEnd = addMonthsClamp(baseDate, durationMonths);
const newTrialEndUnix = Math.floor(newTrialEnd.getTime() / 1000);
await this.stripe.subscriptions.update(user.stripeSubscriptionId, {
trial_end: newTrialEndUnix,
proration_behavior: 'none',
});
await this.cacheService.set(appliedKey, seconds('365 days'));
Logger.debug(
{
userId: user.id,
subscriptionId: user.stripeSubscriptionId,
baseDate,
newTrialEnd,
durationMonths,
idempotencyKey,
},
'Extended subscription with gift trial period',
);
} catch (error: unknown) {
Logger.error(
{error, userId: user.id, subscriptionId: user.stripeSubscriptionId, idempotencyKey},
'Failed to extend subscription with gift trial',
);
const message = error instanceof Error ? error.message : 'Failed to extend subscription with gift';
throw new StripeError(message);
} finally {
await this.cacheService.delete(inflightKey);
}
}
private async dispatchUser(user: User): Promise<void> {
await this.gatewayService.dispatchPresence({
userId: user.id,
event: 'USER_UPDATE',
data: mapUserToPrivateResponse(user),
});
}
}