mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 05:00:19 -05:00
Revert "[PM-18028] Attempting to enable automatic tax on customer with invali…" (#5375)
This reverts commit 678d5d5d632447ac3431781d8232971eab713edc.
This commit is contained in:
parent
678d5d5d63
commit
a1ef07ea69
@ -1,7 +1,6 @@
|
|||||||
using Bit.Billing.Constants;
|
using Bit.Billing.Constants;
|
||||||
using Bit.Core.AdminConsole.Entities;
|
using Bit.Core.AdminConsole.Entities;
|
||||||
using Bit.Core.AdminConsole.Repositories;
|
using Bit.Core.AdminConsole.Repositories;
|
||||||
using Bit.Core.Billing.Extensions;
|
|
||||||
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;
|
||||||
@ -161,16 +160,16 @@ public class UpcomingInvoiceHandler : IUpcomingInvoiceHandler
|
|||||||
|
|
||||||
private async Task<Subscription> TryEnableAutomaticTaxAsync(Subscription subscription)
|
private async Task<Subscription> TryEnableAutomaticTaxAsync(Subscription subscription)
|
||||||
{
|
{
|
||||||
var customerGetOptions = new CustomerGetOptions { Expand = ["tax"] };
|
if (subscription.AutomaticTax.Enabled)
|
||||||
var customer = await _stripeFacade.GetCustomer(subscription.CustomerId, customerGetOptions);
|
|
||||||
|
|
||||||
var subscriptionUpdateOptions = new SubscriptionUpdateOptions();
|
|
||||||
|
|
||||||
if (!subscriptionUpdateOptions.EnableAutomaticTax(customer, subscription))
|
|
||||||
{
|
{
|
||||||
return subscription;
|
return subscription;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var subscriptionUpdateOptions = new SubscriptionUpdateOptions
|
||||||
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTaxOptions { Enabled = true }
|
||||||
|
};
|
||||||
|
|
||||||
return await _stripeFacade.UpdateSubscription(subscription.Id, subscriptionUpdateOptions);
|
return await _stripeFacade.UpdateSubscription(subscription.Id, subscriptionUpdateOptions);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,16 +0,0 @@
|
|||||||
using Bit.Core.Billing.Constants;
|
|
||||||
using Stripe;
|
|
||||||
|
|
||||||
namespace Bit.Core.Billing.Extensions;
|
|
||||||
|
|
||||||
public static class CustomerExtensions
|
|
||||||
{
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Determines if a Stripe customer supports automatic tax
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="customer"></param>
|
|
||||||
/// <returns></returns>
|
|
||||||
public static bool HasTaxLocationVerified(this Customer customer) =>
|
|
||||||
customer?.Tax?.AutomaticTax == StripeConstants.AutomaticTaxStatus.Supported;
|
|
||||||
}
|
|
@ -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;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,35 +0,0 @@
|
|||||||
using Stripe;
|
|
||||||
|
|
||||||
namespace Bit.Core.Billing.Extensions;
|
|
||||||
|
|
||||||
public static class SubscriptionUpdateOptionsExtensions
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Attempts to enable automatic tax for given subscription options.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="options"></param>
|
|
||||||
/// <param name="customer">The existing customer to which the subscription belongs.</param>
|
|
||||||
/// <param name="subscription">The existing subscription.</param>
|
|
||||||
/// <returns>Returns true when successful, false when conditions are not met.</returns>
|
|
||||||
public static bool EnableAutomaticTax(
|
|
||||||
this SubscriptionUpdateOptions options,
|
|
||||||
Customer customer,
|
|
||||||
Subscription subscription)
|
|
||||||
{
|
|
||||||
if (subscription.AutomaticTax.Enabled)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 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;
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,35 +0,0 @@
|
|||||||
using Stripe;
|
|
||||||
|
|
||||||
namespace Bit.Core.Billing.Extensions;
|
|
||||||
|
|
||||||
public static class UpcomingInvoiceOptionsExtensions
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Attempts to enable automatic tax for given upcoming invoice options.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="options"></param>
|
|
||||||
/// <param name="customer">The existing customer to which the upcoming invoice belongs.</param>
|
|
||||||
/// <param name="subscription">The existing subscription to which the upcoming invoice belongs.</param>
|
|
||||||
/// <returns>Returns true when successful, false when conditions are not met.</returns>
|
|
||||||
public static bool EnableAutomaticTax(
|
|
||||||
this UpcomingInvoiceOptions options,
|
|
||||||
Customer customer,
|
|
||||||
Subscription subscription)
|
|
||||||
{
|
|
||||||
if (subscription != null && subscription.AutomaticTax.Enabled)
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// We might only need to check the automatic tax status.
|
|
||||||
if (!customer.HasTaxLocationVerified() && string.IsNullOrWhiteSpace(customer.Address?.Country))
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
options.AutomaticTax = new InvoiceAutomaticTaxOptions { Enabled = true };
|
|
||||||
options.SubscriptionDefaultTaxRates = [];
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
@ -258,7 +258,7 @@ public class PremiumUserBillingService(
|
|||||||
{
|
{
|
||||||
AutomaticTax = new SubscriptionAutomaticTaxOptions
|
AutomaticTax = new SubscriptionAutomaticTaxOptions
|
||||||
{
|
{
|
||||||
Enabled = customer.Tax?.AutomaticTax == StripeConstants.AutomaticTaxStatus.Supported,
|
Enabled = true
|
||||||
},
|
},
|
||||||
CollectionMethod = StripeConstants.CollectionMethod.ChargeAutomatically,
|
CollectionMethod = StripeConstants.CollectionMethod.ChargeAutomatically,
|
||||||
Customer = customer.Id,
|
Customer = customer.Id,
|
||||||
|
@ -661,21 +661,11 @@ public class SubscriberService(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (SubscriberIsEligibleForAutomaticTax(subscriber, customer))
|
await stripeAdapter.SubscriptionUpdateAsync(subscriber.GatewaySubscriptionId,
|
||||||
{
|
new SubscriptionUpdateOptions
|
||||||
await stripeAdapter.SubscriptionUpdateAsync(subscriber.GatewaySubscriptionId,
|
{
|
||||||
new SubscriptionUpdateOptions
|
AutomaticTax = new SubscriptionAutomaticTaxOptions { Enabled = true },
|
||||||
{
|
});
|
||||||
AutomaticTax = new SubscriptionAutomaticTaxOptions { Enabled = true }
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
return;
|
|
||||||
|
|
||||||
bool SubscriberIsEligibleForAutomaticTax(ISubscriber localSubscriber, Customer localCustomer)
|
|
||||||
=> !string.IsNullOrEmpty(localSubscriber.GatewaySubscriptionId) &&
|
|
||||||
(localCustomer.Subscriptions?.Any(sub => sub.Id == localSubscriber.GatewaySubscriptionId && !sub.AutomaticTax.Enabled) ?? false) &&
|
|
||||||
localCustomer.Tax?.AutomaticTax == StripeConstants.AutomaticTaxStatus.Supported;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task VerifyBankAccount(
|
public async Task VerifyBankAccount(
|
||||||
|
@ -177,7 +177,7 @@ public class StripePaymentService : IPaymentService
|
|||||||
customer = await _stripeAdapter.CustomerCreateAsync(customerCreateOptions);
|
customer = await _stripeAdapter.CustomerCreateAsync(customerCreateOptions);
|
||||||
subCreateOptions.AddExpand("latest_invoice.payment_intent");
|
subCreateOptions.AddExpand("latest_invoice.payment_intent");
|
||||||
subCreateOptions.Customer = customer.Id;
|
subCreateOptions.Customer = customer.Id;
|
||||||
subCreateOptions.EnableAutomaticTax(customer);
|
subCreateOptions.AutomaticTax = new SubscriptionAutomaticTaxOptions { Enabled = true };
|
||||||
|
|
||||||
subscription = await _stripeAdapter.SubscriptionCreateAsync(subCreateOptions);
|
subscription = await _stripeAdapter.SubscriptionCreateAsync(subCreateOptions);
|
||||||
if (subscription.Status == "incomplete" && subscription.LatestInvoice?.PaymentIntent != null)
|
if (subscription.Status == "incomplete" && subscription.LatestInvoice?.PaymentIntent != null)
|
||||||
@ -358,9 +358,10 @@ public class StripePaymentService : IPaymentService
|
|||||||
customer = await _stripeAdapter.CustomerUpdateAsync(org.GatewayCustomerId, customerUpdateOptions);
|
customer = await _stripeAdapter.CustomerUpdateAsync(org.GatewayCustomerId, customerUpdateOptions);
|
||||||
}
|
}
|
||||||
|
|
||||||
var subCreateOptions = new OrganizationUpgradeSubscriptionOptions(customer.Id, org, plan, upgrade);
|
var subCreateOptions = new OrganizationUpgradeSubscriptionOptions(customer.Id, org, plan, upgrade)
|
||||||
|
{
|
||||||
subCreateOptions.EnableAutomaticTax(customer);
|
AutomaticTax = new SubscriptionAutomaticTaxOptions { Enabled = true }
|
||||||
|
};
|
||||||
|
|
||||||
var (stripePaymentMethod, paymentMethodType) = IdentifyPaymentMethod(customer, subCreateOptions);
|
var (stripePaymentMethod, paymentMethodType) = IdentifyPaymentMethod(customer, subCreateOptions);
|
||||||
|
|
||||||
@ -519,6 +520,10 @@ public class StripePaymentService : IPaymentService
|
|||||||
|
|
||||||
var customerCreateOptions = new CustomerCreateOptions
|
var customerCreateOptions = new CustomerCreateOptions
|
||||||
{
|
{
|
||||||
|
Tax = new CustomerTaxOptions
|
||||||
|
{
|
||||||
|
ValidateLocation = StripeConstants.ValidateTaxLocationTiming.Immediately
|
||||||
|
},
|
||||||
Description = user.Name,
|
Description = user.Name,
|
||||||
Email = user.Email,
|
Email = user.Email,
|
||||||
Metadata = stripeCustomerMetadata,
|
Metadata = stripeCustomerMetadata,
|
||||||
@ -556,6 +561,7 @@ public class StripePaymentService : IPaymentService
|
|||||||
|
|
||||||
var subCreateOptions = new SubscriptionCreateOptions
|
var subCreateOptions = new SubscriptionCreateOptions
|
||||||
{
|
{
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTaxOptions { Enabled = true },
|
||||||
Customer = customer.Id,
|
Customer = customer.Id,
|
||||||
Items = [],
|
Items = [],
|
||||||
Metadata = new Dictionary<string, string>
|
Metadata = new Dictionary<string, string>
|
||||||
@ -575,12 +581,10 @@ public class StripePaymentService : IPaymentService
|
|||||||
subCreateOptions.Items.Add(new SubscriptionItemOptions
|
subCreateOptions.Items.Add(new SubscriptionItemOptions
|
||||||
{
|
{
|
||||||
Plan = StoragePlanId,
|
Plan = StoragePlanId,
|
||||||
Quantity = additionalStorageGb
|
Quantity = additionalStorageGb,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
subCreateOptions.EnableAutomaticTax(customer);
|
|
||||||
|
|
||||||
var subscription = await ChargeForNewSubscriptionAsync(user, customer, createdStripeCustomer,
|
var subscription = await ChargeForNewSubscriptionAsync(user, customer, createdStripeCustomer,
|
||||||
stripePaymentMethod, paymentMethodType, subCreateOptions, braintreeCustomer);
|
stripePaymentMethod, paymentMethodType, subCreateOptions, braintreeCustomer);
|
||||||
|
|
||||||
@ -618,10 +622,7 @@ public class StripePaymentService : IPaymentService
|
|||||||
SubscriptionItems = ToInvoiceSubscriptionItemOptions(subCreateOptions.Items)
|
SubscriptionItems = ToInvoiceSubscriptionItemOptions(subCreateOptions.Items)
|
||||||
});
|
});
|
||||||
|
|
||||||
if (customer.HasTaxLocationVerified())
|
previewInvoice.AutomaticTax = new InvoiceAutomaticTax { Enabled = true };
|
||||||
{
|
|
||||||
previewInvoice.AutomaticTax = new InvoiceAutomaticTax { Enabled = true };
|
|
||||||
}
|
|
||||||
|
|
||||||
if (previewInvoice.AmountDue > 0)
|
if (previewInvoice.AmountDue > 0)
|
||||||
{
|
{
|
||||||
@ -679,10 +680,12 @@ public class StripePaymentService : IPaymentService
|
|||||||
Customer = customer.Id,
|
Customer = customer.Id,
|
||||||
SubscriptionItems = ToInvoiceSubscriptionItemOptions(subCreateOptions.Items),
|
SubscriptionItems = ToInvoiceSubscriptionItemOptions(subCreateOptions.Items),
|
||||||
SubscriptionDefaultTaxRates = subCreateOptions.DefaultTaxRates,
|
SubscriptionDefaultTaxRates = subCreateOptions.DefaultTaxRates,
|
||||||
|
AutomaticTax = new InvoiceAutomaticTaxOptions
|
||||||
|
{
|
||||||
|
Enabled = true
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
upcomingInvoiceOptions.EnableAutomaticTax(customer, null);
|
|
||||||
|
|
||||||
var previewInvoice = await _stripeAdapter.InvoiceUpcomingAsync(upcomingInvoiceOptions);
|
var previewInvoice = await _stripeAdapter.InvoiceUpcomingAsync(upcomingInvoiceOptions);
|
||||||
|
|
||||||
if (previewInvoice.AmountDue > 0)
|
if (previewInvoice.AmountDue > 0)
|
||||||
@ -801,7 +804,11 @@ public class StripePaymentService : IPaymentService
|
|||||||
Items = updatedItemOptions,
|
Items = updatedItemOptions,
|
||||||
ProrationBehavior = invoiceNow ? Constants.AlwaysInvoice : Constants.CreateProrations,
|
ProrationBehavior = invoiceNow ? Constants.AlwaysInvoice : Constants.CreateProrations,
|
||||||
DaysUntilDue = daysUntilDue ?? 1,
|
DaysUntilDue = daysUntilDue ?? 1,
|
||||||
CollectionMethod = "send_invoice"
|
CollectionMethod = "send_invoice",
|
||||||
|
AutomaticTax = new SubscriptionAutomaticTaxOptions
|
||||||
|
{
|
||||||
|
Enabled = true
|
||||||
|
}
|
||||||
};
|
};
|
||||||
if (!invoiceNow && isAnnualPlan && sub.Status.Trim() != "trialing")
|
if (!invoiceNow && isAnnualPlan && sub.Status.Trim() != "trialing")
|
||||||
{
|
{
|
||||||
@ -809,8 +816,6 @@ public class StripePaymentService : IPaymentService
|
|||||||
new SubscriptionPendingInvoiceItemIntervalOptions { Interval = "month" };
|
new SubscriptionPendingInvoiceItemIntervalOptions { Interval = "month" };
|
||||||
}
|
}
|
||||||
|
|
||||||
subUpdateOptions.EnableAutomaticTax(sub.Customer, sub);
|
|
||||||
|
|
||||||
if (!subscriptionUpdate.UpdateNeeded(sub))
|
if (!subscriptionUpdate.UpdateNeeded(sub))
|
||||||
{
|
{
|
||||||
// No need to update subscription, quantity matches
|
// No need to update subscription, quantity matches
|
||||||
@ -1495,13 +1500,11 @@ public class StripePaymentService : IPaymentService
|
|||||||
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 &&
|
||||||
!sub.AutomaticTax.Enabled) &&
|
!sub.AutomaticTax.Enabled))
|
||||||
customer.HasTaxLocationVerified())
|
|
||||||
{
|
{
|
||||||
var subscriptionUpdateOptions = new SubscriptionUpdateOptions
|
var subscriptionUpdateOptions = new SubscriptionUpdateOptions
|
||||||
{
|
{
|
||||||
AutomaticTax = new SubscriptionAutomaticTaxOptions { Enabled = true },
|
AutomaticTax = new SubscriptionAutomaticTaxOptions { Enabled = true }
|
||||||
DefaultTaxRates = []
|
|
||||||
};
|
};
|
||||||
|
|
||||||
_ = await _stripeAdapter.SubscriptionUpdateAsync(
|
_ = await _stripeAdapter.SubscriptionUpdateAsync(
|
||||||
|
Loading…
x
Reference in New Issue
Block a user