mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 21:18:13 -05:00
[AC-2471] Prevent calls to Stripe when unlinking client org has no Stripe objects (#3999)
* Prevent calls to Stripe when unlinking client org has no Stripe objects * Thomas' feedback * Check for stripe when org unlinked from org page --------- Co-authored-by: Conner Turnbull <cturnbull@bitwarden.com>
This commit is contained in:
parent
fa7b00a728
commit
ac4ccafe19
@ -6,6 +6,7 @@ using Bit.Core.AdminConsole.Providers.Interfaces;
|
|||||||
using Bit.Core.AdminConsole.Repositories;
|
using Bit.Core.AdminConsole.Repositories;
|
||||||
using Bit.Core.Billing.Commands;
|
using Bit.Core.Billing.Commands;
|
||||||
using Bit.Core.Billing.Constants;
|
using Bit.Core.Billing.Constants;
|
||||||
|
using Bit.Core.Billing.Extensions;
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.Exceptions;
|
using Bit.Core.Exceptions;
|
||||||
using Bit.Core.Repositories;
|
using Bit.Core.Repositories;
|
||||||
@ -76,6 +77,35 @@ public class RemoveOrganizationFromProviderCommand : IRemoveOrganizationFromProv
|
|||||||
|
|
||||||
organization.BillingEmail = organizationOwnerEmails.MinBy(email => email);
|
organization.BillingEmail = organizationOwnerEmails.MinBy(email => email);
|
||||||
|
|
||||||
|
await ResetOrganizationBillingAsync(organization, provider, organizationOwnerEmails);
|
||||||
|
|
||||||
|
await _organizationRepository.ReplaceAsync(organization);
|
||||||
|
|
||||||
|
await _providerOrganizationRepository.DeleteAsync(providerOrganization);
|
||||||
|
|
||||||
|
await _eventService.LogProviderOrganizationEventAsync(
|
||||||
|
providerOrganization,
|
||||||
|
EventType.ProviderOrganization_Removed);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// When a client organization is unlinked from a provider, we have to check if they're Stripe-enabled
|
||||||
|
/// and, if they are, we remove their MSP discount and set their Subscription to `send_invoice`. This is because
|
||||||
|
/// the provider's payment method will be removed from their Stripe customer causing ensuing charges to fail. Lastly,
|
||||||
|
/// we email the organization owners letting them know they need to add a new payment method.
|
||||||
|
/// </summary>
|
||||||
|
private async Task ResetOrganizationBillingAsync(
|
||||||
|
Organization organization,
|
||||||
|
Provider provider,
|
||||||
|
IEnumerable<string> organizationOwnerEmails)
|
||||||
|
{
|
||||||
|
if (!organization.IsStripeEnabled())
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var isConsolidatedBillingEnabled = _featureService.IsEnabled(FeatureFlagKeys.EnableConsolidatedBilling);
|
||||||
|
|
||||||
var customerUpdateOptions = new CustomerUpdateOptions
|
var customerUpdateOptions = new CustomerUpdateOptions
|
||||||
{
|
{
|
||||||
Coupon = string.Empty,
|
Coupon = string.Empty,
|
||||||
@ -84,11 +114,10 @@ public class RemoveOrganizationFromProviderCommand : IRemoveOrganizationFromProv
|
|||||||
|
|
||||||
await _stripeAdapter.CustomerUpdateAsync(organization.GatewayCustomerId, customerUpdateOptions);
|
await _stripeAdapter.CustomerUpdateAsync(organization.GatewayCustomerId, customerUpdateOptions);
|
||||||
|
|
||||||
var isConsolidatedBillingEnabled = _featureService.IsEnabled(FeatureFlagKeys.EnableConsolidatedBilling);
|
|
||||||
|
|
||||||
if (isConsolidatedBillingEnabled && provider.Status == ProviderStatusType.Billable)
|
if (isConsolidatedBillingEnabled && provider.Status == ProviderStatusType.Billable)
|
||||||
{
|
{
|
||||||
var plan = StaticStore.GetPlan(organization.PlanType).PasswordManager;
|
var plan = StaticStore.GetPlan(organization.PlanType).PasswordManager;
|
||||||
|
|
||||||
var subscriptionCreateOptions = new SubscriptionCreateOptions
|
var subscriptionCreateOptions = new SubscriptionCreateOptions
|
||||||
{
|
{
|
||||||
Customer = organization.GatewayCustomerId,
|
Customer = organization.GatewayCustomerId,
|
||||||
@ -103,8 +132,11 @@ public class RemoveOrganizationFromProviderCommand : IRemoveOrganizationFromProv
|
|||||||
ProrationBehavior = StripeConstants.ProrationBehavior.CreateProrations,
|
ProrationBehavior = StripeConstants.ProrationBehavior.CreateProrations,
|
||||||
Items = [new SubscriptionItemOptions { Price = plan.StripeSeatPlanId, Quantity = organization.Seats }]
|
Items = [new SubscriptionItemOptions { Price = plan.StripeSeatPlanId, Quantity = organization.Seats }]
|
||||||
};
|
};
|
||||||
|
|
||||||
var subscription = await _stripeAdapter.SubscriptionCreateAsync(subscriptionCreateOptions);
|
var subscription = await _stripeAdapter.SubscriptionCreateAsync(subscriptionCreateOptions);
|
||||||
|
|
||||||
organization.GatewaySubscriptionId = subscription.Id;
|
organization.GatewaySubscriptionId = subscription.Id;
|
||||||
|
|
||||||
await _scaleSeatsCommand.ScalePasswordManagerSeats(provider, organization.PlanType,
|
await _scaleSeatsCommand.ScalePasswordManagerSeats(provider, organization.PlanType,
|
||||||
-(organization.Seats ?? 0));
|
-(organization.Seats ?? 0));
|
||||||
}
|
}
|
||||||
@ -115,21 +147,14 @@ public class RemoveOrganizationFromProviderCommand : IRemoveOrganizationFromProv
|
|||||||
CollectionMethod = "send_invoice",
|
CollectionMethod = "send_invoice",
|
||||||
DaysUntilDue = 30
|
DaysUntilDue = 30
|
||||||
};
|
};
|
||||||
|
|
||||||
await _stripeAdapter.SubscriptionUpdateAsync(organization.GatewaySubscriptionId, subscriptionUpdateOptions);
|
await _stripeAdapter.SubscriptionUpdateAsync(organization.GatewaySubscriptionId, subscriptionUpdateOptions);
|
||||||
}
|
}
|
||||||
|
|
||||||
await _organizationRepository.ReplaceAsync(organization);
|
|
||||||
|
|
||||||
await _mailService.SendProviderUpdatePaymentMethod(
|
await _mailService.SendProviderUpdatePaymentMethod(
|
||||||
organization.Id,
|
organization.Id,
|
||||||
organization.Name,
|
organization.Name,
|
||||||
provider.Name,
|
provider.Name,
|
||||||
organizationOwnerEmails);
|
organizationOwnerEmails);
|
||||||
|
|
||||||
await _providerOrganizationRepository.DeleteAsync(providerOrganization);
|
|
||||||
|
|
||||||
await _eventService.LogProviderOrganizationEventAsync(
|
|
||||||
providerOrganization,
|
|
||||||
EventType.ProviderOrganization_Removed);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -14,6 +14,7 @@ using Bit.Test.Common.AutoFixture.Attributes;
|
|||||||
using NSubstitute;
|
using NSubstitute;
|
||||||
using Stripe;
|
using Stripe;
|
||||||
using Xunit;
|
using Xunit;
|
||||||
|
using IMailService = Bit.Core.Services.IMailService;
|
||||||
|
|
||||||
namespace Bit.Commercial.Core.Test.AdminConsole.ProviderFeatures;
|
namespace Bit.Commercial.Core.Test.AdminConsole.ProviderFeatures;
|
||||||
|
|
||||||
@ -83,6 +84,55 @@ public class RemoveOrganizationFromProviderCommandTests
|
|||||||
Assert.Equal("Organization must have at least one confirmed owner.", exception.Message);
|
Assert.Equal("Organization must have at least one confirmed owner.", exception.Message);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task RemoveOrganizationFromProvider_NoStripeObjects_MakesCorrectInvocations(
|
||||||
|
Provider provider,
|
||||||
|
ProviderOrganization providerOrganization,
|
||||||
|
Organization organization,
|
||||||
|
SutProvider<RemoveOrganizationFromProviderCommand> sutProvider)
|
||||||
|
{
|
||||||
|
organization.GatewayCustomerId = null;
|
||||||
|
organization.GatewaySubscriptionId = null;
|
||||||
|
|
||||||
|
providerOrganization.ProviderId = provider.Id;
|
||||||
|
|
||||||
|
var organizationRepository = sutProvider.GetDependency<IOrganizationRepository>();
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IOrganizationService>().HasConfirmedOwnersExceptAsync(
|
||||||
|
providerOrganization.OrganizationId,
|
||||||
|
Array.Empty<Guid>(),
|
||||||
|
includeProvider: false)
|
||||||
|
.Returns(true);
|
||||||
|
|
||||||
|
var organizationOwnerEmails = new List<string> { "a@gmail.com", "b@gmail.com" };
|
||||||
|
|
||||||
|
organizationRepository.GetOwnerEmailAddressesById(organization.Id).Returns(organizationOwnerEmails);
|
||||||
|
|
||||||
|
await sutProvider.Sut.RemoveOrganizationFromProvider(provider, providerOrganization, organization);
|
||||||
|
|
||||||
|
await organizationRepository.Received(1).ReplaceAsync(Arg.Is<Organization>(
|
||||||
|
org => org.Id == organization.Id && org.BillingEmail == "a@gmail.com"));
|
||||||
|
|
||||||
|
var stripeAdapter = sutProvider.GetDependency<IStripeAdapter>();
|
||||||
|
|
||||||
|
await stripeAdapter.DidNotReceiveWithAnyArgs().CustomerUpdateAsync(Arg.Any<string>(), Arg.Any<CustomerUpdateOptions>());
|
||||||
|
|
||||||
|
await stripeAdapter.DidNotReceiveWithAnyArgs().SubscriptionUpdateAsync(Arg.Any<string>(), Arg.Any<SubscriptionUpdateOptions>());
|
||||||
|
|
||||||
|
await sutProvider.GetDependency<IMailService>().DidNotReceiveWithAnyArgs().SendProviderUpdatePaymentMethod(
|
||||||
|
Arg.Any<Guid>(),
|
||||||
|
Arg.Any<string>(),
|
||||||
|
Arg.Any<string>(),
|
||||||
|
Arg.Any<IEnumerable<string>>());
|
||||||
|
|
||||||
|
await sutProvider.GetDependency<IProviderOrganizationRepository>().Received(1)
|
||||||
|
.DeleteAsync(providerOrganization);
|
||||||
|
|
||||||
|
await sutProvider.GetDependency<IEventService>().Received(1).LogProviderOrganizationEventAsync(
|
||||||
|
providerOrganization,
|
||||||
|
EventType.ProviderOrganization_Removed);
|
||||||
|
}
|
||||||
|
|
||||||
[Theory, BitAutoData]
|
[Theory, BitAutoData]
|
||||||
public async Task RemoveOrganizationFromProvider_MakesCorrectInvocations__FeatureFlagOff(
|
public async Task RemoveOrganizationFromProvider_MakesCorrectInvocations__FeatureFlagOff(
|
||||||
Provider provider,
|
Provider provider,
|
||||||
|
@ -349,7 +349,10 @@ public class OrganizationsController : Controller
|
|||||||
providerOrganization,
|
providerOrganization,
|
||||||
organization);
|
organization);
|
||||||
|
|
||||||
await _removePaymentMethodCommand.RemovePaymentMethod(organization);
|
if (organization.IsStripeEnabled())
|
||||||
|
{
|
||||||
|
await _removePaymentMethodCommand.RemovePaymentMethod(organization);
|
||||||
|
}
|
||||||
|
|
||||||
return Json(null);
|
return Json(null);
|
||||||
}
|
}
|
||||||
|
@ -3,6 +3,7 @@ using Bit.Admin.Utilities;
|
|||||||
using Bit.Core.AdminConsole.Providers.Interfaces;
|
using Bit.Core.AdminConsole.Providers.Interfaces;
|
||||||
using Bit.Core.AdminConsole.Repositories;
|
using Bit.Core.AdminConsole.Repositories;
|
||||||
using Bit.Core.Billing.Commands;
|
using Bit.Core.Billing.Commands;
|
||||||
|
using Bit.Core.Billing.Extensions;
|
||||||
using Bit.Core.Exceptions;
|
using Bit.Core.Exceptions;
|
||||||
using Bit.Core.Repositories;
|
using Bit.Core.Repositories;
|
||||||
using Bit.Core.Utilities;
|
using Bit.Core.Utilities;
|
||||||
@ -69,7 +70,10 @@ public class ProviderOrganizationsController : Controller
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
await _removePaymentMethodCommand.RemovePaymentMethod(organization);
|
if (organization.IsStripeEnabled())
|
||||||
|
{
|
||||||
|
await _removePaymentMethodCommand.RemovePaymentMethod(organization);
|
||||||
|
}
|
||||||
|
|
||||||
return Json(null);
|
return Json(null);
|
||||||
}
|
}
|
||||||
|
@ -5,6 +5,7 @@ using Bit.Core.AdminConsole.Providers.Interfaces;
|
|||||||
using Bit.Core.AdminConsole.Repositories;
|
using Bit.Core.AdminConsole.Repositories;
|
||||||
using Bit.Core.AdminConsole.Services;
|
using Bit.Core.AdminConsole.Services;
|
||||||
using Bit.Core.Billing.Commands;
|
using Bit.Core.Billing.Commands;
|
||||||
|
using Bit.Core.Billing.Extensions;
|
||||||
using Bit.Core.Context;
|
using Bit.Core.Context;
|
||||||
using Bit.Core.Exceptions;
|
using Bit.Core.Exceptions;
|
||||||
using Bit.Core.Repositories;
|
using Bit.Core.Repositories;
|
||||||
@ -112,6 +113,9 @@ public class ProviderOrganizationsController : Controller
|
|||||||
providerOrganization,
|
providerOrganization,
|
||||||
organization);
|
organization);
|
||||||
|
|
||||||
await _removePaymentMethodCommand.RemovePaymentMethod(organization);
|
if (organization.IsStripeEnabled())
|
||||||
|
{
|
||||||
|
await _removePaymentMethodCommand.RemovePaymentMethod(organization);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -22,6 +22,10 @@ public static class BillingExtensions
|
|||||||
PlanType: PlanType.TeamsMonthly or PlanType.EnterpriseMonthly
|
PlanType: PlanType.TeamsMonthly or PlanType.EnterpriseMonthly
|
||||||
};
|
};
|
||||||
|
|
||||||
|
public static bool IsStripeEnabled(this Organization organization)
|
||||||
|
=> !string.IsNullOrEmpty(organization.GatewayCustomerId) &&
|
||||||
|
!string.IsNullOrEmpty(organization.GatewaySubscriptionId);
|
||||||
|
|
||||||
public static bool SupportsConsolidatedBilling(this PlanType planType)
|
public static bool SupportsConsolidatedBilling(this PlanType planType)
|
||||||
=> planType is PlanType.TeamsMonthly or PlanType.EnterpriseMonthly;
|
=> planType is PlanType.TeamsMonthly or PlanType.EnterpriseMonthly;
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user