mirror of
https://github.com/bitwarden/server.git
synced 2025-04-04 20:50:21 -05:00

* Pm 19147 2 (#5544) * Pm 19147 2 (#5544) * Unit tests for tax strategies `GetUpdateOptions` * Only allow automatic tax flag to be updated for complete subscription updates such as plan changes, not when upgrading additional storage, seats, etc * unit tests for factory * Fix build * Automatic tax for tax estimation * Fix stub * Fix stub * "customer.tax_ids" isn't expanded in some flows. * Fix SubscriberServiceTests.cs * BusinessUseAutomaticTaxStrategy > SetUpdateOptions tests * Fix ProviderBillingServiceTests.cs
393 lines
14 KiB
C#
393 lines
14 KiB
C#
using Bit.Core.Billing.Caches;
|
|
using Bit.Core.Billing.Constants;
|
|
using Bit.Core.Billing.Models;
|
|
using Bit.Core.Billing.Models.Sales;
|
|
using Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
|
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.DependencyInjection;
|
|
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,
|
|
IFeatureService featureService,
|
|
IGlobalSettings globalSettings,
|
|
ILogger<PremiumUserBillingService> logger,
|
|
ISetupIntentCache setupIntentCache,
|
|
IStripeAdapter stripeAdapter,
|
|
ISubscriberService subscriberService,
|
|
IUserRepository userRepository,
|
|
[FromKeyedServices(AutomaticTaxFactory.PersonalUse)] IAutomaticTaxStrategy automaticTaxStrategy) : 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<string, string>
|
|
{
|
|
["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<string> 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<Customer> 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<string, string>
|
|
{
|
|
["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<Subscription> CreateSubscriptionAsync(
|
|
Guid userId,
|
|
Customer customer,
|
|
int? storage)
|
|
{
|
|
var subscriptionItemOptionsList = new List<SubscriptionItemOptions>
|
|
{
|
|
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
|
|
{
|
|
CollectionMethod = StripeConstants.CollectionMethod.ChargeAutomatically,
|
|
Customer = customer.Id,
|
|
Items = subscriptionItemOptionsList,
|
|
Metadata = new Dictionary<string, string>
|
|
{
|
|
["userId"] = userId.ToString()
|
|
},
|
|
PaymentBehavior = usingPayPal
|
|
? StripeConstants.PaymentBehavior.DefaultIncomplete
|
|
: null,
|
|
OffSession = true
|
|
};
|
|
|
|
if (featureService.IsEnabled(FeatureFlagKeys.PM19147_AutomaticTaxImprovements))
|
|
{
|
|
automaticTaxStrategy.SetCreateOptions(subscriptionCreateOptions, customer);
|
|
}
|
|
else
|
|
{
|
|
subscriptionCreateOptions.AutomaticTax = new SubscriptionAutomaticTaxOptions
|
|
{
|
|
Enabled = customer.Tax?.AutomaticTax == StripeConstants.AutomaticTaxStatus.Supported,
|
|
};
|
|
}
|
|
|
|
var subscription = await stripeAdapter.SubscriptionCreateAsync(subscriptionCreateOptions);
|
|
|
|
if (usingPayPal)
|
|
{
|
|
await stripeAdapter.InvoiceUpdateAsync(subscription.LatestInvoiceId, new InvoiceUpdateOptions
|
|
{
|
|
AutoAdvance = false
|
|
});
|
|
}
|
|
|
|
return subscription;
|
|
}
|
|
|
|
private async Task<Customer> 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);
|
|
}
|
|
}
|