From 2833fde590c3431a5e4ff8b9a34ab73c0fe8b9b0 Mon Sep 17 00:00:00 2001 From: Oleksandr Pavlovskyi Date: Tue, 17 Oct 2023 15:18:36 +0300 Subject: [PATCH] feat: create invoice item (#20) --- libs/stripe/package.json | 2 +- .../src/lib/controllers/invoice.controller.ts | 10 +- .../src/lib/dto/create-invoice-item.dto.ts | 122 ++++++++++++++++++ libs/stripe/src/lib/dto/index.ts | 3 + .../src/lib/dto/stripe/invoice-item.dto.ts | 106 +++++++++++++++ libs/stripe/src/lib/stripe.service.ts | 91 ++++++++++++- 6 files changed, 328 insertions(+), 6 deletions(-) create mode 100644 libs/stripe/src/lib/dto/create-invoice-item.dto.ts create mode 100644 libs/stripe/src/lib/dto/stripe/invoice-item.dto.ts diff --git a/libs/stripe/package.json b/libs/stripe/package.json index 3dd8817..5b83390 100644 --- a/libs/stripe/package.json +++ b/libs/stripe/package.json @@ -1,6 +1,6 @@ { "name": "@valor/nestjs-stripe", - "version": "0.0.15", + "version": "0.0.16", "type": "commonjs", "private": false, "author": "opavlovskyi-valor-software", diff --git a/libs/stripe/src/lib/controllers/invoice.controller.ts b/libs/stripe/src/lib/controllers/invoice.controller.ts index c6f9fe1..53da6f0 100644 --- a/libs/stripe/src/lib/controllers/invoice.controller.ts +++ b/libs/stripe/src/lib/controllers/invoice.controller.ts @@ -8,12 +8,10 @@ import { Param, Get, Query, - Logger, Patch} from '@nestjs/common'; import { ApiBearerAuth, ApiTags, ApiResponse } from '@nestjs/swagger'; import { BaseDataResponse, - BaseResponse, BaseSearchInvoiceDto, InvoiceDto, InvoiceFinalizeInvoiceDto, @@ -77,4 +75,12 @@ export class InvoiceController { return this.stripeService.finalizeInvoice(invoiceId, dto); } + @ApiResponse({ type: BaseDataResponse }) + @Patch(':invoiceId/send') + sendInvoice( + @Param('invoiceId') invoiceId: string, + ): Promise> { + return this.stripeService.sendInvoice(invoiceId); + } + } \ No newline at end of file diff --git a/libs/stripe/src/lib/dto/create-invoice-item.dto.ts b/libs/stripe/src/lib/dto/create-invoice-item.dto.ts new file mode 100644 index 0000000..42971e9 --- /dev/null +++ b/libs/stripe/src/lib/dto/create-invoice-item.dto.ts @@ -0,0 +1,122 @@ +import { ApiProperty, ApiPropertyOptional } from '@nestjs/swagger'; +import { DiscountDto, MetadataDto, PeriodDto } from './shared.dto'; + +type TaxBehavior = 'exclusive' | 'inclusive' | 'unspecified' +const TaxBehaviors = ['exclusive', 'inclusive', 'unspecified']; + +export class InvoiceItemPriceDataDto { + @ApiProperty() + currency: string; + + @ApiProperty() + product: string; + + @ApiPropertyOptional({ + description: 'Only required if a [default tax behavior](https://stripe.com/docs/tax/products-prices-tax-categories-tax-behavior#setting-a-default-tax-behavior-(recommended)) was not provided in the Stripe Tax settings. Specifies whether the price is considered inclusive of taxes or exclusive of taxes. One of `inclusive`, `exclusive`, or `unspecified`. Once specified as either `inclusive` or `exclusive`, it cannot be changed.', + enum: TaxBehaviors + }) + taxBehavior?: TaxBehavior; + + @ApiPropertyOptional({ + description: 'A positive integer in cents (or local equivalent) (or 0 for a free price) representing how much to charge.' + }) + unitAmount?: number; + + @ApiPropertyOptional({ + description: 'Same as `unit_amount`, but accepts a decimal value in cents (or local equivalent) with at most 12 decimal places. Only one of `unit_amount` and `unit_amount_decimal` can be set.' + }) + unitAmountDecimal?: string; +} + +export class CreateInvoiceItemDto { + @ApiProperty() + customer: string; + + @ApiPropertyOptional({ + description: 'The integer amount in cents (or local equivalent) of the charge to be applied to the upcoming invoice. Passing in a negative `amount` will reduce the `amount_due` on the invoice.' + }) + amount?: number; + + @ApiPropertyOptional() + currency?: string; + + @ApiPropertyOptional() + description?: string; + + @ApiPropertyOptional({ + description: 'Controls whether discounts apply to this invoice item. Defaults to false for prorations or negative invoice items, and true for all other invoice items.' + }) + discountable?: boolean; + + @ApiPropertyOptional({ + description: 'The coupons to redeem into discounts for the invoice item or invoice line item.', + isArray: true, + type: DiscountDto + }) + discounts?: DiscountDto[]; + + @ApiPropertyOptional({ + isArray: true, + type: String + }) + expand?: Array; + + @ApiPropertyOptional({ + description: 'The ID of an existing invoice to add this invoice item to. When left blank, the invoice item will be added to the next upcoming scheduled invoice. This is useful when adding invoice items in response to an invoice.created webhook. You can only add invoice items to draft invoices and there is a maximum of 250 items per invoice.' + }) + invoice?: string; + + @ApiPropertyOptional({ + description: 'Set of [key-value pairs](https://stripe.com/docs/api/metadata) that you can attach to an object. This can be useful for storing additional information about the object in a structured format. Individual keys can be unset by posting an empty value to them. All keys can be unset by posting an empty value to `metadata`.' + }) + metadata?: MetadataDto; + + @ApiPropertyOptional({ + description: 'The period associated with this invoice item. When set to different values, the period will be rendered on the invoice. If you have [Stripe Revenue Recognition](https://stripe.com/docs/revenue-recognition) enabled, the period will be used to recognize and defer revenue. See the [Revenue Recognition documentation](https://stripe.com/docs/revenue-recognition/methodology/subscriptions-and-invoicing) for details.' + }) + period?: PeriodDto; + + @ApiPropertyOptional() + price?: string; + + @ApiPropertyOptional() + priceData?: InvoiceItemPriceDataDto; + + @ApiPropertyOptional({ + description: 'Non-negative integer. The quantity of units for the invoice item.' + }) + quantity?: number; + + @ApiPropertyOptional({ + description: 'The ID of a subscription to add this invoice item to. When left blank, the invoice item will be be added to the next upcoming scheduled invoice. When set, scheduled invoices for subscriptions other than the specified subscription will ignore the invoice item. Use this when you want to express that an invoice item has been accrued within the context of a particular subscription.' + }) + subscription?: string; + + @ApiPropertyOptional({ + description: 'Only required if a [default tax behavior](https://stripe.com/docs/tax/products-prices-tax-categories-tax-behavior#setting-a-default-tax-behavior-(recommended)) was not provided in the Stripe Tax settings. Specifies whether the price is considered inclusive of taxes or exclusive of taxes. One of `inclusive`, `exclusive`, or `unspecified`. Once specified as either `inclusive` or `exclusive`, it cannot be changed.', + enum: TaxBehaviors + }) + taxBehavior?: TaxBehavior; + + @ApiPropertyOptional({ + description: 'A [tax code](https://stripe.com/docs/tax/tax-categories) ID.' + }) + taxCode?:string; + + @ApiPropertyOptional({ + description: 'The tax rates which apply to the invoice item. When set, the `default_tax_rates` on the invoice do not apply to this invoice item.', + isArray: true, + type: String + }) + taxRates?: Array; + + @ApiPropertyOptional({ + description: 'The integer unit amount in cents (or local equivalent) of the charge to be applied to the upcoming invoice. This `unit_amount` will be multiplied by the quantity to get the full amount. Passing in a negative `unit_amount` will reduce the `amount_due` on the invoice.' + }) + unitAmount?: number; + + @ApiPropertyOptional({ + description: 'Same as `unit_amount`, but accepts a decimal value in cents (or local equivalent) with at most 12 decimal places. Only one of `unit_amount` and `unit_amount_decimal` can be set.' + }) + unitAmountDecimal?: string; +} diff --git a/libs/stripe/src/lib/dto/index.ts b/libs/stripe/src/lib/dto/index.ts index e61ea12..f22e1df 100644 --- a/libs/stripe/src/lib/dto/index.ts +++ b/libs/stripe/src/lib/dto/index.ts @@ -22,6 +22,8 @@ export * from './update-customer.dto'; export * from './save-webhook-endpoint.dto'; export * from './save-test-clock.dto'; export * from './update-invoice.dto'; +export * from './create-invoice.dto'; +export * from './create-invoice-item.dto'; export * from './stripe/customer.dto'; export * from './stripe/subscription-item.dto'; @@ -29,6 +31,7 @@ export * from './stripe/subscription.dto'; export * from './subscription-schedule.dto'; export * from './stripe/invoice-line-item.dto'; export * from './stripe/invoice.dto'; +export * from './stripe/invoice-item.dto'; export * from './stripe/usage-record.dto'; export * from './stripe/plan.dto'; export * from './stripe/price.dto'; diff --git a/libs/stripe/src/lib/dto/stripe/invoice-item.dto.ts b/libs/stripe/src/lib/dto/stripe/invoice-item.dto.ts new file mode 100644 index 0000000..c3f1806 --- /dev/null +++ b/libs/stripe/src/lib/dto/stripe/invoice-item.dto.ts @@ -0,0 +1,106 @@ +import { ApiPropertyOptional } from '@nestjs/swagger'; +import { BaseDto } from '../base.dto'; +import { CustomerDto } from './customer.dto'; +import { DiscountDto, PeriodDto } from '../shared.dto'; +import { InvoiceDto } from './invoice.dto'; +import { PlanDto } from './plan.dto'; +import { PriceDto } from './price.dto'; + +export class InvoiceItemDto extends BaseDto { + @ApiPropertyOptional({ + description: 'Amount (in the `currency` specified) of the invoice item. This should always be equal to `unit_amount * quantity`.' + }) + amount: number; + + @ApiPropertyOptional() + currency: string; + + @ApiPropertyOptional() + customerId?: string; + + @ApiPropertyOptional() + customer?: CustomerDto; + + @ApiPropertyOptional({ + description: 'Time at which the object was created. Measured in seconds since the Unix epoch.' + }) + date?: Date; + + @ApiPropertyOptional() + description: string; + + @ApiPropertyOptional({ + description: 'If true, discounts will apply to this invoice item. Always false for prorations.' + }) + discountable?: boolean; + + @ApiPropertyOptional({ + isArray: true, + type: String + }) + discountIds?: Array; + + @ApiPropertyOptional({ + description: 'The discounts which apply to the invoice item. Item discounts are applied before invoice discounts. Use `expand[]=discounts` to expand each discount.', + isArray: true, + type: String + }) + discounts?: Array; + + @ApiPropertyOptional() + invoiceId?: string; + + @ApiPropertyOptional() + invoice?: InvoiceDto; + + @ApiPropertyOptional() + period?: PeriodDto; + + @ApiPropertyOptional({ + description: 'If the invoice item is a proration, the plan of the subscription that the proration was computed for.' + }) + plan?: PlanDto; + + @ApiPropertyOptional({ + description: 'The price of the invoice item.' + }) + price?: PriceDto; + + @ApiPropertyOptional({ + description: 'Whether the invoice item was created automatically as a proration adjustment when the customer switched plans.' + }) + proration?: boolean; + + @ApiPropertyOptional({ + description: 'Quantity of units for the invoice item. If the invoice item is a proration, the quantity of the subscription that the proration was computed for.' + }) + quantity?: number; + + @ApiPropertyOptional({ + description: ' The subscription that this invoice item has been created for, if any.' + }) + subscriptionId?: string; + + @ApiPropertyOptional({ + description: 'The subscription item that this invoice item has been created for, if any.' + }) + subscriptionItemId?: string; + + @ApiPropertyOptional({ + description: 'The tax rates which apply to the invoice item. When set, the `default_tax_rates` on the invoice do not apply to this invoice item.', + isArray: true, + type: String + }) + taxRates?: Array; + + @ApiPropertyOptional({ + description: 'The integer unit amount in cents (or local equivalent) of the charge to be applied to the upcoming invoice. This `unit_amount` will be multiplied by the quantity to get the full amount. Passing in a negative `unit_amount` will reduce the `amount_due` on the invoice.' + }) + unitAmount?: number; + + @ApiPropertyOptional({ + description: 'Same as `unit_amount`, but accepts a decimal value in cents (or local equivalent) with at most 12 decimal places. Only one of `unit_amount` and `unit_amount_decimal` can be set.' + }) + unitAmountDecimal?: string; + +} diff --git a/libs/stripe/src/lib/stripe.service.ts b/libs/stripe/src/lib/stripe.service.ts index 6842dfe..093060a 100644 --- a/libs/stripe/src/lib/stripe.service.ts +++ b/libs/stripe/src/lib/stripe.service.ts @@ -61,10 +61,13 @@ import { BaseSearchInvoiceDto, InvoiceVoidInvoiceDto, InvoiceFinalizeInvoiceDto, + CreateInvoiceDto, + CreateInvoiceItemDto, + InvoiceItemDto, + DiscountDto, } from './dto'; import { StripeConfig, STRIPE_CONFIG } from './stripe.config'; import { StripeLogger } from './stripe.logger'; -import { CreateInvoiceDto } from './dto/create-invoice.dto'; @Injectable() export class StripeService { @@ -1058,7 +1061,7 @@ export class StripeService { } } - async voidInvoice(id: string, dto: InvoiceVoidInvoiceDto): Promise> { + async voidInvoice(id: string, dto?: InvoiceVoidInvoiceDto): Promise> { try { const invoice = await this.stripe.invoices.voidInvoice(id, { expand: dto.expand @@ -1072,7 +1075,7 @@ export class StripeService { } } - async finalizeInvoice(id: string, dto: InvoiceFinalizeInvoiceDto): Promise> { + async finalizeInvoice(id: string, dto?: InvoiceFinalizeInvoiceDto): Promise> { try { const invoice = await this.stripe.invoices.finalizeInvoice(id, { auto_advance: dto.autoAdvance, @@ -1086,6 +1089,55 @@ export class StripeService { return this.handleError(exception, 'Finalize Invoice'); } } + + async sendInvoice(id: string): Promise> { + try { + const invoice = await this.stripe.invoices.sendInvoice(id); + return { + success: true, + data: this.invoiceToDto(invoice) + }; + } catch (exception) { + return this.handleError(exception, 'Sand Invoice'); + } + } + + async createInvoiceItem(invoiceId: string, dto: CreateInvoiceItemDto): Promise> { + try { + const invoiceItem = await this.stripe.invoiceItems.create({ + amount: dto.amount, + currency: dto.currency, + customer: dto.customer, + description: dto.description, + discountable: dto.discountable, + discounts: dto.discounts, + expand: dto.expand, + invoice: dto.invoice, + period: dto.period, + price: dto.price, + price_data: dto.priceData ? { + currency: dto.priceData.currency, + product: dto.priceData.product, + tax_behavior: dto.priceData.taxBehavior, + unit_amount: dto.priceData.unitAmount, + unit_amount_decimal: dto.priceData.unitAmountDecimal + } : undefined, + quantity: dto.quantity, + subscription: dto.subscription, + tax_behavior: dto.taxBehavior, + tax_code: dto.taxCode, + tax_rates: dto.taxRates, + unit_amount: dto.unitAmount, + unit_amount_decimal: dto.unitAmountDecimal, + }); + return { + success: true, + data: this.invoiceItemToDto(invoiceItem as Stripe.InvoiceItem), + } + } catch (exception) { + return this.handleError(exception, 'Invoice Item Create'); + } + } //#endregion //#region Quote @@ -1661,6 +1713,39 @@ export class StripeService { } } + private invoiceItemToDto(item: Stripe.InvoiceItem): InvoiceItemDto { + if (!item) { + return item as undefined | null; + } + return { + id: item.id, + object: item.object, + amount: item.amount, + currency: item.currency, + customerId: stripeObjId(item.customer), + customer: this.customerToDto(item.customer as Stripe.Customer), + date: item.date && new Date(item.date * 1000), + description: item.description, + discountable: item.discountable, + discountIds: item.discounts?.map(d => stripeObjId(d)), + discounts: item.discounts?.map(d => d as DiscountDto)?.filter(d => d?.coupon || d?.discount), + liveMode: item.livemode, + metadata: item.metadata, + invoiceId: stripeObjId(item.invoice), + invoice: this.invoiceToDto(item.invoice as Stripe.Invoice), + period: item.period, + plan: item.plan ? this.planToDto(item.plan) : null, + price: item.price ? this.priceToDto(item.price) : null, + proration: item.proration, + quantity: item.quantity, + subscriptionId: stripeObjId(item.subscription), + subscriptionItemId: stripeObjId(item.subscription_item), + taxRates: item.tax_rates?.map(tr => tr as unknown as string), + unitAmount: item.unit_amount, + unitAmountDecimal: item.unit_amount_decimal + } + } + private quoteToDto(quote: Stripe.Quote): QuoteDto { return { id: quote.id,