diff --git a/src/Core/Billing/Services/IAutomaticTaxStrategy.cs b/src/Core/Billing/Services/IAutomaticTaxStrategy.cs index b826e43672..292f2d0939 100644 --- a/src/Core/Billing/Services/IAutomaticTaxStrategy.cs +++ b/src/Core/Billing/Services/IAutomaticTaxStrategy.cs @@ -28,4 +28,6 @@ public interface IAutomaticTaxStrategy /// /// void SetUpdateOptions(SubscriptionUpdateOptions options, Subscription subscription); + + void SetInvoiceCreatePreviewOptions(InvoiceCreatePreviewOptions options); } diff --git a/src/Core/Billing/Services/Implementations/AutomaticTax/BusinessUseAutomaticTaxStrategy.cs b/src/Core/Billing/Services/Implementations/AutomaticTax/BusinessUseAutomaticTaxStrategy.cs index 9f516bff77..670822afcd 100644 --- a/src/Core/Billing/Services/Implementations/AutomaticTax/BusinessUseAutomaticTaxStrategy.cs +++ b/src/Core/Billing/Services/Implementations/AutomaticTax/BusinessUseAutomaticTaxStrategy.cs @@ -61,6 +61,19 @@ public class BusinessUseAutomaticTaxStrategy(IFeatureService featureService) : I 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()) diff --git a/src/Core/Billing/Services/Implementations/AutomaticTax/PersonalUseAutomaticTaxStrategy.cs b/src/Core/Billing/Services/Implementations/AutomaticTax/PersonalUseAutomaticTaxStrategy.cs index 7b1b021910..15ee1adf8f 100644 --- a/src/Core/Billing/Services/Implementations/AutomaticTax/PersonalUseAutomaticTaxStrategy.cs +++ b/src/Core/Billing/Services/Implementations/AutomaticTax/PersonalUseAutomaticTaxStrategy.cs @@ -52,6 +52,11 @@ public class PersonalUseAutomaticTaxStrategy(IFeatureService featureService) : I return options; } + public void SetInvoiceCreatePreviewOptions(InvoiceCreatePreviewOptions options) + { + options.AutomaticTax = new InvoiceAutomaticTaxOptions { Enabled = true }; + } + private static bool ShouldBeEnabled(Customer customer) { return customer.HasTaxLocationVerified(); diff --git a/src/Core/Services/Implementations/StripePaymentService.cs b/src/Core/Services/Implementations/StripePaymentService.cs index cdc818e8c1..f79c2f651c 100644 --- a/src/Core/Services/Implementations/StripePaymentService.cs +++ b/src/Core/Services/Implementations/StripePaymentService.cs @@ -10,6 +10,7 @@ using Bit.Core.Billing.Models.Business; using Bit.Core.Billing.Pricing; 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.Enums; using Bit.Core.Exceptions; @@ -17,6 +18,7 @@ using Bit.Core.Models.BitStripe; using Bit.Core.Models.Business; using Bit.Core.Repositories; using Bit.Core.Settings; +using Microsoft.Extensions.DependencyInjection; using Microsoft.Extensions.Logging; using Stripe; using PaymentMethod = Stripe.PaymentMethod; @@ -38,6 +40,7 @@ public class StripePaymentService : IPaymentService private readonly ISubscriberService _subscriberService; private readonly IPricingClient _pricingClient; private readonly IAutomaticTaxFactory _automaticTaxFactory; + private readonly IAutomaticTaxStrategy _personalUseTaxStrategy; public StripePaymentService( ITransactionRepository transactionRepository, @@ -49,7 +52,8 @@ public class StripePaymentService : IPaymentService ITaxService taxService, ISubscriberService subscriberService, IPricingClient pricingClient, - IAutomaticTaxFactory automaticTaxFactory) + IAutomaticTaxFactory automaticTaxFactory, + [FromKeyedServices(AutomaticTaxFactory.PersonalUse)] IAutomaticTaxStrategy personalUseTaxStrategy) { _transactionRepository = transactionRepository; _logger = logger; @@ -61,6 +65,7 @@ public class StripePaymentService : IPaymentService _subscriberService = subscriberService; _pricingClient = pricingClient; _automaticTaxFactory = automaticTaxFactory; + _personalUseTaxStrategy = personalUseTaxStrategy; } private async Task ChangeOrganizationSponsorship( @@ -1251,6 +1256,8 @@ public class StripePaymentService : IPaymentService } } + _personalUseTaxStrategy.SetInvoiceCreatePreviewOptions(options); + try { var invoice = await _stripeAdapter.InvoiceCreatePreviewAsync(options); @@ -1293,10 +1300,6 @@ public class StripePaymentService : IPaymentService var options = new InvoiceCreatePreviewOptions { - AutomaticTax = new InvoiceAutomaticTaxOptions - { - Enabled = true, - }, Currency = "usd", SubscriptionDetails = new InvoiceSubscriptionDetailsOptions { @@ -1384,9 +1387,11 @@ public class StripePaymentService : IPaymentService ]; } + Customer gatewayCustomer = null; + if (!string.IsNullOrWhiteSpace(gatewayCustomerId)) { - var gatewayCustomer = await _stripeAdapter.CustomerGetAsync(gatewayCustomerId); + gatewayCustomer = await _stripeAdapter.CustomerGetAsync(gatewayCustomerId); if (gatewayCustomer.Discount != null) { @@ -1404,6 +1409,10 @@ public class StripePaymentService : IPaymentService } } + var automaticTaxFactoryParameters = new AutomaticTaxFactoryParameters(parameters.PasswordManager.Plan); + var automaticTaxStrategy = await _automaticTaxFactory.CreateAsync(automaticTaxFactoryParameters); + automaticTaxStrategy.SetInvoiceCreatePreviewOptions(options); + try { var invoice = await _stripeAdapter.InvoiceCreatePreviewAsync(options);