mirror of
https://github.com/bitwarden/server.git
synced 2025-04-06 13:38:13 -05:00
[PM-19147] Automatic Tax Improvements (#5545)
* 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
This commit is contained in:
parent
10ea2cb3eb
commit
b309de141d
@ -1,4 +1,5 @@
|
|||||||
using Bit.Core.AdminConsole.Entities;
|
using Bit.Core;
|
||||||
|
using Bit.Core.AdminConsole.Entities;
|
||||||
using Bit.Core.AdminConsole.Entities.Provider;
|
using Bit.Core.AdminConsole.Entities.Provider;
|
||||||
using Bit.Core.AdminConsole.OrganizationFeatures.OrganizationUsers.Interfaces;
|
using Bit.Core.AdminConsole.OrganizationFeatures.OrganizationUsers.Interfaces;
|
||||||
using Bit.Core.AdminConsole.Providers.Interfaces;
|
using Bit.Core.AdminConsole.Providers.Interfaces;
|
||||||
@ -7,10 +8,12 @@ using Bit.Core.Billing.Constants;
|
|||||||
using Bit.Core.Billing.Extensions;
|
using Bit.Core.Billing.Extensions;
|
||||||
using Bit.Core.Billing.Pricing;
|
using Bit.Core.Billing.Pricing;
|
||||||
using Bit.Core.Billing.Services;
|
using Bit.Core.Billing.Services;
|
||||||
|
using Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.Exceptions;
|
using Bit.Core.Exceptions;
|
||||||
using Bit.Core.Repositories;
|
using Bit.Core.Repositories;
|
||||||
using Bit.Core.Services;
|
using Bit.Core.Services;
|
||||||
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
using Stripe;
|
using Stripe;
|
||||||
|
|
||||||
namespace Bit.Commercial.Core.AdminConsole.Providers;
|
namespace Bit.Commercial.Core.AdminConsole.Providers;
|
||||||
@ -28,6 +31,7 @@ public class RemoveOrganizationFromProviderCommand : IRemoveOrganizationFromProv
|
|||||||
private readonly ISubscriberService _subscriberService;
|
private readonly ISubscriberService _subscriberService;
|
||||||
private readonly IHasConfirmedOwnersExceptQuery _hasConfirmedOwnersExceptQuery;
|
private readonly IHasConfirmedOwnersExceptQuery _hasConfirmedOwnersExceptQuery;
|
||||||
private readonly IPricingClient _pricingClient;
|
private readonly IPricingClient _pricingClient;
|
||||||
|
private readonly IAutomaticTaxStrategy _automaticTaxStrategy;
|
||||||
|
|
||||||
public RemoveOrganizationFromProviderCommand(
|
public RemoveOrganizationFromProviderCommand(
|
||||||
IEventService eventService,
|
IEventService eventService,
|
||||||
@ -40,7 +44,8 @@ public class RemoveOrganizationFromProviderCommand : IRemoveOrganizationFromProv
|
|||||||
IProviderBillingService providerBillingService,
|
IProviderBillingService providerBillingService,
|
||||||
ISubscriberService subscriberService,
|
ISubscriberService subscriberService,
|
||||||
IHasConfirmedOwnersExceptQuery hasConfirmedOwnersExceptQuery,
|
IHasConfirmedOwnersExceptQuery hasConfirmedOwnersExceptQuery,
|
||||||
IPricingClient pricingClient)
|
IPricingClient pricingClient,
|
||||||
|
[FromKeyedServices(AutomaticTaxFactory.BusinessUse)] IAutomaticTaxStrategy automaticTaxStrategy)
|
||||||
{
|
{
|
||||||
_eventService = eventService;
|
_eventService = eventService;
|
||||||
_mailService = mailService;
|
_mailService = mailService;
|
||||||
@ -53,6 +58,7 @@ public class RemoveOrganizationFromProviderCommand : IRemoveOrganizationFromProv
|
|||||||
_subscriberService = subscriberService;
|
_subscriberService = subscriberService;
|
||||||
_hasConfirmedOwnersExceptQuery = hasConfirmedOwnersExceptQuery;
|
_hasConfirmedOwnersExceptQuery = hasConfirmedOwnersExceptQuery;
|
||||||
_pricingClient = pricingClient;
|
_pricingClient = pricingClient;
|
||||||
|
_automaticTaxStrategy = automaticTaxStrategy;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task RemoveOrganizationFromProvider(
|
public async Task RemoveOrganizationFromProvider(
|
||||||
@ -107,10 +113,11 @@ public class RemoveOrganizationFromProviderCommand : IRemoveOrganizationFromProv
|
|||||||
organization.IsValidClient() &&
|
organization.IsValidClient() &&
|
||||||
!string.IsNullOrEmpty(organization.GatewayCustomerId))
|
!string.IsNullOrEmpty(organization.GatewayCustomerId))
|
||||||
{
|
{
|
||||||
await _stripeAdapter.CustomerUpdateAsync(organization.GatewayCustomerId, new CustomerUpdateOptions
|
var customer = await _stripeAdapter.CustomerUpdateAsync(organization.GatewayCustomerId, new CustomerUpdateOptions
|
||||||
{
|
{
|
||||||
Description = string.Empty,
|
Description = string.Empty,
|
||||||
Email = organization.BillingEmail
|
Email = organization.BillingEmail,
|
||||||
|
Expand = ["tax", "tax_ids"]
|
||||||
});
|
});
|
||||||
|
|
||||||
var plan = await _pricingClient.GetPlanOrThrow(organization.PlanType);
|
var plan = await _pricingClient.GetPlanOrThrow(organization.PlanType);
|
||||||
@ -120,7 +127,6 @@ public class RemoveOrganizationFromProviderCommand : IRemoveOrganizationFromProv
|
|||||||
Customer = organization.GatewayCustomerId,
|
Customer = organization.GatewayCustomerId,
|
||||||
CollectionMethod = StripeConstants.CollectionMethod.SendInvoice,
|
CollectionMethod = StripeConstants.CollectionMethod.SendInvoice,
|
||||||
DaysUntilDue = 30,
|
DaysUntilDue = 30,
|
||||||
AutomaticTax = new SubscriptionAutomaticTaxOptions { Enabled = true },
|
|
||||||
Metadata = new Dictionary<string, string>
|
Metadata = new Dictionary<string, string>
|
||||||
{
|
{
|
||||||
{ "organizationId", organization.Id.ToString() }
|
{ "organizationId", organization.Id.ToString() }
|
||||||
@ -130,6 +136,18 @@ public class RemoveOrganizationFromProviderCommand : IRemoveOrganizationFromProv
|
|||||||
Items = [new SubscriptionItemOptions { Price = plan.PasswordManager.StripeSeatPlanId, Quantity = organization.Seats }]
|
Items = [new SubscriptionItemOptions { Price = plan.PasswordManager.StripeSeatPlanId, Quantity = organization.Seats }]
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (_featureService.IsEnabled(FeatureFlagKeys.PM19147_AutomaticTaxImprovements))
|
||||||
|
{
|
||||||
|
_automaticTaxStrategy.SetCreateOptions(subscriptionCreateOptions, customer);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
subscriptionCreateOptions.AutomaticTax ??= new SubscriptionAutomaticTaxOptions
|
||||||
|
{
|
||||||
|
Enabled = true
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
var subscription = await _stripeAdapter.SubscriptionCreateAsync(subscriptionCreateOptions);
|
var subscription = await _stripeAdapter.SubscriptionCreateAsync(subscriptionCreateOptions);
|
||||||
|
|
||||||
organization.GatewaySubscriptionId = subscription.Id;
|
organization.GatewaySubscriptionId = subscription.Id;
|
||||||
|
@ -14,6 +14,7 @@ using Bit.Core.Billing.Pricing;
|
|||||||
using Bit.Core.Billing.Repositories;
|
using Bit.Core.Billing.Repositories;
|
||||||
using Bit.Core.Billing.Services;
|
using Bit.Core.Billing.Services;
|
||||||
using Bit.Core.Billing.Services.Contracts;
|
using Bit.Core.Billing.Services.Contracts;
|
||||||
|
using Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.Exceptions;
|
using Bit.Core.Exceptions;
|
||||||
using Bit.Core.Models.Business;
|
using Bit.Core.Models.Business;
|
||||||
@ -22,6 +23,7 @@ using Bit.Core.Services;
|
|||||||
using Bit.Core.Settings;
|
using Bit.Core.Settings;
|
||||||
using Bit.Core.Utilities;
|
using Bit.Core.Utilities;
|
||||||
using CsvHelper;
|
using CsvHelper;
|
||||||
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
using Stripe;
|
using Stripe;
|
||||||
|
|
||||||
@ -29,6 +31,7 @@ namespace Bit.Commercial.Core.Billing;
|
|||||||
|
|
||||||
public class ProviderBillingService(
|
public class ProviderBillingService(
|
||||||
IEventService eventService,
|
IEventService eventService,
|
||||||
|
IFeatureService featureService,
|
||||||
IGlobalSettings globalSettings,
|
IGlobalSettings globalSettings,
|
||||||
ILogger<ProviderBillingService> logger,
|
ILogger<ProviderBillingService> logger,
|
||||||
IOrganizationRepository organizationRepository,
|
IOrganizationRepository organizationRepository,
|
||||||
@ -40,7 +43,9 @@ public class ProviderBillingService(
|
|||||||
IProviderUserRepository providerUserRepository,
|
IProviderUserRepository providerUserRepository,
|
||||||
IStripeAdapter stripeAdapter,
|
IStripeAdapter stripeAdapter,
|
||||||
ISubscriberService subscriberService,
|
ISubscriberService subscriberService,
|
||||||
ITaxService taxService) : IProviderBillingService
|
ITaxService taxService,
|
||||||
|
[FromKeyedServices(AutomaticTaxFactory.BusinessUse)] IAutomaticTaxStrategy automaticTaxStrategy)
|
||||||
|
: IProviderBillingService
|
||||||
{
|
{
|
||||||
[RequireFeature(FeatureFlagKeys.P15179_AddExistingOrgsFromProviderPortal)]
|
[RequireFeature(FeatureFlagKeys.P15179_AddExistingOrgsFromProviderPortal)]
|
||||||
public async Task AddExistingOrganization(
|
public async Task AddExistingOrganization(
|
||||||
@ -557,7 +562,8 @@ public class ProviderBillingService(
|
|||||||
{
|
{
|
||||||
ArgumentNullException.ThrowIfNull(provider);
|
ArgumentNullException.ThrowIfNull(provider);
|
||||||
|
|
||||||
var customer = await subscriberService.GetCustomerOrThrow(provider);
|
var customerGetOptions = new CustomerGetOptions { Expand = ["tax", "tax_ids"] };
|
||||||
|
var customer = await subscriberService.GetCustomerOrThrow(provider, customerGetOptions);
|
||||||
|
|
||||||
var providerPlans = await providerPlanRepository.GetByProviderId(provider.Id);
|
var providerPlans = await providerPlanRepository.GetByProviderId(provider.Id);
|
||||||
|
|
||||||
@ -589,10 +595,6 @@ public class ProviderBillingService(
|
|||||||
|
|
||||||
var subscriptionCreateOptions = new SubscriptionCreateOptions
|
var subscriptionCreateOptions = new SubscriptionCreateOptions
|
||||||
{
|
{
|
||||||
AutomaticTax = new SubscriptionAutomaticTaxOptions
|
|
||||||
{
|
|
||||||
Enabled = true
|
|
||||||
},
|
|
||||||
CollectionMethod = StripeConstants.CollectionMethod.SendInvoice,
|
CollectionMethod = StripeConstants.CollectionMethod.SendInvoice,
|
||||||
Customer = customer.Id,
|
Customer = customer.Id,
|
||||||
DaysUntilDue = 30,
|
DaysUntilDue = 30,
|
||||||
@ -605,6 +607,15 @@ public class ProviderBillingService(
|
|||||||
ProrationBehavior = StripeConstants.ProrationBehavior.CreateProrations
|
ProrationBehavior = StripeConstants.ProrationBehavior.CreateProrations
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (featureService.IsEnabled(FeatureFlagKeys.PM19147_AutomaticTaxImprovements))
|
||||||
|
{
|
||||||
|
automaticTaxStrategy.SetCreateOptions(subscriptionCreateOptions, customer);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
subscriptionCreateOptions.AutomaticTax = new SubscriptionAutomaticTaxOptions { Enabled = true };
|
||||||
|
}
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var subscription = await stripeAdapter.SubscriptionCreateAsync(subscriptionCreateOptions);
|
var subscription = await stripeAdapter.SubscriptionCreateAsync(subscriptionCreateOptions);
|
||||||
|
@ -228,6 +228,26 @@ public class RemoveOrganizationFromProviderCommandTests
|
|||||||
Id = "subscription_id"
|
Id = "subscription_id"
|
||||||
});
|
});
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IAutomaticTaxStrategy>()
|
||||||
|
.When(x => x.SetCreateOptions(
|
||||||
|
Arg.Is<SubscriptionCreateOptions>(options =>
|
||||||
|
options.Customer == organization.GatewayCustomerId &&
|
||||||
|
options.CollectionMethod == StripeConstants.CollectionMethod.SendInvoice &&
|
||||||
|
options.DaysUntilDue == 30 &&
|
||||||
|
options.Metadata["organizationId"] == organization.Id.ToString() &&
|
||||||
|
options.OffSession == true &&
|
||||||
|
options.ProrationBehavior == StripeConstants.ProrationBehavior.CreateProrations &&
|
||||||
|
options.Items.First().Price == teamsMonthlyPlan.PasswordManager.StripeSeatPlanId &&
|
||||||
|
options.Items.First().Quantity == organization.Seats)
|
||||||
|
, Arg.Any<Customer>()))
|
||||||
|
.Do(x =>
|
||||||
|
{
|
||||||
|
x.Arg<SubscriptionCreateOptions>().AutomaticTax = new SubscriptionAutomaticTaxOptions
|
||||||
|
{
|
||||||
|
Enabled = true
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
await sutProvider.Sut.RemoveOrganizationFromProvider(provider, providerOrganization, organization);
|
await sutProvider.Sut.RemoveOrganizationFromProvider(provider, providerOrganization, organization);
|
||||||
|
|
||||||
await stripeAdapter.Received(1).SubscriptionCreateAsync(Arg.Is<SubscriptionCreateOptions>(options =>
|
await stripeAdapter.Received(1).SubscriptionCreateAsync(Arg.Is<SubscriptionCreateOptions>(options =>
|
||||||
|
@ -924,7 +924,11 @@ public class ProviderBillingServiceTests
|
|||||||
{
|
{
|
||||||
provider.GatewaySubscriptionId = null;
|
provider.GatewaySubscriptionId = null;
|
||||||
|
|
||||||
sutProvider.GetDependency<ISubscriberService>().GetCustomerOrThrow(provider).Returns(new Customer
|
sutProvider.GetDependency<ISubscriberService>()
|
||||||
|
.GetCustomerOrThrow(
|
||||||
|
provider,
|
||||||
|
Arg.Is<CustomerGetOptions>(p => p.Expand.Contains("tax") || p.Expand.Contains("tax_ids")))
|
||||||
|
.Returns(new Customer
|
||||||
{
|
{
|
||||||
Id = "customer_id",
|
Id = "customer_id",
|
||||||
Tax = new CustomerTax { AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported }
|
Tax = new CustomerTax { AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported }
|
||||||
@ -975,11 +979,15 @@ public class ProviderBillingServiceTests
|
|||||||
{
|
{
|
||||||
provider.GatewaySubscriptionId = null;
|
provider.GatewaySubscriptionId = null;
|
||||||
|
|
||||||
sutProvider.GetDependency<ISubscriberService>().GetCustomerOrThrow(provider).Returns(new Customer
|
var customer = new Customer
|
||||||
{
|
{
|
||||||
Id = "customer_id",
|
Id = "customer_id",
|
||||||
Tax = new CustomerTax { AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported }
|
Tax = new CustomerTax { AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported }
|
||||||
});
|
};
|
||||||
|
sutProvider.GetDependency<ISubscriberService>()
|
||||||
|
.GetCustomerOrThrow(
|
||||||
|
provider,
|
||||||
|
Arg.Is<CustomerGetOptions>(p => p.Expand.Contains("tax") || p.Expand.Contains("tax_ids"))).Returns(customer);
|
||||||
|
|
||||||
var providerPlans = new List<ProviderPlan>
|
var providerPlans = new List<ProviderPlan>
|
||||||
{
|
{
|
||||||
@ -1017,6 +1025,19 @@ public class ProviderBillingServiceTests
|
|||||||
|
|
||||||
var expected = new Subscription { Id = "subscription_id", Status = StripeConstants.SubscriptionStatus.Active };
|
var expected = new Subscription { Id = "subscription_id", Status = StripeConstants.SubscriptionStatus.Active };
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IAutomaticTaxStrategy>()
|
||||||
|
.When(x => x.SetCreateOptions(
|
||||||
|
Arg.Is<SubscriptionCreateOptions>(options =>
|
||||||
|
options.Customer == "customer_id")
|
||||||
|
, Arg.Is<Customer>(p => p == customer)))
|
||||||
|
.Do(x =>
|
||||||
|
{
|
||||||
|
x.Arg<SubscriptionCreateOptions>().AutomaticTax = new SubscriptionAutomaticTaxOptions
|
||||||
|
{
|
||||||
|
Enabled = true
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
sutProvider.GetDependency<IStripeAdapter>().SubscriptionCreateAsync(Arg.Is<SubscriptionCreateOptions>(
|
sutProvider.GetDependency<IStripeAdapter>().SubscriptionCreateAsync(Arg.Is<SubscriptionCreateOptions>(
|
||||||
sub =>
|
sub =>
|
||||||
sub.AutomaticTax.Enabled == true &&
|
sub.AutomaticTax.Enabled == true &&
|
||||||
|
@ -1,8 +1,11 @@
|
|||||||
using Bit.Core.AdminConsole.Repositories;
|
using Bit.Core;
|
||||||
|
using Bit.Core.AdminConsole.Repositories;
|
||||||
using Bit.Core.Billing.Constants;
|
using Bit.Core.Billing.Constants;
|
||||||
using Bit.Core.Billing.Enums;
|
using Bit.Core.Billing.Enums;
|
||||||
using Bit.Core.Billing.Extensions;
|
using Bit.Core.Billing.Extensions;
|
||||||
using Bit.Core.Billing.Pricing;
|
using Bit.Core.Billing.Pricing;
|
||||||
|
using Bit.Core.Billing.Services;
|
||||||
|
using Bit.Core.Billing.Services.Contracts;
|
||||||
using Bit.Core.OrganizationFeatures.OrganizationSponsorships.FamiliesForEnterprise.Interfaces;
|
using Bit.Core.OrganizationFeatures.OrganizationSponsorships.FamiliesForEnterprise.Interfaces;
|
||||||
using Bit.Core.Repositories;
|
using Bit.Core.Repositories;
|
||||||
using Bit.Core.Services;
|
using Bit.Core.Services;
|
||||||
@ -12,6 +15,7 @@ using Event = Stripe.Event;
|
|||||||
namespace Bit.Billing.Services.Implementations;
|
namespace Bit.Billing.Services.Implementations;
|
||||||
|
|
||||||
public class UpcomingInvoiceHandler(
|
public class UpcomingInvoiceHandler(
|
||||||
|
IFeatureService featureService,
|
||||||
ILogger<StripeEventProcessor> logger,
|
ILogger<StripeEventProcessor> logger,
|
||||||
IMailService mailService,
|
IMailService mailService,
|
||||||
IOrganizationRepository organizationRepository,
|
IOrganizationRepository organizationRepository,
|
||||||
@ -21,7 +25,8 @@ public class UpcomingInvoiceHandler(
|
|||||||
IStripeEventService stripeEventService,
|
IStripeEventService stripeEventService,
|
||||||
IStripeEventUtilityService stripeEventUtilityService,
|
IStripeEventUtilityService stripeEventUtilityService,
|
||||||
IUserRepository userRepository,
|
IUserRepository userRepository,
|
||||||
IValidateSponsorshipCommand validateSponsorshipCommand)
|
IValidateSponsorshipCommand validateSponsorshipCommand,
|
||||||
|
IAutomaticTaxFactory automaticTaxFactory)
|
||||||
: IUpcomingInvoiceHandler
|
: IUpcomingInvoiceHandler
|
||||||
{
|
{
|
||||||
public async Task HandleAsync(Event parsedEvent)
|
public async Task HandleAsync(Event parsedEvent)
|
||||||
@ -136,6 +141,21 @@ public class UpcomingInvoiceHandler(
|
|||||||
|
|
||||||
private async Task TryEnableAutomaticTaxAsync(Subscription subscription)
|
private async Task TryEnableAutomaticTaxAsync(Subscription subscription)
|
||||||
{
|
{
|
||||||
|
if (featureService.IsEnabled(FeatureFlagKeys.PM19147_AutomaticTaxImprovements))
|
||||||
|
{
|
||||||
|
var automaticTaxParameters = new AutomaticTaxFactoryParameters(subscription.Items.Select(x => x.Price.Id));
|
||||||
|
var automaticTaxStrategy = await automaticTaxFactory.CreateAsync(automaticTaxParameters);
|
||||||
|
var updateOptions = automaticTaxStrategy.GetUpdateOptions(subscription);
|
||||||
|
|
||||||
|
if (updateOptions == null)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await stripeFacade.UpdateSubscription(subscription.Id, updateOptions);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (subscription.AutomaticTax.Enabled ||
|
if (subscription.AutomaticTax.Enabled ||
|
||||||
!subscription.Customer.HasBillingLocation() ||
|
!subscription.Customer.HasBillingLocation() ||
|
||||||
await IsNonTaxableNonUSBusinessUseSubscription(subscription))
|
await IsNonTaxableNonUSBusinessUseSubscription(subscription))
|
||||||
|
@ -47,6 +47,8 @@ public static class StripeConstants
|
|||||||
public static class MetadataKeys
|
public static class MetadataKeys
|
||||||
{
|
{
|
||||||
public const string OrganizationId = "organizationId";
|
public const string OrganizationId = "organizationId";
|
||||||
|
public const string ProviderId = "providerId";
|
||||||
|
public const string UserId = "userId";
|
||||||
}
|
}
|
||||||
|
|
||||||
public static class PaymentBehavior
|
public static class PaymentBehavior
|
||||||
|
@ -21,7 +21,7 @@ public static class CustomerExtensions
|
|||||||
/// <param name="customer"></param>
|
/// <param name="customer"></param>
|
||||||
/// <returns></returns>
|
/// <returns></returns>
|
||||||
public static bool HasTaxLocationVerified(this Customer customer) =>
|
public static bool HasTaxLocationVerified(this Customer customer) =>
|
||||||
customer?.Tax?.AutomaticTax == StripeConstants.AutomaticTaxStatus.Supported;
|
customer?.Tax?.AutomaticTax != StripeConstants.AutomaticTaxStatus.UnrecognizedLocation;
|
||||||
|
|
||||||
public static decimal GetBillingBalance(this Customer customer)
|
public static decimal GetBillingBalance(this Customer customer)
|
||||||
{
|
{
|
||||||
|
@ -4,6 +4,7 @@ using Bit.Core.Billing.Licenses.Extensions;
|
|||||||
using Bit.Core.Billing.Pricing;
|
using Bit.Core.Billing.Pricing;
|
||||||
using Bit.Core.Billing.Services;
|
using Bit.Core.Billing.Services;
|
||||||
using Bit.Core.Billing.Services.Implementations;
|
using Bit.Core.Billing.Services.Implementations;
|
||||||
|
using Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
||||||
|
|
||||||
namespace Bit.Core.Billing.Extensions;
|
namespace Bit.Core.Billing.Extensions;
|
||||||
|
|
||||||
@ -18,6 +19,9 @@ public static class ServiceCollectionExtensions
|
|||||||
services.AddTransient<IPremiumUserBillingService, PremiumUserBillingService>();
|
services.AddTransient<IPremiumUserBillingService, PremiumUserBillingService>();
|
||||||
services.AddTransient<ISetupIntentCache, SetupIntentDistributedCache>();
|
services.AddTransient<ISetupIntentCache, SetupIntentDistributedCache>();
|
||||||
services.AddTransient<ISubscriberService, SubscriberService>();
|
services.AddTransient<ISubscriberService, SubscriberService>();
|
||||||
|
services.AddKeyedTransient<IAutomaticTaxStrategy, PersonalUseAutomaticTaxStrategy>(AutomaticTaxFactory.PersonalUse);
|
||||||
|
services.AddKeyedTransient<IAutomaticTaxStrategy, BusinessUseAutomaticTaxStrategy>(AutomaticTaxFactory.BusinessUse);
|
||||||
|
services.AddTransient<IAutomaticTaxFactory, AutomaticTaxFactory>();
|
||||||
services.AddLicenseServices();
|
services.AddLicenseServices();
|
||||||
services.AddPricingClient();
|
services.AddPricingClient();
|
||||||
}
|
}
|
||||||
|
@ -1,26 +0,0 @@
|
|||||||
using Stripe;
|
|
||||||
|
|
||||||
namespace Bit.Core.Billing.Extensions;
|
|
||||||
|
|
||||||
public static class SubscriptionCreateOptionsExtensions
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Attempts to enable automatic tax for given new subscription options.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="options"></param>
|
|
||||||
/// <param name="customer">The existing customer.</param>
|
|
||||||
/// <returns>Returns true when successful, false when conditions are not met.</returns>
|
|
||||||
public static bool EnableAutomaticTax(this SubscriptionCreateOptions options, Customer customer)
|
|
||||||
{
|
|
||||||
// We might only need to check the automatic tax status.
|
|
||||||
if (!customer.HasTaxLocationVerified() && string.IsNullOrWhiteSpace(customer.Address?.Country))
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
options.DefaultTaxRates = [];
|
|
||||||
options.AutomaticTax = new SubscriptionAutomaticTaxOptions { Enabled = true };
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
@ -0,0 +1,30 @@
|
|||||||
|
#nullable enable
|
||||||
|
using Bit.Core.Billing.Enums;
|
||||||
|
using Bit.Core.Entities;
|
||||||
|
|
||||||
|
namespace Bit.Core.Billing.Services.Contracts;
|
||||||
|
|
||||||
|
public class AutomaticTaxFactoryParameters
|
||||||
|
{
|
||||||
|
public AutomaticTaxFactoryParameters(PlanType planType)
|
||||||
|
{
|
||||||
|
PlanType = planType;
|
||||||
|
}
|
||||||
|
|
||||||
|
public AutomaticTaxFactoryParameters(ISubscriber subscriber, IEnumerable<string> prices)
|
||||||
|
{
|
||||||
|
Subscriber = subscriber;
|
||||||
|
Prices = prices;
|
||||||
|
}
|
||||||
|
|
||||||
|
public AutomaticTaxFactoryParameters(IEnumerable<string> prices)
|
||||||
|
{
|
||||||
|
Prices = prices;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ISubscriber? Subscriber { get; init; }
|
||||||
|
|
||||||
|
public PlanType? PlanType { get; init; }
|
||||||
|
|
||||||
|
public IEnumerable<string>? Prices { get; init; }
|
||||||
|
}
|
11
src/Core/Billing/Services/IAutomaticTaxFactory.cs
Normal file
11
src/Core/Billing/Services/IAutomaticTaxFactory.cs
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
using Bit.Core.Billing.Services.Contracts;
|
||||||
|
|
||||||
|
namespace Bit.Core.Billing.Services;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Responsible for defining the correct automatic tax strategy for either personal use of business use.
|
||||||
|
/// </summary>
|
||||||
|
public interface IAutomaticTaxFactory
|
||||||
|
{
|
||||||
|
Task<IAutomaticTaxStrategy> CreateAsync(AutomaticTaxFactoryParameters parameters);
|
||||||
|
}
|
33
src/Core/Billing/Services/IAutomaticTaxStrategy.cs
Normal file
33
src/Core/Billing/Services/IAutomaticTaxStrategy.cs
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
#nullable enable
|
||||||
|
using Stripe;
|
||||||
|
|
||||||
|
namespace Bit.Core.Billing.Services;
|
||||||
|
|
||||||
|
public interface IAutomaticTaxStrategy
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
///
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="subscription"></param>
|
||||||
|
/// <returns>
|
||||||
|
/// Returns <see cref="SubscriptionUpdateOptions" /> if changes are to be applied to the subscription, returns null
|
||||||
|
/// otherwise.
|
||||||
|
/// </returns>
|
||||||
|
SubscriptionUpdateOptions? GetUpdateOptions(Subscription subscription);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Modifies an existing <see cref="SubscriptionCreateOptions" /> object with the automatic tax flag set correctly.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="options"></param>
|
||||||
|
/// <param name="customer"></param>
|
||||||
|
void SetCreateOptions(SubscriptionCreateOptions options, Customer customer);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Modifies an existing <see cref="SubscriptionUpdateOptions" /> object with the automatic tax flag set correctly.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="options"></param>
|
||||||
|
/// <param name="subscription"></param>
|
||||||
|
void SetUpdateOptions(SubscriptionUpdateOptions options, Subscription subscription);
|
||||||
|
|
||||||
|
void SetInvoiceCreatePreviewOptions(InvoiceCreatePreviewOptions options);
|
||||||
|
}
|
@ -0,0 +1,50 @@
|
|||||||
|
#nullable enable
|
||||||
|
using Bit.Core.Billing.Enums;
|
||||||
|
using Bit.Core.Billing.Pricing;
|
||||||
|
using Bit.Core.Billing.Services.Contracts;
|
||||||
|
using Bit.Core.Entities;
|
||||||
|
using Bit.Core.Services;
|
||||||
|
|
||||||
|
namespace Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
||||||
|
|
||||||
|
public class AutomaticTaxFactory(
|
||||||
|
IFeatureService featureService,
|
||||||
|
IPricingClient pricingClient) : IAutomaticTaxFactory
|
||||||
|
{
|
||||||
|
public const string BusinessUse = "business-use";
|
||||||
|
public const string PersonalUse = "personal-use";
|
||||||
|
|
||||||
|
private readonly Lazy<Task<IEnumerable<string>>> _personalUsePlansTask = new(async () =>
|
||||||
|
{
|
||||||
|
var plans = await Task.WhenAll(
|
||||||
|
pricingClient.GetPlanOrThrow(PlanType.FamiliesAnnually2019),
|
||||||
|
pricingClient.GetPlanOrThrow(PlanType.FamiliesAnnually));
|
||||||
|
|
||||||
|
return plans.Select(plan => plan.PasswordManager.StripePlanId);
|
||||||
|
});
|
||||||
|
|
||||||
|
public async Task<IAutomaticTaxStrategy> CreateAsync(AutomaticTaxFactoryParameters parameters)
|
||||||
|
{
|
||||||
|
if (parameters.Subscriber is User)
|
||||||
|
{
|
||||||
|
return new PersonalUseAutomaticTaxStrategy(featureService);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (parameters.PlanType.HasValue)
|
||||||
|
{
|
||||||
|
var plan = await pricingClient.GetPlanOrThrow(parameters.PlanType.Value);
|
||||||
|
return plan.CanBeUsedByBusiness
|
||||||
|
? new BusinessUseAutomaticTaxStrategy(featureService)
|
||||||
|
: new PersonalUseAutomaticTaxStrategy(featureService);
|
||||||
|
}
|
||||||
|
|
||||||
|
var personalUsePlans = await _personalUsePlansTask.Value;
|
||||||
|
|
||||||
|
if (parameters.Prices != null && parameters.Prices.Any(x => personalUsePlans.Any(y => y == x)))
|
||||||
|
{
|
||||||
|
return new PersonalUseAutomaticTaxStrategy(featureService);
|
||||||
|
}
|
||||||
|
|
||||||
|
return new BusinessUseAutomaticTaxStrategy(featureService);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,96 @@
|
|||||||
|
#nullable enable
|
||||||
|
using Bit.Core.Billing.Extensions;
|
||||||
|
using Bit.Core.Services;
|
||||||
|
using Stripe;
|
||||||
|
|
||||||
|
namespace Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
||||||
|
|
||||||
|
public class BusinessUseAutomaticTaxStrategy(IFeatureService featureService) : IAutomaticTaxStrategy
|
||||||
|
{
|
||||||
|
public SubscriptionUpdateOptions? GetUpdateOptions(Subscription subscription)
|
||||||
|
{
|
||||||
|
if (!featureService.IsEnabled(FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var shouldBeEnabled = ShouldBeEnabled(subscription.Customer);
|
||||||
|
if (subscription.AutomaticTax.Enabled == shouldBeEnabled)
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var options = new SubscriptionUpdateOptions
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTaxOptions
|
||||||
|
{
|
||||||
|
Enabled = shouldBeEnabled
|
||||||
|
},
|
||||||
|
DefaultTaxRates = []
|
||||||
|
};
|
||||||
|
|
||||||
|
return options;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetCreateOptions(SubscriptionCreateOptions options, Customer customer)
|
||||||
|
{
|
||||||
|
options.AutomaticTax = new SubscriptionAutomaticTaxOptions
|
||||||
|
{
|
||||||
|
Enabled = ShouldBeEnabled(customer)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetUpdateOptions(SubscriptionUpdateOptions options, Subscription subscription)
|
||||||
|
{
|
||||||
|
if (!featureService.IsEnabled(FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var shouldBeEnabled = ShouldBeEnabled(subscription.Customer);
|
||||||
|
|
||||||
|
if (subscription.AutomaticTax.Enabled == shouldBeEnabled)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
options.AutomaticTax = new SubscriptionAutomaticTaxOptions
|
||||||
|
{
|
||||||
|
Enabled = shouldBeEnabled
|
||||||
|
};
|
||||||
|
options.DefaultTaxRates = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetInvoiceCreatePreviewOptions(InvoiceCreatePreviewOptions options)
|
||||||
|
{
|
||||||
|
options.AutomaticTax ??= new InvoiceAutomaticTaxOptions();
|
||||||
|
|
||||||
|
if (options.CustomerDetails.Address.Country == "US")
|
||||||
|
{
|
||||||
|
options.AutomaticTax.Enabled = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
options.AutomaticTax.Enabled = options.CustomerDetails.TaxIds != null && options.CustomerDetails.TaxIds.Any();
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool ShouldBeEnabled(Customer customer)
|
||||||
|
{
|
||||||
|
if (!customer.HasTaxLocationVerified())
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (customer.Address.Country == "US")
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (customer.TaxIds == null)
|
||||||
|
{
|
||||||
|
throw new ArgumentNullException(nameof(customer.TaxIds), "`customer.tax_ids` must be expanded.");
|
||||||
|
}
|
||||||
|
|
||||||
|
return customer.TaxIds.Any();
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,64 @@
|
|||||||
|
#nullable enable
|
||||||
|
using Bit.Core.Billing.Extensions;
|
||||||
|
using Bit.Core.Services;
|
||||||
|
using Stripe;
|
||||||
|
|
||||||
|
namespace Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
||||||
|
|
||||||
|
public class PersonalUseAutomaticTaxStrategy(IFeatureService featureService) : IAutomaticTaxStrategy
|
||||||
|
{
|
||||||
|
public void SetCreateOptions(SubscriptionCreateOptions options, Customer customer)
|
||||||
|
{
|
||||||
|
options.AutomaticTax = new SubscriptionAutomaticTaxOptions
|
||||||
|
{
|
||||||
|
Enabled = ShouldBeEnabled(customer)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetUpdateOptions(SubscriptionUpdateOptions options, Subscription subscription)
|
||||||
|
{
|
||||||
|
if (!featureService.IsEnabled(FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
options.AutomaticTax = new SubscriptionAutomaticTaxOptions
|
||||||
|
{
|
||||||
|
Enabled = ShouldBeEnabled(subscription.Customer)
|
||||||
|
};
|
||||||
|
options.DefaultTaxRates = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
public SubscriptionUpdateOptions? GetUpdateOptions(Subscription subscription)
|
||||||
|
{
|
||||||
|
if (!featureService.IsEnabled(FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (subscription.AutomaticTax.Enabled == ShouldBeEnabled(subscription.Customer))
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var options = new SubscriptionUpdateOptions
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTaxOptions
|
||||||
|
{
|
||||||
|
Enabled = ShouldBeEnabled(subscription.Customer),
|
||||||
|
},
|
||||||
|
DefaultTaxRates = []
|
||||||
|
};
|
||||||
|
|
||||||
|
return options;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetInvoiceCreatePreviewOptions(InvoiceCreatePreviewOptions options)
|
||||||
|
{
|
||||||
|
options.AutomaticTax = new InvoiceAutomaticTaxOptions { Enabled = true };
|
||||||
|
}
|
||||||
|
|
||||||
|
private static bool ShouldBeEnabled(Customer customer)
|
||||||
|
{
|
||||||
|
return customer.HasTaxLocationVerified();
|
||||||
|
}
|
||||||
|
}
|
@ -1,9 +1,11 @@
|
|||||||
using Bit.Core.AdminConsole.Entities;
|
using Bit.Core.AdminConsole.Entities;
|
||||||
using Bit.Core.Billing.Caches;
|
using Bit.Core.Billing.Caches;
|
||||||
using Bit.Core.Billing.Constants;
|
using Bit.Core.Billing.Constants;
|
||||||
|
using Bit.Core.Billing.Extensions;
|
||||||
using Bit.Core.Billing.Models;
|
using Bit.Core.Billing.Models;
|
||||||
using Bit.Core.Billing.Models.Sales;
|
using Bit.Core.Billing.Models.Sales;
|
||||||
using Bit.Core.Billing.Pricing;
|
using Bit.Core.Billing.Pricing;
|
||||||
|
using Bit.Core.Billing.Services.Contracts;
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.Exceptions;
|
using Bit.Core.Exceptions;
|
||||||
using Bit.Core.Repositories;
|
using Bit.Core.Repositories;
|
||||||
@ -23,6 +25,7 @@ namespace Bit.Core.Billing.Services.Implementations;
|
|||||||
|
|
||||||
public class OrganizationBillingService(
|
public class OrganizationBillingService(
|
||||||
IBraintreeGateway braintreeGateway,
|
IBraintreeGateway braintreeGateway,
|
||||||
|
IFeatureService featureService,
|
||||||
IGlobalSettings globalSettings,
|
IGlobalSettings globalSettings,
|
||||||
ILogger<OrganizationBillingService> logger,
|
ILogger<OrganizationBillingService> logger,
|
||||||
IOrganizationRepository organizationRepository,
|
IOrganizationRepository organizationRepository,
|
||||||
@ -30,7 +33,8 @@ public class OrganizationBillingService(
|
|||||||
ISetupIntentCache setupIntentCache,
|
ISetupIntentCache setupIntentCache,
|
||||||
IStripeAdapter stripeAdapter,
|
IStripeAdapter stripeAdapter,
|
||||||
ISubscriberService subscriberService,
|
ISubscriberService subscriberService,
|
||||||
ITaxService taxService) : IOrganizationBillingService
|
ITaxService taxService,
|
||||||
|
IAutomaticTaxFactory automaticTaxFactory) : IOrganizationBillingService
|
||||||
{
|
{
|
||||||
public async Task Finalize(OrganizationSale sale)
|
public async Task Finalize(OrganizationSale sale)
|
||||||
{
|
{
|
||||||
@ -143,7 +147,7 @@ public class OrganizationBillingService(
|
|||||||
Coupon = customerSetup.Coupon,
|
Coupon = customerSetup.Coupon,
|
||||||
Description = organization.DisplayBusinessName(),
|
Description = organization.DisplayBusinessName(),
|
||||||
Email = organization.BillingEmail,
|
Email = organization.BillingEmail,
|
||||||
Expand = ["tax"],
|
Expand = ["tax", "tax_ids"],
|
||||||
InvoiceSettings = new CustomerInvoiceSettingsOptions
|
InvoiceSettings = new CustomerInvoiceSettingsOptions
|
||||||
{
|
{
|
||||||
CustomFields = [
|
CustomFields = [
|
||||||
@ -369,21 +373,8 @@ public class OrganizationBillingService(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var customerHasTaxInfo = customer is
|
|
||||||
{
|
|
||||||
Address:
|
|
||||||
{
|
|
||||||
Country: not null and not "",
|
|
||||||
PostalCode: not null and not ""
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
var subscriptionCreateOptions = new SubscriptionCreateOptions
|
var subscriptionCreateOptions = new SubscriptionCreateOptions
|
||||||
{
|
{
|
||||||
AutomaticTax = new SubscriptionAutomaticTaxOptions
|
|
||||||
{
|
|
||||||
Enabled = customerHasTaxInfo
|
|
||||||
},
|
|
||||||
CollectionMethod = StripeConstants.CollectionMethod.ChargeAutomatically,
|
CollectionMethod = StripeConstants.CollectionMethod.ChargeAutomatically,
|
||||||
Customer = customer.Id,
|
Customer = customer.Id,
|
||||||
Items = subscriptionItemOptionsList,
|
Items = subscriptionItemOptionsList,
|
||||||
@ -395,6 +386,18 @@ public class OrganizationBillingService(
|
|||||||
TrialPeriodDays = subscriptionSetup.SkipTrial ? 0 : plan.TrialPeriodDays
|
TrialPeriodDays = subscriptionSetup.SkipTrial ? 0 : plan.TrialPeriodDays
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (featureService.IsEnabled(FeatureFlagKeys.PM19147_AutomaticTaxImprovements))
|
||||||
|
{
|
||||||
|
var automaticTaxParameters = new AutomaticTaxFactoryParameters(subscriptionSetup.PlanType);
|
||||||
|
var automaticTaxStrategy = await automaticTaxFactory.CreateAsync(automaticTaxParameters);
|
||||||
|
automaticTaxStrategy.SetCreateOptions(subscriptionCreateOptions, customer);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
subscriptionCreateOptions.AutomaticTax ??= new SubscriptionAutomaticTaxOptions();
|
||||||
|
subscriptionCreateOptions.AutomaticTax.Enabled = customer.HasBillingLocation();
|
||||||
|
}
|
||||||
|
|
||||||
return await stripeAdapter.SubscriptionCreateAsync(subscriptionCreateOptions);
|
return await stripeAdapter.SubscriptionCreateAsync(subscriptionCreateOptions);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
using Bit.Core.Billing.Constants;
|
using Bit.Core.Billing.Constants;
|
||||||
using Bit.Core.Billing.Models;
|
using Bit.Core.Billing.Models;
|
||||||
using Bit.Core.Billing.Models.Sales;
|
using Bit.Core.Billing.Models.Sales;
|
||||||
|
using Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
||||||
using Bit.Core.Entities;
|
using Bit.Core.Entities;
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.Exceptions;
|
using Bit.Core.Exceptions;
|
||||||
@ -9,6 +10,7 @@ using Bit.Core.Repositories;
|
|||||||
using Bit.Core.Services;
|
using Bit.Core.Services;
|
||||||
using Bit.Core.Settings;
|
using Bit.Core.Settings;
|
||||||
using Braintree;
|
using Braintree;
|
||||||
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
using Stripe;
|
using Stripe;
|
||||||
using Customer = Stripe.Customer;
|
using Customer = Stripe.Customer;
|
||||||
@ -20,12 +22,14 @@ using static Utilities;
|
|||||||
|
|
||||||
public class PremiumUserBillingService(
|
public class PremiumUserBillingService(
|
||||||
IBraintreeGateway braintreeGateway,
|
IBraintreeGateway braintreeGateway,
|
||||||
|
IFeatureService featureService,
|
||||||
IGlobalSettings globalSettings,
|
IGlobalSettings globalSettings,
|
||||||
ILogger<PremiumUserBillingService> logger,
|
ILogger<PremiumUserBillingService> logger,
|
||||||
ISetupIntentCache setupIntentCache,
|
ISetupIntentCache setupIntentCache,
|
||||||
IStripeAdapter stripeAdapter,
|
IStripeAdapter stripeAdapter,
|
||||||
ISubscriberService subscriberService,
|
ISubscriberService subscriberService,
|
||||||
IUserRepository userRepository) : IPremiumUserBillingService
|
IUserRepository userRepository,
|
||||||
|
[FromKeyedServices(AutomaticTaxFactory.PersonalUse)] IAutomaticTaxStrategy automaticTaxStrategy) : IPremiumUserBillingService
|
||||||
{
|
{
|
||||||
public async Task Credit(User user, decimal amount)
|
public async Task Credit(User user, decimal amount)
|
||||||
{
|
{
|
||||||
@ -318,10 +322,6 @@ public class PremiumUserBillingService(
|
|||||||
|
|
||||||
var subscriptionCreateOptions = new SubscriptionCreateOptions
|
var subscriptionCreateOptions = new SubscriptionCreateOptions
|
||||||
{
|
{
|
||||||
AutomaticTax = new SubscriptionAutomaticTaxOptions
|
|
||||||
{
|
|
||||||
Enabled = customer.Tax?.AutomaticTax == StripeConstants.AutomaticTaxStatus.Supported,
|
|
||||||
},
|
|
||||||
CollectionMethod = StripeConstants.CollectionMethod.ChargeAutomatically,
|
CollectionMethod = StripeConstants.CollectionMethod.ChargeAutomatically,
|
||||||
Customer = customer.Id,
|
Customer = customer.Id,
|
||||||
Items = subscriptionItemOptionsList,
|
Items = subscriptionItemOptionsList,
|
||||||
@ -335,6 +335,18 @@ public class PremiumUserBillingService(
|
|||||||
OffSession = true
|
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);
|
var subscription = await stripeAdapter.SubscriptionCreateAsync(subscriptionCreateOptions);
|
||||||
|
|
||||||
if (usingPayPal)
|
if (usingPayPal)
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
using Bit.Core.Billing.Caches;
|
using Bit.Core.Billing.Caches;
|
||||||
using Bit.Core.Billing.Constants;
|
using Bit.Core.Billing.Constants;
|
||||||
using Bit.Core.Billing.Models;
|
using Bit.Core.Billing.Models;
|
||||||
|
using Bit.Core.Billing.Services.Contracts;
|
||||||
using Bit.Core.Entities;
|
using Bit.Core.Entities;
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.Exceptions;
|
using Bit.Core.Exceptions;
|
||||||
@ -20,11 +21,13 @@ namespace Bit.Core.Billing.Services.Implementations;
|
|||||||
|
|
||||||
public class SubscriberService(
|
public class SubscriberService(
|
||||||
IBraintreeGateway braintreeGateway,
|
IBraintreeGateway braintreeGateway,
|
||||||
|
IFeatureService featureService,
|
||||||
IGlobalSettings globalSettings,
|
IGlobalSettings globalSettings,
|
||||||
ILogger<SubscriberService> logger,
|
ILogger<SubscriberService> logger,
|
||||||
ISetupIntentCache setupIntentCache,
|
ISetupIntentCache setupIntentCache,
|
||||||
IStripeAdapter stripeAdapter,
|
IStripeAdapter stripeAdapter,
|
||||||
ITaxService taxService) : ISubscriberService
|
ITaxService taxService,
|
||||||
|
IAutomaticTaxFactory automaticTaxFactory) : ISubscriberService
|
||||||
{
|
{
|
||||||
public async Task CancelSubscription(
|
public async Task CancelSubscription(
|
||||||
ISubscriber subscriber,
|
ISubscriber subscriber,
|
||||||
@ -438,7 +441,8 @@ public class SubscriberService(
|
|||||||
ArgumentNullException.ThrowIfNull(subscriber);
|
ArgumentNullException.ThrowIfNull(subscriber);
|
||||||
ArgumentNullException.ThrowIfNull(tokenizedPaymentSource);
|
ArgumentNullException.ThrowIfNull(tokenizedPaymentSource);
|
||||||
|
|
||||||
var customer = await GetCustomerOrThrow(subscriber);
|
var customerGetOptions = new CustomerGetOptions { Expand = ["tax", "tax_ids"] };
|
||||||
|
var customer = await GetCustomerOrThrow(subscriber, customerGetOptions);
|
||||||
|
|
||||||
var (type, token) = tokenizedPaymentSource;
|
var (type, token) = tokenizedPaymentSource;
|
||||||
|
|
||||||
@ -597,7 +601,7 @@ public class SubscriberService(
|
|||||||
Expand = ["subscriptions", "tax", "tax_ids"]
|
Expand = ["subscriptions", "tax", "tax_ids"]
|
||||||
});
|
});
|
||||||
|
|
||||||
await stripeAdapter.CustomerUpdateAsync(customer.Id, new CustomerUpdateOptions
|
customer = await stripeAdapter.CustomerUpdateAsync(customer.Id, new CustomerUpdateOptions
|
||||||
{
|
{
|
||||||
Address = new AddressOptions
|
Address = new AddressOptions
|
||||||
{
|
{
|
||||||
@ -607,7 +611,8 @@ public class SubscriberService(
|
|||||||
Line2 = taxInformation.Line2,
|
Line2 = taxInformation.Line2,
|
||||||
City = taxInformation.City,
|
City = taxInformation.City,
|
||||||
State = taxInformation.State
|
State = taxInformation.State
|
||||||
}
|
},
|
||||||
|
Expand = ["subscriptions", "tax", "tax_ids"]
|
||||||
});
|
});
|
||||||
|
|
||||||
var taxId = customer.TaxIds?.FirstOrDefault();
|
var taxId = customer.TaxIds?.FirstOrDefault();
|
||||||
@ -661,6 +666,26 @@ public class SubscriberService(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (featureService.IsEnabled(FeatureFlagKeys.PM19147_AutomaticTaxImprovements))
|
||||||
|
{
|
||||||
|
if (!string.IsNullOrEmpty(subscriber.GatewaySubscriptionId))
|
||||||
|
{
|
||||||
|
var subscriptionGetOptions = new SubscriptionGetOptions
|
||||||
|
{
|
||||||
|
Expand = ["customer.tax", "customer.tax_ids"]
|
||||||
|
};
|
||||||
|
var subscription = await stripeAdapter.SubscriptionGetAsync(subscriber.GatewaySubscriptionId, subscriptionGetOptions);
|
||||||
|
var automaticTaxParameters = new AutomaticTaxFactoryParameters(subscriber, subscription.Items.Select(x => x.Price.Id));
|
||||||
|
var automaticTaxStrategy = await automaticTaxFactory.CreateAsync(automaticTaxParameters);
|
||||||
|
var automaticTaxOptions = automaticTaxStrategy.GetUpdateOptions(subscription);
|
||||||
|
if (automaticTaxOptions?.AutomaticTax?.Enabled != null)
|
||||||
|
{
|
||||||
|
await stripeAdapter.SubscriptionUpdateAsync(subscriber.GatewaySubscriptionId, automaticTaxOptions);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
if (SubscriberIsEligibleForAutomaticTax(subscriber, customer))
|
if (SubscriberIsEligibleForAutomaticTax(subscriber, customer))
|
||||||
{
|
{
|
||||||
await stripeAdapter.SubscriptionUpdateAsync(subscriber.GatewaySubscriptionId,
|
await stripeAdapter.SubscriptionUpdateAsync(subscriber.GatewaySubscriptionId,
|
||||||
@ -677,6 +702,7 @@ public class SubscriberService(
|
|||||||
(localCustomer.Subscriptions?.Any(sub => sub.Id == localSubscriber.GatewaySubscriptionId && !sub.AutomaticTax.Enabled) ?? false) &&
|
(localCustomer.Subscriptions?.Any(sub => sub.Id == localSubscriber.GatewaySubscriptionId && !sub.AutomaticTax.Enabled) ?? false) &&
|
||||||
localCustomer.Tax?.AutomaticTax == StripeConstants.AutomaticTaxStatus.Supported;
|
localCustomer.Tax?.AutomaticTax == StripeConstants.AutomaticTaxStatus.Supported;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public async Task VerifyBankAccount(
|
public async Task VerifyBankAccount(
|
||||||
ISubscriber subscriber,
|
ISubscriber subscriber,
|
||||||
|
@ -148,6 +148,8 @@ public static class FeatureFlagKeys
|
|||||||
public const string P15179_AddExistingOrgsFromProviderPortal = "pm-15179-add-existing-orgs-from-provider-portal";
|
public const string P15179_AddExistingOrgsFromProviderPortal = "pm-15179-add-existing-orgs-from-provider-portal";
|
||||||
public const string PM12276Breadcrumbing = "pm-12276-breadcrumbing-for-business-features";
|
public const string PM12276Breadcrumbing = "pm-12276-breadcrumbing-for-business-features";
|
||||||
public const string PM18794_ProviderPaymentMethod = "pm-18794-provider-payment-method";
|
public const string PM18794_ProviderPaymentMethod = "pm-18794-provider-payment-method";
|
||||||
|
public const string PM19147_AutomaticTaxImprovements = "pm-19147-automatic-tax-improvements";
|
||||||
|
public const string PM19422_AllowAutomaticTaxUpdates = "pm-19422-allow-automatic-tax-updates";
|
||||||
|
|
||||||
/* Key Management Team */
|
/* Key Management Team */
|
||||||
public const string ReturnErrorOnExistingKeypair = "return-error-on-existing-keypair";
|
public const string ReturnErrorOnExistingKeypair = "return-error-on-existing-keypair";
|
||||||
@ -169,6 +171,7 @@ public static class FeatureFlagKeys
|
|||||||
public const string SingleTapPasskeyAuthentication = "single-tap-passkey-authentication";
|
public const string SingleTapPasskeyAuthentication = "single-tap-passkey-authentication";
|
||||||
public const string EnablePMAuthenticatorSync = "enable-pm-bwa-sync";
|
public const string EnablePMAuthenticatorSync = "enable-pm-bwa-sync";
|
||||||
public const string PM3503_MobileAnonAddySelfHostAlias = "anon-addy-self-host-alias";
|
public const string PM3503_MobileAnonAddySelfHostAlias = "anon-addy-self-host-alias";
|
||||||
|
|
||||||
public const string PM3553_MobileSimpleLoginSelfHostAlias = "simple-login-self-host-alias";
|
public const string PM3553_MobileSimpleLoginSelfHostAlias = "simple-login-self-host-alias";
|
||||||
|
|
||||||
/* Platform Team */
|
/* Platform Team */
|
||||||
|
@ -9,6 +9,8 @@ using Bit.Core.Billing.Models.Api.Responses;
|
|||||||
using Bit.Core.Billing.Models.Business;
|
using Bit.Core.Billing.Models.Business;
|
||||||
using Bit.Core.Billing.Pricing;
|
using Bit.Core.Billing.Pricing;
|
||||||
using Bit.Core.Billing.Services;
|
using Bit.Core.Billing.Services;
|
||||||
|
using Bit.Core.Billing.Services.Contracts;
|
||||||
|
using Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
||||||
using Bit.Core.Entities;
|
using Bit.Core.Entities;
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.Exceptions;
|
using Bit.Core.Exceptions;
|
||||||
@ -16,6 +18,7 @@ using Bit.Core.Models.BitStripe;
|
|||||||
using Bit.Core.Models.Business;
|
using Bit.Core.Models.Business;
|
||||||
using Bit.Core.Repositories;
|
using Bit.Core.Repositories;
|
||||||
using Bit.Core.Settings;
|
using Bit.Core.Settings;
|
||||||
|
using Microsoft.Extensions.DependencyInjection;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
using Stripe;
|
using Stripe;
|
||||||
using PaymentMethod = Stripe.PaymentMethod;
|
using PaymentMethod = Stripe.PaymentMethod;
|
||||||
@ -36,6 +39,8 @@ public class StripePaymentService : IPaymentService
|
|||||||
private readonly ITaxService _taxService;
|
private readonly ITaxService _taxService;
|
||||||
private readonly ISubscriberService _subscriberService;
|
private readonly ISubscriberService _subscriberService;
|
||||||
private readonly IPricingClient _pricingClient;
|
private readonly IPricingClient _pricingClient;
|
||||||
|
private readonly IAutomaticTaxFactory _automaticTaxFactory;
|
||||||
|
private readonly IAutomaticTaxStrategy _personalUseTaxStrategy;
|
||||||
|
|
||||||
public StripePaymentService(
|
public StripePaymentService(
|
||||||
ITransactionRepository transactionRepository,
|
ITransactionRepository transactionRepository,
|
||||||
@ -46,7 +51,9 @@ public class StripePaymentService : IPaymentService
|
|||||||
IFeatureService featureService,
|
IFeatureService featureService,
|
||||||
ITaxService taxService,
|
ITaxService taxService,
|
||||||
ISubscriberService subscriberService,
|
ISubscriberService subscriberService,
|
||||||
IPricingClient pricingClient)
|
IPricingClient pricingClient,
|
||||||
|
IAutomaticTaxFactory automaticTaxFactory,
|
||||||
|
[FromKeyedServices(AutomaticTaxFactory.PersonalUse)] IAutomaticTaxStrategy personalUseTaxStrategy)
|
||||||
{
|
{
|
||||||
_transactionRepository = transactionRepository;
|
_transactionRepository = transactionRepository;
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
@ -57,6 +64,8 @@ public class StripePaymentService : IPaymentService
|
|||||||
_taxService = taxService;
|
_taxService = taxService;
|
||||||
_subscriberService = subscriberService;
|
_subscriberService = subscriberService;
|
||||||
_pricingClient = pricingClient;
|
_pricingClient = pricingClient;
|
||||||
|
_automaticTaxFactory = automaticTaxFactory;
|
||||||
|
_personalUseTaxStrategy = personalUseTaxStrategy;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task ChangeOrganizationSponsorship(
|
private async Task ChangeOrganizationSponsorship(
|
||||||
@ -91,9 +100,7 @@ public class StripePaymentService : IPaymentService
|
|||||||
SubscriptionUpdate subscriptionUpdate, bool invoiceNow = false)
|
SubscriptionUpdate subscriptionUpdate, bool invoiceNow = false)
|
||||||
{
|
{
|
||||||
// remember, when in doubt, throw
|
// remember, when in doubt, throw
|
||||||
var subGetOptions = new SubscriptionGetOptions();
|
var subGetOptions = new SubscriptionGetOptions { Expand = ["customer.tax", "customer.tax_ids"] };
|
||||||
// subGetOptions.AddExpand("customer");
|
|
||||||
subGetOptions.AddExpand("customer.tax");
|
|
||||||
var sub = await _stripeAdapter.SubscriptionGetAsync(subscriber.GatewaySubscriptionId, subGetOptions);
|
var sub = await _stripeAdapter.SubscriptionGetAsync(subscriber.GatewaySubscriptionId, subGetOptions);
|
||||||
if (sub == null)
|
if (sub == null)
|
||||||
{
|
{
|
||||||
@ -124,7 +131,19 @@ public class StripePaymentService : IPaymentService
|
|||||||
new SubscriptionPendingInvoiceItemIntervalOptions { Interval = "month" };
|
new SubscriptionPendingInvoiceItemIntervalOptions { Interval = "month" };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (subscriptionUpdate is CompleteSubscriptionUpdate)
|
||||||
|
{
|
||||||
|
if (_featureService.IsEnabled(FeatureFlagKeys.PM19147_AutomaticTaxImprovements))
|
||||||
|
{
|
||||||
|
var automaticTaxParameters = new AutomaticTaxFactoryParameters(subscriber, updatedItemOptions.Select(x => x.Plan ?? x.Price));
|
||||||
|
var automaticTaxStrategy = await _automaticTaxFactory.CreateAsync(automaticTaxParameters);
|
||||||
|
automaticTaxStrategy.SetUpdateOptions(subUpdateOptions, sub);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
subUpdateOptions.EnableAutomaticTax(sub.Customer, sub);
|
subUpdateOptions.EnableAutomaticTax(sub.Customer, sub);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (!subscriptionUpdate.UpdateNeeded(sub))
|
if (!subscriptionUpdate.UpdateNeeded(sub))
|
||||||
{
|
{
|
||||||
@ -811,6 +830,30 @@ public class StripePaymentService : IPaymentService
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (_featureService.IsEnabled(FeatureFlagKeys.PM19147_AutomaticTaxImprovements))
|
||||||
|
{
|
||||||
|
if (!string.IsNullOrEmpty(subscriber.GatewaySubscriptionId))
|
||||||
|
{
|
||||||
|
var subscriptionGetOptions = new SubscriptionGetOptions
|
||||||
|
{
|
||||||
|
Expand = ["customer.tax", "customer.tax_ids"]
|
||||||
|
};
|
||||||
|
var subscription = await _stripeAdapter.SubscriptionGetAsync(subscriber.GatewaySubscriptionId, subscriptionGetOptions);
|
||||||
|
|
||||||
|
var automaticTaxParameters = new AutomaticTaxFactoryParameters(subscriber, subscription.Items.Select(x => x.Price.Id));
|
||||||
|
var automaticTaxStrategy = await _automaticTaxFactory.CreateAsync(automaticTaxParameters);
|
||||||
|
var subscriptionUpdateOptions = automaticTaxStrategy.GetUpdateOptions(subscription);
|
||||||
|
|
||||||
|
if (subscriptionUpdateOptions != null)
|
||||||
|
{
|
||||||
|
_ = await _stripeAdapter.SubscriptionUpdateAsync(
|
||||||
|
subscriber.GatewaySubscriptionId,
|
||||||
|
subscriptionUpdateOptions);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
if (!string.IsNullOrEmpty(subscriber.GatewaySubscriptionId) &&
|
if (!string.IsNullOrEmpty(subscriber.GatewaySubscriptionId) &&
|
||||||
customer.Subscriptions.Any(sub =>
|
customer.Subscriptions.Any(sub =>
|
||||||
sub.Id == subscriber.GatewaySubscriptionId &&
|
sub.Id == subscriber.GatewaySubscriptionId &&
|
||||||
@ -828,6 +871,7 @@ public class StripePaymentService : IPaymentService
|
|||||||
subscriptionUpdateOptions);
|
subscriptionUpdateOptions);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
catch
|
catch
|
||||||
{
|
{
|
||||||
if (braintreeCustomer != null && !hadBtCustomer)
|
if (braintreeCustomer != null && !hadBtCustomer)
|
||||||
@ -1214,6 +1258,8 @@ public class StripePaymentService : IPaymentService
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_personalUseTaxStrategy.SetInvoiceCreatePreviewOptions(options);
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var invoice = await _stripeAdapter.InvoiceCreatePreviewAsync(options);
|
var invoice = await _stripeAdapter.InvoiceCreatePreviewAsync(options);
|
||||||
@ -1256,10 +1302,6 @@ public class StripePaymentService : IPaymentService
|
|||||||
|
|
||||||
var options = new InvoiceCreatePreviewOptions
|
var options = new InvoiceCreatePreviewOptions
|
||||||
{
|
{
|
||||||
AutomaticTax = new InvoiceAutomaticTaxOptions
|
|
||||||
{
|
|
||||||
Enabled = true,
|
|
||||||
},
|
|
||||||
Currency = "usd",
|
Currency = "usd",
|
||||||
SubscriptionDetails = new InvoiceSubscriptionDetailsOptions
|
SubscriptionDetails = new InvoiceSubscriptionDetailsOptions
|
||||||
{
|
{
|
||||||
@ -1347,9 +1389,11 @@ public class StripePaymentService : IPaymentService
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Customer gatewayCustomer = null;
|
||||||
|
|
||||||
if (!string.IsNullOrWhiteSpace(gatewayCustomerId))
|
if (!string.IsNullOrWhiteSpace(gatewayCustomerId))
|
||||||
{
|
{
|
||||||
var gatewayCustomer = await _stripeAdapter.CustomerGetAsync(gatewayCustomerId);
|
gatewayCustomer = await _stripeAdapter.CustomerGetAsync(gatewayCustomerId);
|
||||||
|
|
||||||
if (gatewayCustomer.Discount != null)
|
if (gatewayCustomer.Discount != null)
|
||||||
{
|
{
|
||||||
@ -1367,6 +1411,10 @@ public class StripePaymentService : IPaymentService
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var automaticTaxFactoryParameters = new AutomaticTaxFactoryParameters(parameters.PasswordManager.Plan);
|
||||||
|
var automaticTaxStrategy = await _automaticTaxFactory.CreateAsync(automaticTaxFactoryParameters);
|
||||||
|
automaticTaxStrategy.SetInvoiceCreatePreviewOptions(options);
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
var invoice = await _stripeAdapter.InvoiceCreatePreviewAsync(options);
|
var invoice = await _stripeAdapter.InvoiceCreatePreviewAsync(options);
|
||||||
|
@ -0,0 +1,492 @@
|
|||||||
|
using Bit.Core.Billing.Constants;
|
||||||
|
using Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
||||||
|
using Bit.Core.Services;
|
||||||
|
using Bit.Test.Common.AutoFixture;
|
||||||
|
using Bit.Test.Common.AutoFixture.Attributes;
|
||||||
|
using NSubstitute;
|
||||||
|
using Stripe;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace Bit.Core.Test.Billing.Services.Implementations.AutomaticTax;
|
||||||
|
|
||||||
|
[SutProviderCustomize]
|
||||||
|
public class BusinessUseAutomaticTaxStrategyTests
|
||||||
|
{
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void GetUpdateOptions_ReturnsNull_WhenFeatureFlagAllowingToUpdateSubscriptionsIsDisabled(
|
||||||
|
SutProvider<BusinessUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription();
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(false);
|
||||||
|
|
||||||
|
var actual = sutProvider.Sut.GetUpdateOptions(subscription);
|
||||||
|
|
||||||
|
Assert.Null(actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void GetUpdateOptions_ReturnsNull_WhenSubscriptionDoesNotNeedUpdating(
|
||||||
|
SutProvider<BusinessUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = true
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Address = new Address
|
||||||
|
{
|
||||||
|
Country = "US",
|
||||||
|
},
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
var actual = sutProvider.Sut.GetUpdateOptions(subscription);
|
||||||
|
|
||||||
|
Assert.Null(actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void GetUpdateOptions_SetsAutomaticTaxToFalse_WhenTaxLocationIsUnrecognizedOrInvalid(
|
||||||
|
SutProvider<BusinessUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = true
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.UnrecognizedLocation
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
var actual = sutProvider.Sut.GetUpdateOptions(subscription);
|
||||||
|
|
||||||
|
Assert.NotNull(actual);
|
||||||
|
Assert.False(actual.AutomaticTax.Enabled);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void GetUpdateOptions_SetsAutomaticTaxToTrue_ForAmericanCustomers(
|
||||||
|
SutProvider<BusinessUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = false
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Address = new Address
|
||||||
|
{
|
||||||
|
Country = "US",
|
||||||
|
},
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
var actual = sutProvider.Sut.GetUpdateOptions(subscription);
|
||||||
|
|
||||||
|
Assert.NotNull(actual);
|
||||||
|
Assert.True(actual.AutomaticTax.Enabled);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void GetUpdateOptions_SetsAutomaticTaxToTrue_ForGlobalCustomersWithTaxIds(
|
||||||
|
SutProvider<BusinessUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = false
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Address = new Address
|
||||||
|
{
|
||||||
|
Country = "ES",
|
||||||
|
},
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported
|
||||||
|
},
|
||||||
|
TaxIds = new StripeList<TaxId>
|
||||||
|
{
|
||||||
|
Data = new List<TaxId>
|
||||||
|
{
|
||||||
|
new()
|
||||||
|
{
|
||||||
|
Country = "ES",
|
||||||
|
Type = "eu_vat",
|
||||||
|
Value = "ESZ8880999Z"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
var actual = sutProvider.Sut.GetUpdateOptions(subscription);
|
||||||
|
|
||||||
|
Assert.NotNull(actual);
|
||||||
|
Assert.True(actual.AutomaticTax.Enabled);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void GetUpdateOptions_ThrowsArgumentNullException_WhenTaxIdsIsNull(
|
||||||
|
SutProvider<BusinessUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = true
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Address = new Address
|
||||||
|
{
|
||||||
|
Country = "ES",
|
||||||
|
},
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported
|
||||||
|
},
|
||||||
|
TaxIds = null
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
Assert.Throws<ArgumentNullException>(() => sutProvider.Sut.GetUpdateOptions(subscription));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void GetUpdateOptions_SetsAutomaticTaxToTrue_ForGlobalCustomersWithoutTaxIds(
|
||||||
|
SutProvider<BusinessUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = true
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Address = new Address
|
||||||
|
{
|
||||||
|
Country = "ES",
|
||||||
|
},
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported
|
||||||
|
},
|
||||||
|
TaxIds = new StripeList<TaxId>
|
||||||
|
{
|
||||||
|
Data = new List<TaxId>()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
var actual = sutProvider.Sut.GetUpdateOptions(subscription);
|
||||||
|
|
||||||
|
Assert.NotNull(actual);
|
||||||
|
Assert.False(actual.AutomaticTax.Enabled);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void SetUpdateOptions_SetsNothing_WhenFeatureFlagAllowingToUpdateSubscriptionsIsDisabled(
|
||||||
|
SutProvider<BusinessUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var options = new SubscriptionUpdateOptions();
|
||||||
|
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Address = new()
|
||||||
|
{
|
||||||
|
Country = "US"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(false);
|
||||||
|
|
||||||
|
sutProvider.Sut.SetUpdateOptions(options, subscription);
|
||||||
|
|
||||||
|
Assert.Null(options.AutomaticTax);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void SetUpdateOptions_SetsNothing_WhenSubscriptionDoesNotNeedUpdating(
|
||||||
|
SutProvider<BusinessUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var options = new SubscriptionUpdateOptions();
|
||||||
|
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = true
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Address = new Address
|
||||||
|
{
|
||||||
|
Country = "US",
|
||||||
|
},
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
sutProvider.Sut.SetUpdateOptions(options, subscription);
|
||||||
|
|
||||||
|
Assert.Null(options.AutomaticTax);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void SetUpdateOptions_SetsAutomaticTaxToFalse_WhenTaxLocationIsUnrecognizedOrInvalid(
|
||||||
|
SutProvider<BusinessUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var options = new SubscriptionUpdateOptions();
|
||||||
|
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = true
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.UnrecognizedLocation
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
sutProvider.Sut.SetUpdateOptions(options, subscription);
|
||||||
|
|
||||||
|
Assert.False(options.AutomaticTax.Enabled);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void SetUpdateOptions_SetsAutomaticTaxToTrue_ForAmericanCustomers(
|
||||||
|
SutProvider<BusinessUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var options = new SubscriptionUpdateOptions();
|
||||||
|
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = false
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Address = new Address
|
||||||
|
{
|
||||||
|
Country = "US",
|
||||||
|
},
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
sutProvider.Sut.SetUpdateOptions(options, subscription);
|
||||||
|
|
||||||
|
Assert.True(options.AutomaticTax!.Enabled);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void SetUpdateOptions_SetsAutomaticTaxToTrue_ForGlobalCustomersWithTaxIds(
|
||||||
|
SutProvider<BusinessUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var options = new SubscriptionUpdateOptions();
|
||||||
|
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = false
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Address = new Address
|
||||||
|
{
|
||||||
|
Country = "ES",
|
||||||
|
},
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported
|
||||||
|
},
|
||||||
|
TaxIds = new StripeList<TaxId>
|
||||||
|
{
|
||||||
|
Data = new List<TaxId>
|
||||||
|
{
|
||||||
|
new()
|
||||||
|
{
|
||||||
|
Country = "ES",
|
||||||
|
Type = "eu_vat",
|
||||||
|
Value = "ESZ8880999Z"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
sutProvider.Sut.SetUpdateOptions(options, subscription);
|
||||||
|
|
||||||
|
Assert.True(options.AutomaticTax!.Enabled);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void SetUpdateOptions_ThrowsArgumentNullException_WhenTaxIdsIsNull(
|
||||||
|
SutProvider<BusinessUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var options = new SubscriptionUpdateOptions();
|
||||||
|
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = true
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Address = new Address
|
||||||
|
{
|
||||||
|
Country = "ES",
|
||||||
|
},
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported
|
||||||
|
},
|
||||||
|
TaxIds = null
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
Assert.Throws<ArgumentNullException>(() => sutProvider.Sut.SetUpdateOptions(options, subscription));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void SetUpdateOptions_SetsAutomaticTaxToTrue_ForGlobalCustomersWithoutTaxIds(
|
||||||
|
SutProvider<BusinessUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var options = new SubscriptionUpdateOptions();
|
||||||
|
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = true
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Address = new Address
|
||||||
|
{
|
||||||
|
Country = "ES",
|
||||||
|
},
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported
|
||||||
|
},
|
||||||
|
TaxIds = new StripeList<TaxId>
|
||||||
|
{
|
||||||
|
Data = new List<TaxId>()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
sutProvider.Sut.SetUpdateOptions(options, subscription);
|
||||||
|
|
||||||
|
Assert.False(options.AutomaticTax!.Enabled);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,217 @@
|
|||||||
|
using Bit.Core.Billing.Constants;
|
||||||
|
using Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
||||||
|
using Bit.Core.Services;
|
||||||
|
using Bit.Test.Common.AutoFixture;
|
||||||
|
using Bit.Test.Common.AutoFixture.Attributes;
|
||||||
|
using NSubstitute;
|
||||||
|
using Stripe;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace Bit.Core.Test.Billing.Services.Implementations.AutomaticTax;
|
||||||
|
|
||||||
|
[SutProviderCustomize]
|
||||||
|
public class PersonalUseAutomaticTaxStrategyTests
|
||||||
|
{
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void GetUpdateOptions_ReturnsNull_WhenFeatureFlagAllowingToUpdateSubscriptionsIsDisabled(
|
||||||
|
SutProvider<PersonalUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription();
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(false);
|
||||||
|
|
||||||
|
var actual = sutProvider.Sut.GetUpdateOptions(subscription);
|
||||||
|
|
||||||
|
Assert.Null(actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void GetUpdateOptions_ReturnsNull_WhenSubscriptionDoesNotNeedUpdating(
|
||||||
|
SutProvider<PersonalUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = true
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Address = new Address
|
||||||
|
{
|
||||||
|
Country = "US",
|
||||||
|
},
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
var actual = sutProvider.Sut.GetUpdateOptions(subscription);
|
||||||
|
|
||||||
|
Assert.Null(actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public void GetUpdateOptions_SetsAutomaticTaxToFalse_WhenTaxLocationIsUnrecognizedOrInvalid(
|
||||||
|
SutProvider<PersonalUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = true
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.UnrecognizedLocation
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
var actual = sutProvider.Sut.GetUpdateOptions(subscription);
|
||||||
|
|
||||||
|
Assert.NotNull(actual);
|
||||||
|
Assert.False(actual.AutomaticTax.Enabled);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData("CA")]
|
||||||
|
[BitAutoData("ES")]
|
||||||
|
[BitAutoData("US")]
|
||||||
|
public void GetUpdateOptions_SetsAutomaticTaxToTrue_ForAllCountries(
|
||||||
|
string country, SutProvider<PersonalUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = false
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Address = new Address
|
||||||
|
{
|
||||||
|
Country = country
|
||||||
|
},
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
var actual = sutProvider.Sut.GetUpdateOptions(subscription);
|
||||||
|
|
||||||
|
Assert.NotNull(actual);
|
||||||
|
Assert.True(actual.AutomaticTax.Enabled);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData("CA")]
|
||||||
|
[BitAutoData("ES")]
|
||||||
|
[BitAutoData("US")]
|
||||||
|
public void GetUpdateOptions_SetsAutomaticTaxToTrue_ForGlobalCustomersWithTaxIds(
|
||||||
|
string country, SutProvider<PersonalUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = false
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Address = new Address
|
||||||
|
{
|
||||||
|
Country = country,
|
||||||
|
},
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported
|
||||||
|
},
|
||||||
|
TaxIds = new StripeList<TaxId>
|
||||||
|
{
|
||||||
|
Data = new List<TaxId>
|
||||||
|
{
|
||||||
|
new()
|
||||||
|
{
|
||||||
|
Country = "ES",
|
||||||
|
Type = "eu_vat",
|
||||||
|
Value = "ESZ8880999Z"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
var actual = sutProvider.Sut.GetUpdateOptions(subscription);
|
||||||
|
|
||||||
|
Assert.NotNull(actual);
|
||||||
|
Assert.True(actual.AutomaticTax.Enabled);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData("CA")]
|
||||||
|
[BitAutoData("ES")]
|
||||||
|
[BitAutoData("US")]
|
||||||
|
public void GetUpdateOptions_SetsAutomaticTaxToTrue_ForGlobalCustomersWithoutTaxIds(
|
||||||
|
string country, SutProvider<PersonalUseAutomaticTaxStrategy> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTax
|
||||||
|
{
|
||||||
|
Enabled = false
|
||||||
|
},
|
||||||
|
Customer = new Customer
|
||||||
|
{
|
||||||
|
Address = new Address
|
||||||
|
{
|
||||||
|
Country = country
|
||||||
|
},
|
||||||
|
Tax = new CustomerTax
|
||||||
|
{
|
||||||
|
AutomaticTax = StripeConstants.AutomaticTaxStatus.Supported
|
||||||
|
},
|
||||||
|
TaxIds = new StripeList<TaxId>
|
||||||
|
{
|
||||||
|
Data = new List<TaxId>()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IFeatureService>()
|
||||||
|
.IsEnabled(Arg.Is<string>(p => p == FeatureFlagKeys.PM19422_AllowAutomaticTaxUpdates))
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
var actual = sutProvider.Sut.GetUpdateOptions(subscription);
|
||||||
|
|
||||||
|
Assert.NotNull(actual);
|
||||||
|
Assert.True(actual.AutomaticTax.Enabled);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,105 @@
|
|||||||
|
using Bit.Core.AdminConsole.Entities;
|
||||||
|
using Bit.Core.Billing.Enums;
|
||||||
|
using Bit.Core.Billing.Models.StaticStore.Plans;
|
||||||
|
using Bit.Core.Billing.Pricing;
|
||||||
|
using Bit.Core.Billing.Services.Contracts;
|
||||||
|
using Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
||||||
|
using Bit.Core.Entities;
|
||||||
|
using Bit.Test.Common.AutoFixture;
|
||||||
|
using Bit.Test.Common.AutoFixture.Attributes;
|
||||||
|
using NSubstitute;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace Bit.Core.Test.Billing.Services.Implementations;
|
||||||
|
|
||||||
|
[SutProviderCustomize]
|
||||||
|
public class AutomaticTaxFactoryTests
|
||||||
|
{
|
||||||
|
[BitAutoData]
|
||||||
|
[Theory]
|
||||||
|
public async Task CreateAsync_ReturnsPersonalUseStrategy_WhenSubscriberIsUser(SutProvider<AutomaticTaxFactory> sut)
|
||||||
|
{
|
||||||
|
var parameters = new AutomaticTaxFactoryParameters(new User(), []);
|
||||||
|
|
||||||
|
var actual = await sut.Sut.CreateAsync(parameters);
|
||||||
|
|
||||||
|
Assert.IsType<PersonalUseAutomaticTaxStrategy>(actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
[BitAutoData]
|
||||||
|
[Theory]
|
||||||
|
public async Task CreateAsync_ReturnsPersonalUseStrategy_WhenSubscriberIsOrganizationWithFamiliesAnnuallyPrice(
|
||||||
|
SutProvider<AutomaticTaxFactory> sut)
|
||||||
|
{
|
||||||
|
var familiesPlan = new FamiliesPlan();
|
||||||
|
var parameters = new AutomaticTaxFactoryParameters(new Organization(), [familiesPlan.PasswordManager.StripePlanId]);
|
||||||
|
|
||||||
|
sut.GetDependency<IPricingClient>()
|
||||||
|
.GetPlanOrThrow(Arg.Is<PlanType>(p => p == PlanType.FamiliesAnnually))
|
||||||
|
.Returns(new FamiliesPlan());
|
||||||
|
|
||||||
|
sut.GetDependency<IPricingClient>()
|
||||||
|
.GetPlanOrThrow(Arg.Is<PlanType>(p => p == PlanType.FamiliesAnnually2019))
|
||||||
|
.Returns(new Families2019Plan());
|
||||||
|
|
||||||
|
var actual = await sut.Sut.CreateAsync(parameters);
|
||||||
|
|
||||||
|
Assert.IsType<PersonalUseAutomaticTaxStrategy>(actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public async Task CreateAsync_ReturnsBusinessUseStrategy_WhenSubscriberIsOrganizationWithBusinessUsePrice(
|
||||||
|
EnterpriseAnnually plan,
|
||||||
|
SutProvider<AutomaticTaxFactory> sut)
|
||||||
|
{
|
||||||
|
var parameters = new AutomaticTaxFactoryParameters(new Organization(), [plan.PasswordManager.StripePlanId]);
|
||||||
|
|
||||||
|
sut.GetDependency<IPricingClient>()
|
||||||
|
.GetPlanOrThrow(Arg.Is<PlanType>(p => p == PlanType.FamiliesAnnually))
|
||||||
|
.Returns(new FamiliesPlan());
|
||||||
|
|
||||||
|
sut.GetDependency<IPricingClient>()
|
||||||
|
.GetPlanOrThrow(Arg.Is<PlanType>(p => p == PlanType.FamiliesAnnually2019))
|
||||||
|
.Returns(new Families2019Plan());
|
||||||
|
|
||||||
|
var actual = await sut.Sut.CreateAsync(parameters);
|
||||||
|
|
||||||
|
Assert.IsType<BusinessUseAutomaticTaxStrategy>(actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public async Task CreateAsync_ReturnsPersonalUseStrategy_WhenPlanIsMeantForPersonalUse(SutProvider<AutomaticTaxFactory> sut)
|
||||||
|
{
|
||||||
|
var parameters = new AutomaticTaxFactoryParameters(PlanType.FamiliesAnnually);
|
||||||
|
sut.GetDependency<IPricingClient>()
|
||||||
|
.GetPlanOrThrow(Arg.Is<PlanType>(p => p == parameters.PlanType.Value))
|
||||||
|
.Returns(new FamiliesPlan());
|
||||||
|
|
||||||
|
var actual = await sut.Sut.CreateAsync(parameters);
|
||||||
|
|
||||||
|
Assert.IsType<PersonalUseAutomaticTaxStrategy>(actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory]
|
||||||
|
[BitAutoData]
|
||||||
|
public async Task CreateAsync_ReturnsBusinessUseStrategy_WhenPlanIsMeantForBusinessUse(SutProvider<AutomaticTaxFactory> sut)
|
||||||
|
{
|
||||||
|
var parameters = new AutomaticTaxFactoryParameters(PlanType.EnterpriseAnnually);
|
||||||
|
sut.GetDependency<IPricingClient>()
|
||||||
|
.GetPlanOrThrow(Arg.Is<PlanType>(p => p == parameters.PlanType.Value))
|
||||||
|
.Returns(new EnterprisePlan(true));
|
||||||
|
|
||||||
|
var actual = await sut.Sut.CreateAsync(parameters);
|
||||||
|
|
||||||
|
Assert.IsType<BusinessUseAutomaticTaxStrategy>(actual);
|
||||||
|
}
|
||||||
|
|
||||||
|
public record EnterpriseAnnually : EnterprisePlan
|
||||||
|
{
|
||||||
|
public EnterpriseAnnually() : base(true)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -3,10 +3,13 @@ using Bit.Core.AdminConsole.Entities.Provider;
|
|||||||
using Bit.Core.Billing.Caches;
|
using Bit.Core.Billing.Caches;
|
||||||
using Bit.Core.Billing.Constants;
|
using Bit.Core.Billing.Constants;
|
||||||
using Bit.Core.Billing.Models;
|
using Bit.Core.Billing.Models;
|
||||||
|
using Bit.Core.Billing.Services;
|
||||||
|
using Bit.Core.Billing.Services.Contracts;
|
||||||
using Bit.Core.Billing.Services.Implementations;
|
using Bit.Core.Billing.Services.Implementations;
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.Services;
|
using Bit.Core.Services;
|
||||||
using Bit.Core.Settings;
|
using Bit.Core.Settings;
|
||||||
|
using Bit.Core.Test.Billing.Stubs;
|
||||||
using Bit.Test.Common.AutoFixture;
|
using Bit.Test.Common.AutoFixture;
|
||||||
using Bit.Test.Common.AutoFixture.Attributes;
|
using Bit.Test.Common.AutoFixture.Attributes;
|
||||||
using Braintree;
|
using Braintree;
|
||||||
@ -1167,7 +1170,9 @@ public class SubscriberServiceTests
|
|||||||
{
|
{
|
||||||
var stripeAdapter = sutProvider.GetDependency<IStripeAdapter>();
|
var stripeAdapter = sutProvider.GetDependency<IStripeAdapter>();
|
||||||
|
|
||||||
stripeAdapter.CustomerGetAsync(provider.GatewayCustomerId)
|
stripeAdapter.CustomerGetAsync(
|
||||||
|
provider.GatewayCustomerId,
|
||||||
|
Arg.Is<CustomerGetOptions>(p => p.Expand.Contains("tax") || p.Expand.Contains("tax_ids")))
|
||||||
.Returns(new Customer
|
.Returns(new Customer
|
||||||
{
|
{
|
||||||
Id = provider.GatewayCustomerId,
|
Id = provider.GatewayCustomerId,
|
||||||
@ -1213,7 +1218,10 @@ public class SubscriberServiceTests
|
|||||||
{
|
{
|
||||||
var stripeAdapter = sutProvider.GetDependency<IStripeAdapter>();
|
var stripeAdapter = sutProvider.GetDependency<IStripeAdapter>();
|
||||||
|
|
||||||
stripeAdapter.CustomerGetAsync(provider.GatewayCustomerId)
|
stripeAdapter.CustomerGetAsync(
|
||||||
|
provider.GatewayCustomerId,
|
||||||
|
Arg.Is<CustomerGetOptions>(p => p.Expand.Contains("tax") || p.Expand.Contains("tax_ids"))
|
||||||
|
)
|
||||||
.Returns(new Customer
|
.Returns(new Customer
|
||||||
{
|
{
|
||||||
Id = provider.GatewayCustomerId,
|
Id = provider.GatewayCustomerId,
|
||||||
@ -1321,7 +1329,9 @@ public class SubscriberServiceTests
|
|||||||
{
|
{
|
||||||
const string braintreeCustomerId = "braintree_customer_id";
|
const string braintreeCustomerId = "braintree_customer_id";
|
||||||
|
|
||||||
sutProvider.GetDependency<IStripeAdapter>().CustomerGetAsync(provider.GatewayCustomerId)
|
sutProvider.GetDependency<IStripeAdapter>().CustomerGetAsync(
|
||||||
|
provider.GatewayCustomerId,
|
||||||
|
Arg.Is<CustomerGetOptions>(p => p.Expand.Contains("tax") || p.Expand.Contains("tax_ids")))
|
||||||
.Returns(new Customer
|
.Returns(new Customer
|
||||||
{
|
{
|
||||||
Id = provider.GatewayCustomerId,
|
Id = provider.GatewayCustomerId,
|
||||||
@ -1373,7 +1383,9 @@ public class SubscriberServiceTests
|
|||||||
{
|
{
|
||||||
const string braintreeCustomerId = "braintree_customer_id";
|
const string braintreeCustomerId = "braintree_customer_id";
|
||||||
|
|
||||||
sutProvider.GetDependency<IStripeAdapter>().CustomerGetAsync(provider.GatewayCustomerId)
|
sutProvider.GetDependency<IStripeAdapter>().CustomerGetAsync(
|
||||||
|
provider.GatewayCustomerId,
|
||||||
|
Arg.Is<CustomerGetOptions>(p => p.Expand.Contains("tax") || p.Expand.Contains("tax_ids")))
|
||||||
.Returns(new Customer
|
.Returns(new Customer
|
||||||
{
|
{
|
||||||
Id = provider.GatewayCustomerId,
|
Id = provider.GatewayCustomerId,
|
||||||
@ -1482,7 +1494,9 @@ public class SubscriberServiceTests
|
|||||||
{
|
{
|
||||||
const string braintreeCustomerId = "braintree_customer_id";
|
const string braintreeCustomerId = "braintree_customer_id";
|
||||||
|
|
||||||
sutProvider.GetDependency<IStripeAdapter>().CustomerGetAsync(provider.GatewayCustomerId)
|
sutProvider.GetDependency<IStripeAdapter>().CustomerGetAsync(
|
||||||
|
provider.GatewayCustomerId,
|
||||||
|
Arg.Is<CustomerGetOptions>(p => p.Expand.Contains("tax") || p.Expand.Contains("tax_ids")))
|
||||||
.Returns(new Customer
|
.Returns(new Customer
|
||||||
{
|
{
|
||||||
Id = provider.GatewayCustomerId
|
Id = provider.GatewayCustomerId
|
||||||
@ -1561,6 +1575,37 @@ public class SubscriberServiceTests
|
|||||||
"Example Town",
|
"Example Town",
|
||||||
"NY");
|
"NY");
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IStripeAdapter>()
|
||||||
|
.CustomerUpdateAsync(
|
||||||
|
Arg.Is<string>(p => p == provider.GatewayCustomerId),
|
||||||
|
Arg.Is<CustomerUpdateOptions>(options =>
|
||||||
|
options.Address.Country == "US" &&
|
||||||
|
options.Address.PostalCode == "12345" &&
|
||||||
|
options.Address.Line1 == "123 Example St." &&
|
||||||
|
options.Address.Line2 == null &&
|
||||||
|
options.Address.City == "Example Town" &&
|
||||||
|
options.Address.State == "NY"))
|
||||||
|
.Returns(new Customer
|
||||||
|
{
|
||||||
|
Id = provider.GatewayCustomerId,
|
||||||
|
Address = new Address
|
||||||
|
{
|
||||||
|
Country = "US",
|
||||||
|
PostalCode = "12345",
|
||||||
|
Line1 = "123 Example St.",
|
||||||
|
Line2 = null,
|
||||||
|
City = "Example Town",
|
||||||
|
State = "NY"
|
||||||
|
},
|
||||||
|
TaxIds = new StripeList<TaxId> { Data = [new TaxId { Id = "tax_id_1", Type = "us_ein" }] }
|
||||||
|
});
|
||||||
|
|
||||||
|
var subscription = new Subscription { Items = new StripeList<SubscriptionItem>() };
|
||||||
|
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(Arg.Any<string>())
|
||||||
|
.Returns(subscription);
|
||||||
|
sutProvider.GetDependency<IAutomaticTaxFactory>().CreateAsync(Arg.Any<AutomaticTaxFactoryParameters>())
|
||||||
|
.Returns(new FakeAutomaticTaxStrategy(true));
|
||||||
|
|
||||||
await sutProvider.Sut.UpdateTaxInformation(provider, taxInformation);
|
await sutProvider.Sut.UpdateTaxInformation(provider, taxInformation);
|
||||||
|
|
||||||
await stripeAdapter.Received(1).CustomerUpdateAsync(provider.GatewayCustomerId, Arg.Is<CustomerUpdateOptions>(
|
await stripeAdapter.Received(1).CustomerUpdateAsync(provider.GatewayCustomerId, Arg.Is<CustomerUpdateOptions>(
|
||||||
|
35
test/Core.Test/Billing/Stubs/FakeAutomaticTaxStrategy.cs
Normal file
35
test/Core.Test/Billing/Stubs/FakeAutomaticTaxStrategy.cs
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
using Bit.Core.Billing.Services;
|
||||||
|
using Stripe;
|
||||||
|
|
||||||
|
namespace Bit.Core.Test.Billing.Stubs;
|
||||||
|
|
||||||
|
/// <param name="isAutomaticTaxEnabled">
|
||||||
|
/// Whether the subscription options will have automatic tax enabled or not.
|
||||||
|
/// </param>
|
||||||
|
public class FakeAutomaticTaxStrategy(
|
||||||
|
bool isAutomaticTaxEnabled) : IAutomaticTaxStrategy
|
||||||
|
{
|
||||||
|
public SubscriptionUpdateOptions? GetUpdateOptions(Subscription subscription)
|
||||||
|
{
|
||||||
|
return new SubscriptionUpdateOptions
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTaxOptions { Enabled = isAutomaticTaxEnabled }
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetCreateOptions(SubscriptionCreateOptions options, Customer customer)
|
||||||
|
{
|
||||||
|
options.AutomaticTax = new SubscriptionAutomaticTaxOptions { Enabled = isAutomaticTaxEnabled };
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetUpdateOptions(SubscriptionUpdateOptions options, Subscription subscription)
|
||||||
|
{
|
||||||
|
options.AutomaticTax = new SubscriptionAutomaticTaxOptions { Enabled = isAutomaticTaxEnabled };
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetInvoiceCreatePreviewOptions(InvoiceCreatePreviewOptions options)
|
||||||
|
{
|
||||||
|
options.AutomaticTax = new InvoiceAutomaticTaxOptions { Enabled = isAutomaticTaxEnabled };
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user