mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 21:18:13 -05:00
WIP
This commit is contained in:
parent
39d344f742
commit
3eada56467
@ -7,10 +7,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,7 +30,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 IOrganizationAutomaticTaxStrategy _organizationAutomaticTaxStrategy;
|
private readonly IAutomaticTaxStrategy _automaticTaxStrategy;
|
||||||
|
|
||||||
public RemoveOrganizationFromProviderCommand(
|
public RemoveOrganizationFromProviderCommand(
|
||||||
IEventService eventService,
|
IEventService eventService,
|
||||||
@ -42,7 +44,7 @@ public class RemoveOrganizationFromProviderCommand : IRemoveOrganizationFromProv
|
|||||||
ISubscriberService subscriberService,
|
ISubscriberService subscriberService,
|
||||||
IHasConfirmedOwnersExceptQuery hasConfirmedOwnersExceptQuery,
|
IHasConfirmedOwnersExceptQuery hasConfirmedOwnersExceptQuery,
|
||||||
IPricingClient pricingClient,
|
IPricingClient pricingClient,
|
||||||
IOrganizationAutomaticTaxStrategy organizationAutomaticTaxStrategy)
|
[FromKeyedServices(AutomaticTaxFactory.BusinessUse)] IAutomaticTaxStrategy automaticTaxStrategy)
|
||||||
{
|
{
|
||||||
_eventService = eventService;
|
_eventService = eventService;
|
||||||
_mailService = mailService;
|
_mailService = mailService;
|
||||||
@ -55,7 +57,7 @@ public class RemoveOrganizationFromProviderCommand : IRemoveOrganizationFromProv
|
|||||||
_subscriberService = subscriberService;
|
_subscriberService = subscriberService;
|
||||||
_hasConfirmedOwnersExceptQuery = hasConfirmedOwnersExceptQuery;
|
_hasConfirmedOwnersExceptQuery = hasConfirmedOwnersExceptQuery;
|
||||||
_pricingClient = pricingClient;
|
_pricingClient = pricingClient;
|
||||||
_organizationAutomaticTaxStrategy = organizationAutomaticTaxStrategy;
|
_automaticTaxStrategy = automaticTaxStrategy;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task RemoveOrganizationFromProvider(
|
public async Task RemoveOrganizationFromProvider(
|
||||||
@ -132,7 +134,7 @@ public class RemoveOrganizationFromProviderCommand : IRemoveOrganizationFromProv
|
|||||||
Items = [new SubscriptionItemOptions { Price = plan.PasswordManager.StripeSeatPlanId, Quantity = organization.Seats }]
|
Items = [new SubscriptionItemOptions { Price = plan.PasswordManager.StripeSeatPlanId, Quantity = organization.Seats }]
|
||||||
};
|
};
|
||||||
|
|
||||||
await _organizationAutomaticTaxStrategy.SetCreateOptionsAsync(subscriptionCreateOptions, customer);
|
_automaticTaxStrategy.SetCreateOptions(subscriptionCreateOptions, customer);
|
||||||
|
|
||||||
var subscription = await _stripeAdapter.SubscriptionCreateAsync(subscriptionCreateOptions);
|
var subscription = await _stripeAdapter.SubscriptionCreateAsync(subscriptionCreateOptions);
|
||||||
|
|
||||||
|
@ -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;
|
||||||
|
|
||||||
@ -41,7 +43,8 @@ public class ProviderBillingService(
|
|||||||
IStripeAdapter stripeAdapter,
|
IStripeAdapter stripeAdapter,
|
||||||
ISubscriberService subscriberService,
|
ISubscriberService subscriberService,
|
||||||
ITaxService taxService,
|
ITaxService taxService,
|
||||||
IOrganizationAutomaticTaxStrategy organizationAutomaticTaxStrategy) : IProviderBillingService
|
[FromKeyedServices(AutomaticTaxFactory.BusinessUse)] IAutomaticTaxStrategy automaticTaxStrategy)
|
||||||
|
: IProviderBillingService
|
||||||
{
|
{
|
||||||
[RequireFeature(FeatureFlagKeys.P15179_AddExistingOrgsFromProviderPortal)]
|
[RequireFeature(FeatureFlagKeys.P15179_AddExistingOrgsFromProviderPortal)]
|
||||||
public async Task AddExistingOrganization(
|
public async Task AddExistingOrganization(
|
||||||
@ -602,7 +605,7 @@ public class ProviderBillingService(
|
|||||||
ProrationBehavior = StripeConstants.ProrationBehavior.CreateProrations
|
ProrationBehavior = StripeConstants.ProrationBehavior.CreateProrations
|
||||||
};
|
};
|
||||||
|
|
||||||
await organizationAutomaticTaxStrategy.SetCreateOptionsAsync(subscriptionCreateOptions, customer);
|
automaticTaxStrategy.SetCreateOptions(subscriptionCreateOptions, customer);
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
@ -228,8 +228,8 @@ public class RemoveOrganizationFromProviderCommandTests
|
|||||||
Id = "subscription_id"
|
Id = "subscription_id"
|
||||||
});
|
});
|
||||||
|
|
||||||
sutProvider.GetDependency<IOrganizationAutomaticTaxStrategy>()
|
sutProvider.GetDependency<IAutomaticTaxStrategy>()
|
||||||
.When(x => x.SetCreateOptionsAsync(
|
.When(x => x.SetCreateOptions(
|
||||||
Arg.Is<SubscriptionCreateOptions>(options =>
|
Arg.Is<SubscriptionCreateOptions>(options =>
|
||||||
options.Customer == organization.GatewayCustomerId &&
|
options.Customer == organization.GatewayCustomerId &&
|
||||||
options.CollectionMethod == StripeConstants.CollectionMethod.SendInvoice &&
|
options.CollectionMethod == StripeConstants.CollectionMethod.SendInvoice &&
|
||||||
|
@ -1018,8 +1018,8 @@ 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<IOrganizationAutomaticTaxStrategy>()
|
sutProvider.GetDependency<IAutomaticTaxStrategy>()
|
||||||
.When(x => x.SetCreateOptionsAsync(
|
.When(x => x.SetCreateOptions(
|
||||||
Arg.Is<SubscriptionCreateOptions>(options =>
|
Arg.Is<SubscriptionCreateOptions>(options =>
|
||||||
options.Customer == "customer_id")
|
options.Customer == "customer_id")
|
||||||
, Arg.Is<Customer>(p => p == customer)))
|
, Arg.Is<Customer>(p => p == customer)))
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
using Bit.Core.AdminConsole.Repositories;
|
using Bit.Core.AdminConsole.Repositories;
|
||||||
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.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;
|
||||||
@ -21,8 +21,7 @@ public class UpcomingInvoiceHandler(
|
|||||||
IStripeEventUtilityService stripeEventUtilityService,
|
IStripeEventUtilityService stripeEventUtilityService,
|
||||||
IUserRepository userRepository,
|
IUserRepository userRepository,
|
||||||
IValidateSponsorshipCommand validateSponsorshipCommand,
|
IValidateSponsorshipCommand validateSponsorshipCommand,
|
||||||
IIndividualAutomaticTaxStrategy individualAutomaticTaxStrategy,
|
IAutomaticTaxFactory automaticTaxFactory)
|
||||||
IOrganizationAutomaticTaxStrategy organizationAutomaticTaxStrategy)
|
|
||||||
: IUpcomingInvoiceHandler
|
: IUpcomingInvoiceHandler
|
||||||
{
|
{
|
||||||
public async Task HandleAsync(Event parsedEvent)
|
public async Task HandleAsync(Event parsedEvent)
|
||||||
@ -137,9 +136,9 @@ public class UpcomingInvoiceHandler(
|
|||||||
|
|
||||||
private async Task TryEnableAutomaticTaxAsync(Subscription subscription)
|
private async Task TryEnableAutomaticTaxAsync(Subscription subscription)
|
||||||
{
|
{
|
||||||
var updateOptions = subscription.IsOrganization()
|
var automaticTaxParameters = new AutomaticTaxFactoryParameters(subscription.Items.Select(x => x.Price.Id));
|
||||||
? await organizationAutomaticTaxStrategy.GetUpdateOptionsAsync(subscription)
|
var automaticTaxStrategy = await automaticTaxFactory.CreateAsync(automaticTaxParameters);
|
||||||
: individualAutomaticTaxStrategy.GetUpdateOptions(subscription);
|
var updateOptions = automaticTaxStrategy.GetUpdateOptions(subscription);
|
||||||
|
|
||||||
if (updateOptions == null)
|
if (updateOptions == null)
|
||||||
{
|
{
|
||||||
|
@ -19,8 +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.AddTransient<IIndividualAutomaticTaxStrategy, IndividualAutomaticTaxStrategy>();
|
services.AddKeyedTransient<IAutomaticTaxStrategy, PersonalUseAutomaticTaxStrategy>(AutomaticTaxFactory.PersonalUse);
|
||||||
services.AddTransient<IOrganizationAutomaticTaxStrategy, OrganizationAutomaticTaxStrategy>();
|
services.AddKeyedTransient<IAutomaticTaxStrategy, BusinessUseAutomaticTaxStrategy>(AutomaticTaxFactory.BusinessUse);
|
||||||
|
services.AddTransient<IAutomaticTaxFactory, AutomaticTaxFactory>();
|
||||||
services.AddLicenseServices();
|
services.AddLicenseServices();
|
||||||
services.AddPricingClient();
|
services.AddPricingClient();
|
||||||
}
|
}
|
||||||
|
@ -5,7 +5,6 @@ using Bit.Core.Billing.Enums;
|
|||||||
using Bit.Core.Billing.Migration.Models;
|
using Bit.Core.Billing.Migration.Models;
|
||||||
using Bit.Core.Billing.Pricing;
|
using Bit.Core.Billing.Pricing;
|
||||||
using Bit.Core.Billing.Repositories;
|
using Bit.Core.Billing.Repositories;
|
||||||
using Bit.Core.Billing.Services;
|
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.Repositories;
|
using Bit.Core.Repositories;
|
||||||
using Bit.Core.Services;
|
using Bit.Core.Services;
|
||||||
@ -21,8 +20,7 @@ public class OrganizationMigrator(
|
|||||||
IMigrationTrackerCache migrationTrackerCache,
|
IMigrationTrackerCache migrationTrackerCache,
|
||||||
IOrganizationRepository organizationRepository,
|
IOrganizationRepository organizationRepository,
|
||||||
IPricingClient pricingClient,
|
IPricingClient pricingClient,
|
||||||
IStripeAdapter stripeAdapter,
|
IStripeAdapter stripeAdapter) : IOrganizationMigrator
|
||||||
IOrganizationAutomaticTaxStrategy automaticTaxStrategy) : IOrganizationMigrator
|
|
||||||
{
|
{
|
||||||
private const string _cancellationComment = "Cancelled as part of provider migration to Consolidated Billing";
|
private const string _cancellationComment = "Cancelled as part of provider migration to Consolidated Billing";
|
||||||
|
|
||||||
@ -160,145 +158,6 @@ public class OrganizationMigrator(
|
|||||||
|
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region Reverse
|
|
||||||
|
|
||||||
private async Task RemoveMigrationRecordAsync(Guid providerId, Organization organization)
|
|
||||||
{
|
|
||||||
logger.LogInformation("CB: Removing migration record for organization ({OrganizationID})", organization.Id);
|
|
||||||
|
|
||||||
var migrationRecord = await clientOrganizationMigrationRecordRepository.GetByOrganizationId(organization.Id);
|
|
||||||
|
|
||||||
if (migrationRecord != null)
|
|
||||||
{
|
|
||||||
await clientOrganizationMigrationRecordRepository.DeleteAsync(migrationRecord);
|
|
||||||
|
|
||||||
logger.LogInformation(
|
|
||||||
"CB: Removed migration record for organization ({OrganizationID})",
|
|
||||||
organization.Id);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
logger.LogInformation("CB: Did not remove migration record for organization ({OrganizationID}) as it does not exist", organization.Id);
|
|
||||||
}
|
|
||||||
|
|
||||||
await migrationTrackerCache.UpdateTrackingStatus(providerId, organization.Id, ClientMigrationProgress.Reversed);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task RecreateSubscriptionAsync(Guid providerId, Organization organization)
|
|
||||||
{
|
|
||||||
logger.LogInformation("CB: Recreating subscription for organization ({OrganizationID})", organization.Id);
|
|
||||||
|
|
||||||
if (!string.IsNullOrEmpty(organization.GatewaySubscriptionId))
|
|
||||||
{
|
|
||||||
if (string.IsNullOrEmpty(organization.GatewayCustomerId))
|
|
||||||
{
|
|
||||||
logger.LogError(
|
|
||||||
"CB: Cannot recreate subscription for organization ({OrganizationID}) as it does not have a Stripe customer",
|
|
||||||
organization.Id);
|
|
||||||
|
|
||||||
throw new Exception();
|
|
||||||
}
|
|
||||||
|
|
||||||
var customer = await stripeAdapter.CustomerGetAsync(organization.GatewayCustomerId,
|
|
||||||
new CustomerGetOptions { Expand = ["default_source", "invoice_settings.default_payment_method"] });
|
|
||||||
|
|
||||||
var collectionMethod =
|
|
||||||
customer.DefaultSource != null ||
|
|
||||||
customer.InvoiceSettings?.DefaultPaymentMethod != null ||
|
|
||||||
customer.Metadata.ContainsKey(Utilities.BraintreeCustomerIdKey)
|
|
||||||
? StripeConstants.CollectionMethod.ChargeAutomatically
|
|
||||||
: StripeConstants.CollectionMethod.SendInvoice;
|
|
||||||
|
|
||||||
var plan = await pricingClient.GetPlanOrThrow(organization.PlanType);
|
|
||||||
|
|
||||||
var items = new List<SubscriptionItemOptions>
|
|
||||||
{
|
|
||||||
new ()
|
|
||||||
{
|
|
||||||
Price = plan.PasswordManager.StripeSeatPlanId,
|
|
||||||
Quantity = organization.Seats
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
if (organization.MaxStorageGb.HasValue && plan.PasswordManager.BaseStorageGb.HasValue && organization.MaxStorageGb.Value > plan.PasswordManager.BaseStorageGb.Value)
|
|
||||||
{
|
|
||||||
var additionalStorage = organization.MaxStorageGb.Value - plan.PasswordManager.BaseStorageGb.Value;
|
|
||||||
|
|
||||||
items.Add(new SubscriptionItemOptions
|
|
||||||
{
|
|
||||||
Price = plan.PasswordManager.StripeStoragePlanId,
|
|
||||||
Quantity = additionalStorage
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
var subscriptionCreateOptions = new SubscriptionCreateOptions
|
|
||||||
{
|
|
||||||
Customer = customer.Id,
|
|
||||||
CollectionMethod = collectionMethod,
|
|
||||||
DaysUntilDue = collectionMethod == StripeConstants.CollectionMethod.SendInvoice ? 30 : null,
|
|
||||||
Items = items,
|
|
||||||
Metadata = new Dictionary<string, string>
|
|
||||||
{
|
|
||||||
[organization.GatewayIdField()] = organization.Id.ToString()
|
|
||||||
},
|
|
||||||
OffSession = true,
|
|
||||||
ProrationBehavior = StripeConstants.ProrationBehavior.CreateProrations,
|
|
||||||
TrialPeriodDays = plan.TrialPeriodDays
|
|
||||||
};
|
|
||||||
|
|
||||||
await automaticTaxStrategy.SetCreateOptionsAsync(subscriptionCreateOptions, customer);
|
|
||||||
|
|
||||||
var subscription = await stripeAdapter.SubscriptionCreateAsync(subscriptionCreateOptions);
|
|
||||||
|
|
||||||
organization.GatewaySubscriptionId = subscription.Id;
|
|
||||||
|
|
||||||
await organizationRepository.ReplaceAsync(organization);
|
|
||||||
|
|
||||||
logger.LogInformation("CB: Recreated subscription for organization ({OrganizationID})", organization.Id);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
logger.LogInformation(
|
|
||||||
"CB: Did not recreate subscription for organization ({OrganizationID}) as it already exists",
|
|
||||||
organization.Id);
|
|
||||||
}
|
|
||||||
|
|
||||||
await migrationTrackerCache.UpdateTrackingStatus(providerId, organization.Id,
|
|
||||||
ClientMigrationProgress.RecreatedSubscription);
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task ReverseOrganizationUpdateAsync(Guid providerId, Organization organization)
|
|
||||||
{
|
|
||||||
var migrationRecord = await clientOrganizationMigrationRecordRepository.GetByOrganizationId(organization.Id);
|
|
||||||
|
|
||||||
if (migrationRecord == null)
|
|
||||||
{
|
|
||||||
logger.LogError(
|
|
||||||
"CB: Cannot reverse migration for organization ({OrganizationID}) as it does not have a migration record",
|
|
||||||
organization.Id);
|
|
||||||
|
|
||||||
throw new Exception();
|
|
||||||
}
|
|
||||||
|
|
||||||
var plan = await pricingClient.GetPlanOrThrow(migrationRecord.PlanType);
|
|
||||||
|
|
||||||
ResetOrganizationPlan(organization, plan);
|
|
||||||
organization.MaxStorageGb = migrationRecord.MaxStorageGb;
|
|
||||||
organization.ExpirationDate = migrationRecord.ExpirationDate;
|
|
||||||
organization.MaxAutoscaleSeats = migrationRecord.MaxAutoscaleSeats;
|
|
||||||
organization.Status = migrationRecord.Status;
|
|
||||||
|
|
||||||
await organizationRepository.ReplaceAsync(organization);
|
|
||||||
|
|
||||||
logger.LogInformation("CB: Reversed organization ({OrganizationID}) updates",
|
|
||||||
organization.Id);
|
|
||||||
|
|
||||||
await migrationTrackerCache.UpdateTrackingStatus(providerId, organization.Id,
|
|
||||||
ClientMigrationProgress.ResetOrganization);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
|
|
||||||
#region Shared
|
#region Shared
|
||||||
|
|
||||||
private static void ResetOrganizationPlan(Organization organization, Plan plan)
|
private static void ResetOrganizationPlan(Organization organization, Plan plan)
|
||||||
|
@ -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; }
|
||||||
|
}
|
8
src/Core/Billing/Services/IAutomaticTaxFactory.cs
Normal file
8
src/Core/Billing/Services/IAutomaticTaxFactory.cs
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
using Bit.Core.Billing.Services.Contracts;
|
||||||
|
|
||||||
|
namespace Bit.Core.Billing.Services;
|
||||||
|
|
||||||
|
public interface IAutomaticTaxFactory
|
||||||
|
{
|
||||||
|
Task<IAutomaticTaxStrategy> CreateAsync(AutomaticTaxFactoryParameters parameters);
|
||||||
|
}
|
@ -3,7 +3,7 @@ using Stripe;
|
|||||||
|
|
||||||
namespace Bit.Core.Billing.Services;
|
namespace Bit.Core.Billing.Services;
|
||||||
|
|
||||||
public interface IIndividualAutomaticTaxStrategy
|
public interface IAutomaticTaxStrategy
|
||||||
{
|
{
|
||||||
/// <summary>
|
/// <summary>
|
||||||
///
|
///
|
@ -1,19 +0,0 @@
|
|||||||
#nullable enable
|
|
||||||
using Stripe;
|
|
||||||
|
|
||||||
namespace Bit.Core.Billing.Services;
|
|
||||||
|
|
||||||
public interface IOrganizationAutomaticTaxStrategy
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
///
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="subscription"></param>
|
|
||||||
/// <returns>
|
|
||||||
/// Returns <see cref="SubscriptionUpdateOptions" /> if changes are to be applied to the subscription, returns null
|
|
||||||
/// otherwise.
|
|
||||||
/// </returns>
|
|
||||||
Task<SubscriptionUpdateOptions?> GetUpdateOptionsAsync(Subscription subscription);
|
|
||||||
Task SetCreateOptionsAsync(SubscriptionCreateOptions options, Customer customer);
|
|
||||||
Task SetUpdateOptionsAsync(SubscriptionUpdateOptions options, Subscription subscription);
|
|
||||||
}
|
|
@ -0,0 +1,48 @@
|
|||||||
|
#nullable enable
|
||||||
|
using Bit.Core.Billing.Enums;
|
||||||
|
using Bit.Core.Billing.Pricing;
|
||||||
|
using Bit.Core.Billing.Services.Contracts;
|
||||||
|
using Bit.Core.Entities;
|
||||||
|
|
||||||
|
namespace Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
||||||
|
|
||||||
|
public class AutomaticTaxFactory(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();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (parameters.PlanType.HasValue)
|
||||||
|
{
|
||||||
|
var plan = await pricingClient.GetPlanOrThrow(parameters.PlanType.Value);
|
||||||
|
return plan.CanBeUsedByBusiness
|
||||||
|
? new BusinessUseAutomaticTaxStrategy()
|
||||||
|
: new PersonalUseAutomaticTaxStrategy();
|
||||||
|
}
|
||||||
|
|
||||||
|
var personalUsePlans = await _personalUsePlansTask.Value;
|
||||||
|
var plans = await pricingClient.ListPlans();
|
||||||
|
|
||||||
|
if (personalUsePlans.Any(x => plans.Any(y => y.PasswordManager.StripePlanId == x)))
|
||||||
|
{
|
||||||
|
return new PersonalUseAutomaticTaxStrategy();
|
||||||
|
}
|
||||||
|
|
||||||
|
return new BusinessUseAutomaticTaxStrategy();
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,62 @@
|
|||||||
|
#nullable enable
|
||||||
|
using Bit.Core.Billing.Extensions;
|
||||||
|
using Stripe;
|
||||||
|
|
||||||
|
namespace Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
||||||
|
|
||||||
|
public class BusinessUseAutomaticTaxStrategy : IAutomaticTaxStrategy
|
||||||
|
{
|
||||||
|
public SubscriptionUpdateOptions? GetUpdateOptions(Subscription subscription)
|
||||||
|
{
|
||||||
|
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)
|
||||||
|
{
|
||||||
|
var shouldBeEnabled = ShouldBeEnabled(subscription.Customer);
|
||||||
|
|
||||||
|
if (subscription.AutomaticTax.Enabled == shouldBeEnabled)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
options.AutomaticTax = new SubscriptionAutomaticTaxOptions
|
||||||
|
{
|
||||||
|
Enabled = shouldBeEnabled
|
||||||
|
};
|
||||||
|
options.DefaultTaxRates = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool ShouldBeEnabled(Customer customer)
|
||||||
|
{
|
||||||
|
if (!customer.HasTaxLocationVerified())
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return customer.Address.Country == "US";
|
||||||
|
}
|
||||||
|
}
|
@ -1,97 +0,0 @@
|
|||||||
#nullable enable
|
|
||||||
using Bit.Core.Billing.Enums;
|
|
||||||
using Bit.Core.Billing.Extensions;
|
|
||||||
using Bit.Core.Billing.Pricing;
|
|
||||||
using Stripe;
|
|
||||||
|
|
||||||
namespace Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
|
||||||
|
|
||||||
public class OrganizationAutomaticTaxStrategy(
|
|
||||||
IPricingClient pricingClient) : IOrganizationAutomaticTaxStrategy
|
|
||||||
{
|
|
||||||
private readonly Lazy<Task<IEnumerable<string>>> _familyPriceIdsTask = 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<SubscriptionUpdateOptions?> GetUpdateOptionsAsync(Subscription subscription)
|
|
||||||
{
|
|
||||||
var shouldBeEnabled = await ShouldBeEnabledAsync(subscription);
|
|
||||||
|
|
||||||
var options = new SubscriptionUpdateOptions
|
|
||||||
{
|
|
||||||
AutomaticTax = new SubscriptionAutomaticTaxOptions
|
|
||||||
{
|
|
||||||
Enabled = shouldBeEnabled
|
|
||||||
},
|
|
||||||
DefaultTaxRates = []
|
|
||||||
};
|
|
||||||
|
|
||||||
return options;
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task SetCreateOptionsAsync(SubscriptionCreateOptions options, Customer customer)
|
|
||||||
{
|
|
||||||
options.AutomaticTax = new SubscriptionAutomaticTaxOptions
|
|
||||||
{
|
|
||||||
Enabled = await ShouldBeEnabledAsync(options, customer)
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task SetUpdateOptionsAsync(SubscriptionUpdateOptions options, Subscription subscription)
|
|
||||||
{
|
|
||||||
var shouldBeEnabled = await ShouldBeEnabledAsync(subscription);
|
|
||||||
|
|
||||||
if (subscription.AutomaticTax.Enabled == shouldBeEnabled)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
options.AutomaticTax = new SubscriptionAutomaticTaxOptions
|
|
||||||
{
|
|
||||||
Enabled = shouldBeEnabled
|
|
||||||
};
|
|
||||||
options.DefaultTaxRates = [];
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task<bool> ShouldBeEnabledAsync(Subscription subscription)
|
|
||||||
{
|
|
||||||
if (!subscription.Customer.HasTaxLocationVerified())
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool shouldBeEnabled;
|
|
||||||
if (subscription.Customer.Address.Country == "US")
|
|
||||||
{
|
|
||||||
shouldBeEnabled = true;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
var familyPriceIds = await _familyPriceIdsTask.Value;
|
|
||||||
shouldBeEnabled = subscription.Items.Select(item => item.Price.Id).Intersect(familyPriceIds).Any();
|
|
||||||
}
|
|
||||||
|
|
||||||
return shouldBeEnabled;
|
|
||||||
}
|
|
||||||
|
|
||||||
private async Task<bool> ShouldBeEnabledAsync(SubscriptionCreateOptions options, Customer customer)
|
|
||||||
{
|
|
||||||
if (!customer.HasTaxLocationVerified())
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (customer.Address.Country == "US")
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
var familyPriceIds = await _familyPriceIdsTask.Value;
|
|
||||||
return options.Items.Select(item => item.Price).Intersect(familyPriceIds).Any();
|
|
||||||
}
|
|
||||||
}
|
|
@ -4,7 +4,7 @@ using Stripe;
|
|||||||
|
|
||||||
namespace Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
namespace Bit.Core.Billing.Services.Implementations.AutomaticTax;
|
||||||
|
|
||||||
public class IndividualAutomaticTaxStrategy : IIndividualAutomaticTaxStrategy
|
public class PersonalUseAutomaticTaxStrategy : IAutomaticTaxStrategy
|
||||||
{
|
{
|
||||||
public void SetCreateOptions(SubscriptionCreateOptions options, Customer customer)
|
public void SetCreateOptions(SubscriptionCreateOptions options, Customer customer)
|
||||||
{
|
{
|
@ -4,6 +4,7 @@ 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.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;
|
||||||
@ -31,7 +32,7 @@ public class OrganizationBillingService(
|
|||||||
IStripeAdapter stripeAdapter,
|
IStripeAdapter stripeAdapter,
|
||||||
ISubscriberService subscriberService,
|
ISubscriberService subscriberService,
|
||||||
ITaxService taxService,
|
ITaxService taxService,
|
||||||
IOrganizationAutomaticTaxStrategy organizationAutomaticTaxStrategy) : IOrganizationBillingService
|
IAutomaticTaxFactory automaticTaxFactory) : IOrganizationBillingService
|
||||||
{
|
{
|
||||||
public async Task Finalize(OrganizationSale sale)
|
public async Task Finalize(OrganizationSale sale)
|
||||||
{
|
{
|
||||||
@ -370,15 +371,6 @@ 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
|
||||||
{
|
{
|
||||||
CollectionMethod = StripeConstants.CollectionMethod.ChargeAutomatically,
|
CollectionMethod = StripeConstants.CollectionMethod.ChargeAutomatically,
|
||||||
@ -392,7 +384,9 @@ public class OrganizationBillingService(
|
|||||||
TrialPeriodDays = subscriptionSetup.SkipTrial ? 0 : plan.TrialPeriodDays
|
TrialPeriodDays = subscriptionSetup.SkipTrial ? 0 : plan.TrialPeriodDays
|
||||||
};
|
};
|
||||||
|
|
||||||
await organizationAutomaticTaxStrategy.SetCreateOptionsAsync(subscriptionCreateOptions, customer);
|
var automaticTaxParameters = new AutomaticTaxFactoryParameters(subscriptionSetup.PlanType);
|
||||||
|
var automaticTaxStrategy = await automaticTaxFactory.CreateAsync(automaticTaxParameters);
|
||||||
|
automaticTaxStrategy.SetCreateOptions(subscriptionCreateOptions, customer);
|
||||||
|
|
||||||
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;
|
||||||
@ -26,7 +28,7 @@ public class PremiumUserBillingService(
|
|||||||
IStripeAdapter stripeAdapter,
|
IStripeAdapter stripeAdapter,
|
||||||
ISubscriberService subscriberService,
|
ISubscriberService subscriberService,
|
||||||
IUserRepository userRepository,
|
IUserRepository userRepository,
|
||||||
IIndividualAutomaticTaxStrategy individualAutomaticTaxStrategy) : IPremiumUserBillingService
|
[FromKeyedServices(AutomaticTaxFactory.PersonalUse)] IAutomaticTaxStrategy automaticTaxStrategy) : IPremiumUserBillingService
|
||||||
{
|
{
|
||||||
public async Task Credit(User user, decimal amount)
|
public async Task Credit(User user, decimal amount)
|
||||||
{
|
{
|
||||||
@ -332,7 +334,7 @@ public class PremiumUserBillingService(
|
|||||||
OffSession = true
|
OffSession = true
|
||||||
};
|
};
|
||||||
|
|
||||||
individualAutomaticTaxStrategy.SetCreateOptions(subscriptionCreateOptions, customer);
|
automaticTaxStrategy.SetCreateOptions(subscriptionCreateOptions, customer);
|
||||||
|
|
||||||
var subscription = await stripeAdapter.SubscriptionCreateAsync(subscriptionCreateOptions);
|
var subscription = await stripeAdapter.SubscriptionCreateAsync(subscriptionCreateOptions);
|
||||||
|
|
||||||
|
@ -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;
|
||||||
@ -25,8 +26,7 @@ public class SubscriberService(
|
|||||||
ISetupIntentCache setupIntentCache,
|
ISetupIntentCache setupIntentCache,
|
||||||
IStripeAdapter stripeAdapter,
|
IStripeAdapter stripeAdapter,
|
||||||
ITaxService taxService,
|
ITaxService taxService,
|
||||||
IIndividualAutomaticTaxStrategy individualAutomaticTaxStrategy,
|
IAutomaticTaxFactory automaticTaxFactory) : ISubscriberService
|
||||||
IOrganizationAutomaticTaxStrategy organizationAutomaticTaxStrategy) : ISubscriberService
|
|
||||||
{
|
{
|
||||||
public async Task CancelSubscription(
|
public async Task CancelSubscription(
|
||||||
ISubscriber subscriber,
|
ISubscriber subscriber,
|
||||||
@ -666,9 +666,9 @@ public class SubscriberService(
|
|||||||
if (!string.IsNullOrEmpty(subscriber.GatewaySubscriptionId))
|
if (!string.IsNullOrEmpty(subscriber.GatewaySubscriptionId))
|
||||||
{
|
{
|
||||||
var subscription = await stripeAdapter.SubscriptionGetAsync(subscriber.GatewaySubscriptionId);
|
var subscription = await stripeAdapter.SubscriptionGetAsync(subscriber.GatewaySubscriptionId);
|
||||||
var automaticTaxOptions = subscriber.IsUser()
|
var automaticTaxParameters = new AutomaticTaxFactoryParameters(subscriber, subscription.Items.Select(x => x.Price.Id));
|
||||||
? individualAutomaticTaxStrategy.GetUpdateOptions(subscription)
|
var automaticTaxStrategy = await automaticTaxFactory.CreateAsync(automaticTaxParameters);
|
||||||
: await organizationAutomaticTaxStrategy.GetUpdateOptionsAsync(subscription);
|
var automaticTaxOptions = automaticTaxStrategy.GetUpdateOptions(subscription);
|
||||||
if (automaticTaxOptions?.AutomaticTax?.Enabled != null)
|
if (automaticTaxOptions?.AutomaticTax?.Enabled != null)
|
||||||
{
|
{
|
||||||
await stripeAdapter.SubscriptionUpdateAsync(subscriber.GatewaySubscriptionId, automaticTaxOptions);
|
await stripeAdapter.SubscriptionUpdateAsync(subscriber.GatewaySubscriptionId, automaticTaxOptions);
|
||||||
|
@ -9,6 +9,7 @@ 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.Entities;
|
using Bit.Core.Entities;
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.Exceptions;
|
using Bit.Core.Exceptions;
|
||||||
@ -36,8 +37,7 @@ 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 IIndividualAutomaticTaxStrategy _individualAutomaticTaxStrategy;
|
private readonly IAutomaticTaxFactory _automaticTaxFactory;
|
||||||
private readonly IOrganizationAutomaticTaxStrategy _organizationAutomaticTaxStrategy;
|
|
||||||
|
|
||||||
public StripePaymentService(
|
public StripePaymentService(
|
||||||
ITransactionRepository transactionRepository,
|
ITransactionRepository transactionRepository,
|
||||||
@ -49,8 +49,7 @@ public class StripePaymentService : IPaymentService
|
|||||||
ITaxService taxService,
|
ITaxService taxService,
|
||||||
ISubscriberService subscriberService,
|
ISubscriberService subscriberService,
|
||||||
IPricingClient pricingClient,
|
IPricingClient pricingClient,
|
||||||
IIndividualAutomaticTaxStrategy individualAutomaticTaxStrategy,
|
IAutomaticTaxFactory automaticTaxFactory)
|
||||||
IOrganizationAutomaticTaxStrategy organizationAutomaticTaxStrategy)
|
|
||||||
{
|
{
|
||||||
_transactionRepository = transactionRepository;
|
_transactionRepository = transactionRepository;
|
||||||
_logger = logger;
|
_logger = logger;
|
||||||
@ -61,8 +60,7 @@ public class StripePaymentService : IPaymentService
|
|||||||
_taxService = taxService;
|
_taxService = taxService;
|
||||||
_subscriberService = subscriberService;
|
_subscriberService = subscriberService;
|
||||||
_pricingClient = pricingClient;
|
_pricingClient = pricingClient;
|
||||||
_individualAutomaticTaxStrategy = individualAutomaticTaxStrategy;
|
_automaticTaxFactory = automaticTaxFactory;
|
||||||
_organizationAutomaticTaxStrategy = organizationAutomaticTaxStrategy;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task ChangeOrganizationSponsorship(
|
private async Task ChangeOrganizationSponsorship(
|
||||||
@ -130,7 +128,9 @@ public class StripePaymentService : IPaymentService
|
|||||||
new SubscriptionPendingInvoiceItemIntervalOptions { Interval = "month" };
|
new SubscriptionPendingInvoiceItemIntervalOptions { Interval = "month" };
|
||||||
}
|
}
|
||||||
|
|
||||||
await _organizationAutomaticTaxStrategy.SetUpdateOptionsAsync(subUpdateOptions, sub);
|
var automaticTaxParameters = new AutomaticTaxFactoryParameters(subscriber, sub.Items.Select(x => x.Price.Id));
|
||||||
|
var automaticTaxStrategy = await _automaticTaxFactory.CreateAsync(automaticTaxParameters);
|
||||||
|
automaticTaxStrategy.SetUpdateOptions(subUpdateOptions, sub);
|
||||||
|
|
||||||
if (!subscriptionUpdate.UpdateNeeded(sub))
|
if (!subscriptionUpdate.UpdateNeeded(sub))
|
||||||
{
|
{
|
||||||
@ -821,9 +821,9 @@ public class StripePaymentService : IPaymentService
|
|||||||
{
|
{
|
||||||
var subscription = await _stripeAdapter.SubscriptionGetAsync(subscriber.GatewaySubscriptionId);
|
var subscription = await _stripeAdapter.SubscriptionGetAsync(subscriber.GatewaySubscriptionId);
|
||||||
|
|
||||||
var subscriptionUpdateOptions = subscriber is User
|
var automaticTaxParameters = new AutomaticTaxFactoryParameters(subscriber, subscription.Items.Select(x => x.Price.Id));
|
||||||
? _individualAutomaticTaxStrategy.GetUpdateOptions(subscription)
|
var automaticTaxStrategy = await _automaticTaxFactory.CreateAsync(automaticTaxParameters);
|
||||||
: await _organizationAutomaticTaxStrategy.GetUpdateOptionsAsync(subscription);
|
var subscriptionUpdateOptions = automaticTaxStrategy.GetUpdateOptions(subscription);
|
||||||
|
|
||||||
if (subscriptionUpdateOptions != null)
|
if (subscriptionUpdateOptions != null)
|
||||||
{
|
{
|
||||||
|
Loading…
x
Reference in New Issue
Block a user