update price on subscription - command (#11698)
This commit is contained in:
@ -8,6 +8,7 @@ import { BillingResolver } from 'src/engine/core-modules/billing/billing.resolve
|
|||||||
import { BillingAddWorkflowSubscriptionItemCommand } from 'src/engine/core-modules/billing/commands/billing-add-workflow-subscription-item.command';
|
import { BillingAddWorkflowSubscriptionItemCommand } from 'src/engine/core-modules/billing/commands/billing-add-workflow-subscription-item.command';
|
||||||
import { BillingSyncCustomerDataCommand } from 'src/engine/core-modules/billing/commands/billing-sync-customer-data.command';
|
import { BillingSyncCustomerDataCommand } from 'src/engine/core-modules/billing/commands/billing-sync-customer-data.command';
|
||||||
import { BillingSyncPlansDataCommand } from 'src/engine/core-modules/billing/commands/billing-sync-plans-data.command';
|
import { BillingSyncPlansDataCommand } from 'src/engine/core-modules/billing/commands/billing-sync-plans-data.command';
|
||||||
|
import { BillingUpdateSubscriptionPriceCommand } from 'src/engine/core-modules/billing/commands/billing-update-subscription-price.command';
|
||||||
import { BillingCustomer } from 'src/engine/core-modules/billing/entities/billing-customer.entity';
|
import { BillingCustomer } from 'src/engine/core-modules/billing/entities/billing-customer.entity';
|
||||||
import { BillingEntitlement } from 'src/engine/core-modules/billing/entities/billing-entitlement.entity';
|
import { BillingEntitlement } from 'src/engine/core-modules/billing/entities/billing-entitlement.entity';
|
||||||
import { BillingMeter } from 'src/engine/core-modules/billing/entities/billing-meter.entity';
|
import { BillingMeter } from 'src/engine/core-modules/billing/entities/billing-meter.entity';
|
||||||
@ -84,6 +85,7 @@ import { PermissionsModule } from 'src/engine/metadata-modules/permissions/permi
|
|||||||
BillingWebhookCustomerService,
|
BillingWebhookCustomerService,
|
||||||
BillingRestApiExceptionFilter,
|
BillingRestApiExceptionFilter,
|
||||||
BillingSyncCustomerDataCommand,
|
BillingSyncCustomerDataCommand,
|
||||||
|
BillingUpdateSubscriptionPriceCommand,
|
||||||
BillingSyncPlansDataCommand,
|
BillingSyncPlansDataCommand,
|
||||||
BillingAddWorkflowSubscriptionItemCommand,
|
BillingAddWorkflowSubscriptionItemCommand,
|
||||||
BillingUsageService,
|
BillingUsageService,
|
||||||
|
|||||||
@ -0,0 +1,120 @@
|
|||||||
|
/* @license Enterprise */
|
||||||
|
|
||||||
|
import { InjectRepository } from '@nestjs/typeorm';
|
||||||
|
|
||||||
|
import { Command, Option } from 'nest-commander';
|
||||||
|
import { isDefined } from 'twenty-shared/utils';
|
||||||
|
import { Repository } from 'typeorm';
|
||||||
|
|
||||||
|
import {
|
||||||
|
ActiveOrSuspendedWorkspacesMigrationCommandRunner,
|
||||||
|
RunOnWorkspaceArgs,
|
||||||
|
} from 'src/database/commands/command-runners/active-or-suspended-workspaces-migration.command-runner';
|
||||||
|
import {
|
||||||
|
BillingException,
|
||||||
|
BillingExceptionCode,
|
||||||
|
} from 'src/engine/core-modules/billing/billing.exception';
|
||||||
|
import { BillingSubscription } from 'src/engine/core-modules/billing/entities/billing-subscription.entity';
|
||||||
|
import { BillingSubscriptionService } from 'src/engine/core-modules/billing/services/billing-subscription.service';
|
||||||
|
import { StripeSubscriptionItemService } from 'src/engine/core-modules/billing/stripe/services/stripe-subscription-item.service';
|
||||||
|
import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity';
|
||||||
|
import { TwentyORMGlobalManager } from 'src/engine/twenty-orm/twenty-orm-global.manager';
|
||||||
|
|
||||||
|
@Command({
|
||||||
|
name: 'billing:update-subscription-price',
|
||||||
|
description: 'Update subscription price',
|
||||||
|
})
|
||||||
|
export class BillingUpdateSubscriptionPriceCommand extends ActiveOrSuspendedWorkspacesMigrationCommandRunner {
|
||||||
|
private stripePriceIdToUpdate: string;
|
||||||
|
private newStripePriceId: string;
|
||||||
|
private clearUsage = false;
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
@InjectRepository(Workspace, 'core')
|
||||||
|
protected readonly workspaceRepository: Repository<Workspace>,
|
||||||
|
protected readonly twentyORMGlobalManager: TwentyORMGlobalManager,
|
||||||
|
@InjectRepository(BillingSubscription, 'core')
|
||||||
|
protected readonly billingSubscriptionRepository: Repository<BillingSubscription>,
|
||||||
|
private readonly billingSubscriptionService: BillingSubscriptionService,
|
||||||
|
private readonly stripeSubscriptionItemService: StripeSubscriptionItemService,
|
||||||
|
) {
|
||||||
|
super(workspaceRepository, twentyORMGlobalManager);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Option({
|
||||||
|
flags: '--price-to-update-id [stripe_price_id]',
|
||||||
|
description: 'Stripe price id to update',
|
||||||
|
required: true,
|
||||||
|
})
|
||||||
|
parseStripePriceIdToMigrate(val: string): string {
|
||||||
|
this.stripePriceIdToUpdate = val;
|
||||||
|
|
||||||
|
return val;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Option({
|
||||||
|
flags: '--new-price-id [stripe_price_id]',
|
||||||
|
description: 'New Stripe price id',
|
||||||
|
required: true,
|
||||||
|
})
|
||||||
|
parseNewStripePriceId(val: string): string {
|
||||||
|
this.newStripePriceId = val;
|
||||||
|
|
||||||
|
return val;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Option({
|
||||||
|
flags: '--clear-usage',
|
||||||
|
description: 'Clear usage on subscription item',
|
||||||
|
required: false,
|
||||||
|
})
|
||||||
|
parseClearUsage() {
|
||||||
|
this.clearUsage = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
override async runOnWorkspace({
|
||||||
|
workspaceId,
|
||||||
|
options,
|
||||||
|
}: RunOnWorkspaceArgs): Promise<void> {
|
||||||
|
const subscription =
|
||||||
|
await this.billingSubscriptionService.getCurrentBillingSubscriptionOrThrow(
|
||||||
|
{ workspaceId },
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!isDefined(subscription)) {
|
||||||
|
throw new BillingException(
|
||||||
|
`No subscription found for workspace ${workspaceId}`,
|
||||||
|
BillingExceptionCode.BILLING_SUBSCRIPTION_NOT_FOUND,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const subscriptionItemToUpdate = subscription.billingSubscriptionItems.find(
|
||||||
|
(item) => item.stripePriceId === this.stripePriceIdToUpdate,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!isDefined(subscriptionItemToUpdate)) {
|
||||||
|
this.logger.log(`No price to update for workspace ${workspaceId}`);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!options.dryRun) {
|
||||||
|
await this.stripeSubscriptionItemService.deleteSubscriptionItem(
|
||||||
|
subscriptionItemToUpdate.stripeSubscriptionItemId,
|
||||||
|
this.clearUsage,
|
||||||
|
);
|
||||||
|
|
||||||
|
await this.stripeSubscriptionItemService.createSubscriptionItem(
|
||||||
|
subscription.stripeSubscriptionId,
|
||||||
|
this.newStripePriceId,
|
||||||
|
isDefined(subscriptionItemToUpdate.quantity)
|
||||||
|
? subscriptionItemToUpdate.quantity
|
||||||
|
: undefined,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.logger.log(
|
||||||
|
`Update subscription replacing price ${subscriptionItemToUpdate.stripePriceId} by ${this.newStripePriceId} with clear usage ${this.clearUsage} - workspace ${workspaceId}`,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -3,6 +3,7 @@
|
|||||||
import { Injectable, Logger } from '@nestjs/common';
|
import { Injectable, Logger } from '@nestjs/common';
|
||||||
|
|
||||||
import Stripe from 'stripe';
|
import Stripe from 'stripe';
|
||||||
|
import { isDefined } from 'twenty-shared/utils';
|
||||||
|
|
||||||
import { StripeSDKService } from 'src/engine/core-modules/billing/stripe/stripe-sdk/services/stripe-sdk.service';
|
import { StripeSDKService } from 'src/engine/core-modules/billing/stripe/stripe-sdk/services/stripe-sdk.service';
|
||||||
import { TwentyConfigService } from 'src/engine/core-modules/twenty-config/twenty-config.service';
|
import { TwentyConfigService } from 'src/engine/core-modules/twenty-config/twenty-config.service';
|
||||||
@ -34,10 +35,18 @@ export class StripeSubscriptionItemService {
|
|||||||
async createSubscriptionItem(
|
async createSubscriptionItem(
|
||||||
stripeSubscriptionId: string,
|
stripeSubscriptionId: string,
|
||||||
stripePriceId: string,
|
stripePriceId: string,
|
||||||
|
quantity?: number,
|
||||||
) {
|
) {
|
||||||
return this.stripe.subscriptionItems.create({
|
return this.stripe.subscriptionItems.create({
|
||||||
subscription: stripeSubscriptionId,
|
subscription: stripeSubscriptionId,
|
||||||
price: stripePriceId,
|
price: stripePriceId,
|
||||||
|
...(isDefined(quantity) ? { quantity } : {}),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async deleteSubscriptionItem(stripeItemId: string, clearUsage = false) {
|
||||||
|
return this.stripe.subscriptionItems.del(stripeItemId, {
|
||||||
|
clear_usage: clearUsage,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user