using Bit.Core.Billing.Caches; using Bit.Core.Billing.Constants; using Bit.Core.Billing.Models; using Bit.Core.Billing.Models.Sales; using Bit.Core.Entities; using Bit.Core.Enums; using Bit.Core.Exceptions; using Bit.Core.Repositories; using Bit.Core.Services; using Bit.Core.Settings; using Braintree; using Microsoft.Extensions.Logging; using Stripe; using Customer = Stripe.Customer; using Subscription = Stripe.Subscription; namespace Bit.Core.Billing.Services.Implementations; using static Utilities; public class PremiumUserBillingService( IBraintreeGateway braintreeGateway, IGlobalSettings globalSettings, ILogger logger, ISetupIntentCache setupIntentCache, IStripeAdapter stripeAdapter, ISubscriberService subscriberService, IUserRepository userRepository) : IPremiumUserBillingService { public async Task Credit(User user, decimal amount) { var customer = await subscriberService.GetCustomer(user); // Negative credit represents a balance and all Stripe denomination is in cents. var credit = (long)(amount * -100); if (customer == null) { var options = new CustomerCreateOptions { Balance = credit, Description = user.Name, Email = user.Email, InvoiceSettings = new CustomerInvoiceSettingsOptions { CustomFields = [ new CustomerInvoiceSettingsCustomFieldOptions { Name = user.SubscriberType(), Value = user.SubscriberName().Length <= 30 ? user.SubscriberName() : user.SubscriberName()[..30] } ] }, Metadata = new Dictionary { ["region"] = globalSettings.BaseServiceUri.CloudRegion, ["userId"] = user.Id.ToString() } }; customer = await stripeAdapter.CustomerCreateAsync(options); user.Gateway = GatewayType.Stripe; user.GatewayCustomerId = customer.Id; await userRepository.ReplaceAsync(user); } else { var options = new CustomerUpdateOptions { Balance = customer.Balance + credit }; await stripeAdapter.CustomerUpdateAsync(customer.Id, options); } } public async Task Finalize(PremiumUserSale sale) { var (user, customerSetup, storage) = sale; List expand = ["tax"]; var customer = string.IsNullOrEmpty(user.GatewayCustomerId) ? await CreateCustomerAsync(user, customerSetup) : await subscriberService.GetCustomerOrThrow(user, new CustomerGetOptions { Expand = expand }); /* * If the customer was previously set up with credit, which does not require a billing location, * we need to update the customer on the fly before we start the subscription. */ customer = await ReconcileBillingLocationAsync(customer, customerSetup.TaxInformation); var subscription = await CreateSubscriptionAsync(user.Id, customer, storage); switch (customerSetup.TokenizedPaymentSource) { case { Type: PaymentMethodType.PayPal } when subscription.Status == StripeConstants.SubscriptionStatus.Incomplete: case { Type: not PaymentMethodType.PayPal } when subscription.Status == StripeConstants.SubscriptionStatus.Active: { user.Premium = true; user.PremiumExpirationDate = subscription.CurrentPeriodEnd; break; } } user.Gateway = GatewayType.Stripe; user.GatewayCustomerId = customer.Id; user.GatewaySubscriptionId = subscription.Id; await userRepository.ReplaceAsync(user); } public async Task UpdatePaymentMethod( User user, TokenizedPaymentSource tokenizedPaymentSource, TaxInformation taxInformation) { if (string.IsNullOrEmpty(user.GatewayCustomerId)) { var customer = await CreateCustomerAsync(user, new CustomerSetup { TokenizedPaymentSource = tokenizedPaymentSource, TaxInformation = taxInformation }); user.Gateway = GatewayType.Stripe; user.GatewayCustomerId = customer.Id; await userRepository.ReplaceAsync(user); } else { await subscriberService.UpdatePaymentSource(user, tokenizedPaymentSource); await subscriberService.UpdateTaxInformation(user, taxInformation); } } private async Task CreateCustomerAsync( User user, CustomerSetup customerSetup) { /* * Creating a Customer via the adding of a payment method or the purchasing of a subscription requires * an actual payment source. The only time this is not the case is when the Customer is created when the * User purchases credit. */ if (customerSetup.TokenizedPaymentSource is not { Type: PaymentMethodType.BankAccount or PaymentMethodType.Card or PaymentMethodType.PayPal, Token: not null and not "" }) { logger.LogError( "Cannot create customer for user ({UserID}) without a valid payment source", user.Id); throw new BillingException(); } if (customerSetup.TaxInformation is not { Country: not null and not "", PostalCode: not null and not "" }) { logger.LogError( "Cannot create customer for user ({UserID}) without valid tax information", user.Id); throw new BillingException(); } var subscriberName = user.SubscriberName(); var customerCreateOptions = new CustomerCreateOptions { Address = new AddressOptions { Line1 = customerSetup.TaxInformation.Line1, Line2 = customerSetup.TaxInformation.Line2, City = customerSetup.TaxInformation.City, PostalCode = customerSetup.TaxInformation.PostalCode, State = customerSetup.TaxInformation.State, Country = customerSetup.TaxInformation.Country, }, Description = user.Name, Email = user.Email, Expand = ["tax"], InvoiceSettings = new CustomerInvoiceSettingsOptions { CustomFields = [ new CustomerInvoiceSettingsCustomFieldOptions { Name = user.SubscriberType(), Value = subscriberName.Length <= 30 ? subscriberName : subscriberName[..30] } ] }, Metadata = new Dictionary { ["region"] = globalSettings.BaseServiceUri.CloudRegion, ["userId"] = user.Id.ToString() }, Tax = new CustomerTaxOptions { ValidateLocation = StripeConstants.ValidateTaxLocationTiming.Immediately } }; var (paymentMethodType, paymentMethodToken) = customerSetup.TokenizedPaymentSource; var braintreeCustomerId = ""; // ReSharper disable once SwitchStatementHandlesSomeKnownEnumValuesWithDefault switch (paymentMethodType) { case PaymentMethodType.BankAccount: { var setupIntent = (await stripeAdapter.SetupIntentList(new SetupIntentListOptions { PaymentMethod = paymentMethodToken })) .FirstOrDefault(); if (setupIntent == null) { logger.LogError("Cannot create customer for user ({UserID}) without a setup intent for their bank account", user.Id); throw new BillingException(); } await setupIntentCache.Set(user.Id, setupIntent.Id); break; } case PaymentMethodType.Card: { customerCreateOptions.PaymentMethod = paymentMethodToken; customerCreateOptions.InvoiceSettings.DefaultPaymentMethod = paymentMethodToken; break; } case PaymentMethodType.PayPal: { braintreeCustomerId = await subscriberService.CreateBraintreeCustomer(user, paymentMethodToken); customerCreateOptions.Metadata[BraintreeCustomerIdKey] = braintreeCustomerId; break; } default: { logger.LogError("Cannot create customer for user ({UserID}) using payment method type ({PaymentMethodType}) as it is not supported", user.Id, paymentMethodType.ToString()); throw new BillingException(); } } try { return await stripeAdapter.CustomerCreateAsync(customerCreateOptions); } catch (StripeException stripeException) when (stripeException.StripeError?.Code == StripeConstants.ErrorCodes.CustomerTaxLocationInvalid) { await Revert(); throw new BadRequestException( "Your location wasn't recognized. Please ensure your country and postal code are valid."); } catch (StripeException stripeException) when (stripeException.StripeError?.Code == StripeConstants.ErrorCodes.TaxIdInvalid) { await Revert(); throw new BadRequestException( "Your tax ID wasn't recognized for your selected country. Please ensure your country and tax ID are valid."); } catch { await Revert(); throw; } async Task Revert() { // ReSharper disable once SwitchStatementMissingSomeEnumCasesNoDefault switch (customerSetup.TokenizedPaymentSource!.Type) { case PaymentMethodType.BankAccount: { await setupIntentCache.Remove(user.Id); break; } case PaymentMethodType.PayPal when !string.IsNullOrEmpty(braintreeCustomerId): { await braintreeGateway.Customer.DeleteAsync(braintreeCustomerId); break; } } } } private async Task CreateSubscriptionAsync( Guid userId, Customer customer, int? storage) { var subscriptionItemOptionsList = new List { new () { Price = "premium-annually", Quantity = 1 } }; if (storage is > 0) { subscriptionItemOptionsList.Add(new SubscriptionItemOptions { Price = "storage-gb-annually", Quantity = storage }); } var usingPayPal = customer.Metadata?.ContainsKey(BraintreeCustomerIdKey) ?? false; var subscriptionCreateOptions = new SubscriptionCreateOptions { AutomaticTax = new SubscriptionAutomaticTaxOptions { Enabled = customer.Tax?.AutomaticTax == StripeConstants.AutomaticTaxStatus.Supported, }, CollectionMethod = StripeConstants.CollectionMethod.ChargeAutomatically, Customer = customer.Id, Items = subscriptionItemOptionsList, Metadata = new Dictionary { ["userId"] = userId.ToString() }, PaymentBehavior = usingPayPal ? StripeConstants.PaymentBehavior.DefaultIncomplete : null, OffSession = true }; var subscription = await stripeAdapter.SubscriptionCreateAsync(subscriptionCreateOptions); if (usingPayPal) { await stripeAdapter.InvoiceUpdateAsync(subscription.LatestInvoiceId, new InvoiceUpdateOptions { AutoAdvance = false }); } return subscription; } private async Task ReconcileBillingLocationAsync( Customer customer, TaxInformation taxInformation) { if (customer is { Address: { Country: not null and not "", PostalCode: not null and not "" } }) { return customer; } var options = new CustomerUpdateOptions { Address = new AddressOptions { Line1 = taxInformation.Line1, Line2 = taxInformation.Line2, City = taxInformation.City, PostalCode = taxInformation.PostalCode, State = taxInformation.State, Country = taxInformation.Country, }, Expand = ["tax"], Tax = new CustomerTaxOptions { ValidateLocation = StripeConstants.ValidateTaxLocationTiming.Immediately } }; return await stripeAdapter.CustomerUpdateAsync(customer.Id, options); } }