mirror of
https://github.com/bitwarden/server.git
synced 2025-04-06 05:28:15 -05:00
[AC-1904] Implement endpoint to retrieve Provider subscription (#3921)
* Refactor Core.Billing prior to adding new logic * Add ProviderBillingQueries.GetSubscriptionData * Add ProviderBillingController.GetSubscriptionAsync
This commit is contained in:
parent
46dba15194
commit
ffd988eeda
@ -66,7 +66,7 @@ public class OrganizationsController : Controller
|
|||||||
private readonly IAddSecretsManagerSubscriptionCommand _addSecretsManagerSubscriptionCommand;
|
private readonly IAddSecretsManagerSubscriptionCommand _addSecretsManagerSubscriptionCommand;
|
||||||
private readonly IPushNotificationService _pushNotificationService;
|
private readonly IPushNotificationService _pushNotificationService;
|
||||||
private readonly ICancelSubscriptionCommand _cancelSubscriptionCommand;
|
private readonly ICancelSubscriptionCommand _cancelSubscriptionCommand;
|
||||||
private readonly IGetSubscriptionQuery _getSubscriptionQuery;
|
private readonly ISubscriberQueries _subscriberQueries;
|
||||||
private readonly IReferenceEventService _referenceEventService;
|
private readonly IReferenceEventService _referenceEventService;
|
||||||
private readonly IOrganizationEnableCollectionEnhancementsCommand _organizationEnableCollectionEnhancementsCommand;
|
private readonly IOrganizationEnableCollectionEnhancementsCommand _organizationEnableCollectionEnhancementsCommand;
|
||||||
|
|
||||||
@ -93,7 +93,7 @@ public class OrganizationsController : Controller
|
|||||||
IAddSecretsManagerSubscriptionCommand addSecretsManagerSubscriptionCommand,
|
IAddSecretsManagerSubscriptionCommand addSecretsManagerSubscriptionCommand,
|
||||||
IPushNotificationService pushNotificationService,
|
IPushNotificationService pushNotificationService,
|
||||||
ICancelSubscriptionCommand cancelSubscriptionCommand,
|
ICancelSubscriptionCommand cancelSubscriptionCommand,
|
||||||
IGetSubscriptionQuery getSubscriptionQuery,
|
ISubscriberQueries subscriberQueries,
|
||||||
IReferenceEventService referenceEventService,
|
IReferenceEventService referenceEventService,
|
||||||
IOrganizationEnableCollectionEnhancementsCommand organizationEnableCollectionEnhancementsCommand)
|
IOrganizationEnableCollectionEnhancementsCommand organizationEnableCollectionEnhancementsCommand)
|
||||||
{
|
{
|
||||||
@ -119,7 +119,7 @@ public class OrganizationsController : Controller
|
|||||||
_addSecretsManagerSubscriptionCommand = addSecretsManagerSubscriptionCommand;
|
_addSecretsManagerSubscriptionCommand = addSecretsManagerSubscriptionCommand;
|
||||||
_pushNotificationService = pushNotificationService;
|
_pushNotificationService = pushNotificationService;
|
||||||
_cancelSubscriptionCommand = cancelSubscriptionCommand;
|
_cancelSubscriptionCommand = cancelSubscriptionCommand;
|
||||||
_getSubscriptionQuery = getSubscriptionQuery;
|
_subscriberQueries = subscriberQueries;
|
||||||
_referenceEventService = referenceEventService;
|
_referenceEventService = referenceEventService;
|
||||||
_organizationEnableCollectionEnhancementsCommand = organizationEnableCollectionEnhancementsCommand;
|
_organizationEnableCollectionEnhancementsCommand = organizationEnableCollectionEnhancementsCommand;
|
||||||
}
|
}
|
||||||
@ -479,7 +479,7 @@ public class OrganizationsController : Controller
|
|||||||
throw new NotFoundException();
|
throw new NotFoundException();
|
||||||
}
|
}
|
||||||
|
|
||||||
var subscription = await _getSubscriptionQuery.GetSubscription(organization);
|
var subscription = await _subscriberQueries.GetSubscriptionOrThrow(organization);
|
||||||
|
|
||||||
await _cancelSubscriptionCommand.CancelSubscription(subscription,
|
await _cancelSubscriptionCommand.CancelSubscription(subscription,
|
||||||
new OffboardingSurveyResponse
|
new OffboardingSurveyResponse
|
||||||
|
@ -69,7 +69,7 @@ public class AccountsController : Controller
|
|||||||
private readonly IRotateUserKeyCommand _rotateUserKeyCommand;
|
private readonly IRotateUserKeyCommand _rotateUserKeyCommand;
|
||||||
private readonly IFeatureService _featureService;
|
private readonly IFeatureService _featureService;
|
||||||
private readonly ICancelSubscriptionCommand _cancelSubscriptionCommand;
|
private readonly ICancelSubscriptionCommand _cancelSubscriptionCommand;
|
||||||
private readonly IGetSubscriptionQuery _getSubscriptionQuery;
|
private readonly ISubscriberQueries _subscriberQueries;
|
||||||
private readonly IReferenceEventService _referenceEventService;
|
private readonly IReferenceEventService _referenceEventService;
|
||||||
private readonly ICurrentContext _currentContext;
|
private readonly ICurrentContext _currentContext;
|
||||||
|
|
||||||
@ -104,7 +104,7 @@ public class AccountsController : Controller
|
|||||||
IRotateUserKeyCommand rotateUserKeyCommand,
|
IRotateUserKeyCommand rotateUserKeyCommand,
|
||||||
IFeatureService featureService,
|
IFeatureService featureService,
|
||||||
ICancelSubscriptionCommand cancelSubscriptionCommand,
|
ICancelSubscriptionCommand cancelSubscriptionCommand,
|
||||||
IGetSubscriptionQuery getSubscriptionQuery,
|
ISubscriberQueries subscriberQueries,
|
||||||
IReferenceEventService referenceEventService,
|
IReferenceEventService referenceEventService,
|
||||||
ICurrentContext currentContext,
|
ICurrentContext currentContext,
|
||||||
IRotationValidator<IEnumerable<CipherWithIdRequestModel>, IEnumerable<Cipher>> cipherValidator,
|
IRotationValidator<IEnumerable<CipherWithIdRequestModel>, IEnumerable<Cipher>> cipherValidator,
|
||||||
@ -133,7 +133,7 @@ public class AccountsController : Controller
|
|||||||
_rotateUserKeyCommand = rotateUserKeyCommand;
|
_rotateUserKeyCommand = rotateUserKeyCommand;
|
||||||
_featureService = featureService;
|
_featureService = featureService;
|
||||||
_cancelSubscriptionCommand = cancelSubscriptionCommand;
|
_cancelSubscriptionCommand = cancelSubscriptionCommand;
|
||||||
_getSubscriptionQuery = getSubscriptionQuery;
|
_subscriberQueries = subscriberQueries;
|
||||||
_referenceEventService = referenceEventService;
|
_referenceEventService = referenceEventService;
|
||||||
_currentContext = currentContext;
|
_currentContext = currentContext;
|
||||||
_cipherValidator = cipherValidator;
|
_cipherValidator = cipherValidator;
|
||||||
@ -831,7 +831,7 @@ public class AccountsController : Controller
|
|||||||
throw new UnauthorizedAccessException();
|
throw new UnauthorizedAccessException();
|
||||||
}
|
}
|
||||||
|
|
||||||
var subscription = await _getSubscriptionQuery.GetSubscription(user);
|
var subscription = await _subscriberQueries.GetSubscriptionOrThrow(user);
|
||||||
|
|
||||||
await _cancelSubscriptionCommand.CancelSubscription(subscription,
|
await _cancelSubscriptionCommand.CancelSubscription(subscription,
|
||||||
new OffboardingSurveyResponse
|
new OffboardingSurveyResponse
|
||||||
|
44
src/Api/Billing/Controllers/ProviderBillingController.cs
Normal file
44
src/Api/Billing/Controllers/ProviderBillingController.cs
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
using Bit.Api.Billing.Models;
|
||||||
|
using Bit.Core;
|
||||||
|
using Bit.Core.Billing.Queries;
|
||||||
|
using Bit.Core.Context;
|
||||||
|
using Bit.Core.Services;
|
||||||
|
using Microsoft.AspNetCore.Authorization;
|
||||||
|
using Microsoft.AspNetCore.Mvc;
|
||||||
|
|
||||||
|
namespace Bit.Api.Billing.Controllers;
|
||||||
|
|
||||||
|
[Route("providers/{providerId:guid}/billing")]
|
||||||
|
[Authorize("Application")]
|
||||||
|
public class ProviderBillingController(
|
||||||
|
ICurrentContext currentContext,
|
||||||
|
IFeatureService featureService,
|
||||||
|
IProviderBillingQueries providerBillingQueries) : Controller
|
||||||
|
{
|
||||||
|
[HttpGet("subscription")]
|
||||||
|
public async Task<IResult> GetSubscriptionAsync([FromRoute] Guid providerId)
|
||||||
|
{
|
||||||
|
if (!featureService.IsEnabled(FeatureFlagKeys.EnableConsolidatedBilling))
|
||||||
|
{
|
||||||
|
return TypedResults.NotFound();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!currentContext.ProviderProviderAdmin(providerId))
|
||||||
|
{
|
||||||
|
return TypedResults.Unauthorized();
|
||||||
|
}
|
||||||
|
|
||||||
|
var subscriptionData = await providerBillingQueries.GetSubscriptionData(providerId);
|
||||||
|
|
||||||
|
if (subscriptionData == null)
|
||||||
|
{
|
||||||
|
return TypedResults.NotFound();
|
||||||
|
}
|
||||||
|
|
||||||
|
var (providerPlans, subscription) = subscriptionData;
|
||||||
|
|
||||||
|
var providerSubscriptionDTO = ProviderSubscriptionDTO.From(providerPlans, subscription);
|
||||||
|
|
||||||
|
return TypedResults.Ok(providerSubscriptionDTO);
|
||||||
|
}
|
||||||
|
}
|
47
src/Api/Billing/Models/ProviderSubscriptionDTO.cs
Normal file
47
src/Api/Billing/Models/ProviderSubscriptionDTO.cs
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
using Bit.Core.Billing.Models;
|
||||||
|
using Bit.Core.Utilities;
|
||||||
|
using Stripe;
|
||||||
|
|
||||||
|
namespace Bit.Api.Billing.Models;
|
||||||
|
|
||||||
|
public record ProviderSubscriptionDTO(
|
||||||
|
string Status,
|
||||||
|
DateTime CurrentPeriodEndDate,
|
||||||
|
decimal? DiscountPercentage,
|
||||||
|
IEnumerable<ProviderPlanDTO> Plans)
|
||||||
|
{
|
||||||
|
private const string _annualCadence = "Annual";
|
||||||
|
private const string _monthlyCadence = "Monthly";
|
||||||
|
|
||||||
|
public static ProviderSubscriptionDTO From(
|
||||||
|
IEnumerable<ConfiguredProviderPlan> providerPlans,
|
||||||
|
Subscription subscription)
|
||||||
|
{
|
||||||
|
var providerPlansDTO = providerPlans
|
||||||
|
.Select(providerPlan =>
|
||||||
|
{
|
||||||
|
var plan = StaticStore.GetPlan(providerPlan.PlanType);
|
||||||
|
var cost = (providerPlan.SeatMinimum + providerPlan.PurchasedSeats) * plan.PasswordManager.SeatPrice;
|
||||||
|
var cadence = plan.IsAnnual ? _annualCadence : _monthlyCadence;
|
||||||
|
return new ProviderPlanDTO(
|
||||||
|
plan.Name,
|
||||||
|
providerPlan.SeatMinimum,
|
||||||
|
providerPlan.PurchasedSeats,
|
||||||
|
cost,
|
||||||
|
cadence);
|
||||||
|
});
|
||||||
|
|
||||||
|
return new ProviderSubscriptionDTO(
|
||||||
|
subscription.Status,
|
||||||
|
subscription.CurrentPeriodEnd,
|
||||||
|
subscription.Customer?.Discount?.Coupon?.PercentOff,
|
||||||
|
providerPlansDTO);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public record ProviderPlanDTO(
|
||||||
|
string PlanName,
|
||||||
|
int SeatMinimum,
|
||||||
|
int PurchasedSeats,
|
||||||
|
decimal Cost,
|
||||||
|
string Cadence);
|
@ -6,7 +6,7 @@ using Bit.Core.Utilities;
|
|||||||
|
|
||||||
namespace Bit.Core.AdminConsole.Entities.Provider;
|
namespace Bit.Core.AdminConsole.Entities.Provider;
|
||||||
|
|
||||||
public class Provider : ITableObject<Guid>
|
public class Provider : ITableObject<Guid>, ISubscriber
|
||||||
{
|
{
|
||||||
public Guid Id { get; set; }
|
public Guid Id { get; set; }
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -34,6 +34,26 @@ public class Provider : ITableObject<Guid>
|
|||||||
public string GatewayCustomerId { get; set; }
|
public string GatewayCustomerId { get; set; }
|
||||||
public string GatewaySubscriptionId { get; set; }
|
public string GatewaySubscriptionId { get; set; }
|
||||||
|
|
||||||
|
public string BillingEmailAddress() => BillingEmail?.ToLowerInvariant().Trim();
|
||||||
|
|
||||||
|
public string BillingName() => DisplayBusinessName();
|
||||||
|
|
||||||
|
public string SubscriberName() => DisplayName();
|
||||||
|
|
||||||
|
public string BraintreeCustomerIdPrefix() => "p";
|
||||||
|
|
||||||
|
public string BraintreeIdField() => "provider_id";
|
||||||
|
|
||||||
|
public string BraintreeCloudRegionField() => "region";
|
||||||
|
|
||||||
|
public bool IsOrganization() => false;
|
||||||
|
|
||||||
|
public bool IsUser() => false;
|
||||||
|
|
||||||
|
public string SubscriberType() => "Provider";
|
||||||
|
|
||||||
|
public bool IsExpired() => false;
|
||||||
|
|
||||||
public void SetNewId()
|
public void SetNewId()
|
||||||
{
|
{
|
||||||
if (Id == default)
|
if (Id == default)
|
||||||
|
9
src/Core/Billing/BillingException.cs
Normal file
9
src/Core/Billing/BillingException.cs
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
namespace Bit.Core.Billing;
|
||||||
|
|
||||||
|
public class BillingException(
|
||||||
|
string clientFriendlyMessage,
|
||||||
|
string internalMessage = null,
|
||||||
|
Exception innerException = null) : Exception(internalMessage, innerException)
|
||||||
|
{
|
||||||
|
public string ClientFriendlyMessage { get; set; } = clientFriendlyMessage;
|
||||||
|
}
|
@ -1,7 +1,6 @@
|
|||||||
using Bit.Core.AdminConsole.Entities;
|
using Bit.Core.AdminConsole.Entities;
|
||||||
using Bit.Core.Billing.Models;
|
using Bit.Core.Billing.Models;
|
||||||
using Bit.Core.Entities;
|
using Bit.Core.Entities;
|
||||||
using Bit.Core.Exceptions;
|
|
||||||
using Stripe;
|
using Stripe;
|
||||||
|
|
||||||
namespace Bit.Core.Billing.Commands;
|
namespace Bit.Core.Billing.Commands;
|
||||||
@ -17,7 +16,6 @@ public interface ICancelSubscriptionCommand
|
|||||||
/// <param name="subscription">The <see cref="User"/> or <see cref="Organization"/> with the subscription to cancel.</param>
|
/// <param name="subscription">The <see cref="User"/> or <see cref="Organization"/> with the subscription to cancel.</param>
|
||||||
/// <param name="offboardingSurveyResponse">An <see cref="OffboardingSurveyResponse"/> DTO containing user-provided feedback on why they are cancelling the subscription.</param>
|
/// <param name="offboardingSurveyResponse">An <see cref="OffboardingSurveyResponse"/> DTO containing user-provided feedback on why they are cancelling the subscription.</param>
|
||||||
/// <param name="cancelImmediately">A flag indicating whether to cancel the subscription immediately or at the end of the subscription period.</param>
|
/// <param name="cancelImmediately">A flag indicating whether to cancel the subscription immediately or at the end of the subscription period.</param>
|
||||||
/// <exception cref="GatewayException">Thrown when the provided subscription is already in an inactive state.</exception>
|
|
||||||
Task CancelSubscription(
|
Task CancelSubscription(
|
||||||
Subscription subscription,
|
Subscription subscription,
|
||||||
OffboardingSurveyResponse offboardingSurveyResponse,
|
OffboardingSurveyResponse offboardingSurveyResponse,
|
||||||
|
@ -4,5 +4,12 @@ namespace Bit.Core.Billing.Commands;
|
|||||||
|
|
||||||
public interface IRemovePaymentMethodCommand
|
public interface IRemovePaymentMethodCommand
|
||||||
{
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Attempts to remove an Organization's saved payment method. If the Stripe <see cref="Stripe.Customer"/> representing the
|
||||||
|
/// <see cref="Organization"/> contains a valid <b>"btCustomerId"</b> key in its <see cref="Stripe.Customer.Metadata"/> property,
|
||||||
|
/// this command will attempt to remove the Braintree <see cref="Braintree.PaymentMethod"/>. Otherwise, it will attempt to remove the
|
||||||
|
/// Stripe <see cref="Stripe.PaymentMethod"/>.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="organization">The organization to remove the saved payment method for.</param>
|
||||||
Task RemovePaymentMethod(Organization organization);
|
Task RemovePaymentMethod(Organization organization);
|
||||||
}
|
}
|
||||||
|
@ -1,55 +1,41 @@
|
|||||||
using Bit.Core.AdminConsole.Entities;
|
using Bit.Core.AdminConsole.Entities;
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.Exceptions;
|
|
||||||
using Bit.Core.Services;
|
using Bit.Core.Services;
|
||||||
using Braintree;
|
using Braintree;
|
||||||
using Microsoft.Extensions.Logging;
|
using Microsoft.Extensions.Logging;
|
||||||
|
|
||||||
|
using static Bit.Core.Billing.Utilities;
|
||||||
|
|
||||||
namespace Bit.Core.Billing.Commands.Implementations;
|
namespace Bit.Core.Billing.Commands.Implementations;
|
||||||
|
|
||||||
public class RemovePaymentMethodCommand : IRemovePaymentMethodCommand
|
public class RemovePaymentMethodCommand(
|
||||||
{
|
|
||||||
private readonly IBraintreeGateway _braintreeGateway;
|
|
||||||
private readonly ILogger<RemovePaymentMethodCommand> _logger;
|
|
||||||
private readonly IStripeAdapter _stripeAdapter;
|
|
||||||
|
|
||||||
public RemovePaymentMethodCommand(
|
|
||||||
IBraintreeGateway braintreeGateway,
|
IBraintreeGateway braintreeGateway,
|
||||||
ILogger<RemovePaymentMethodCommand> logger,
|
ILogger<RemovePaymentMethodCommand> logger,
|
||||||
IStripeAdapter stripeAdapter)
|
IStripeAdapter stripeAdapter)
|
||||||
{
|
: IRemovePaymentMethodCommand
|
||||||
_braintreeGateway = braintreeGateway;
|
{
|
||||||
_logger = logger;
|
|
||||||
_stripeAdapter = stripeAdapter;
|
|
||||||
}
|
|
||||||
|
|
||||||
public async Task RemovePaymentMethod(Organization organization)
|
public async Task RemovePaymentMethod(Organization organization)
|
||||||
{
|
{
|
||||||
const string braintreeCustomerIdKey = "btCustomerId";
|
ArgumentNullException.ThrowIfNull(organization);
|
||||||
|
|
||||||
if (organization == null)
|
|
||||||
{
|
|
||||||
throw new ArgumentNullException(nameof(organization));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (organization.Gateway is not GatewayType.Stripe || string.IsNullOrEmpty(organization.GatewayCustomerId))
|
if (organization.Gateway is not GatewayType.Stripe || string.IsNullOrEmpty(organization.GatewayCustomerId))
|
||||||
{
|
{
|
||||||
throw ContactSupport();
|
throw ContactSupport();
|
||||||
}
|
}
|
||||||
|
|
||||||
var stripeCustomer = await _stripeAdapter.CustomerGetAsync(organization.GatewayCustomerId, new Stripe.CustomerGetOptions
|
var stripeCustomer = await stripeAdapter.CustomerGetAsync(organization.GatewayCustomerId, new Stripe.CustomerGetOptions
|
||||||
{
|
{
|
||||||
Expand = new List<string> { "invoice_settings.default_payment_method", "sources" }
|
Expand = ["invoice_settings.default_payment_method", "sources"]
|
||||||
});
|
});
|
||||||
|
|
||||||
if (stripeCustomer == null)
|
if (stripeCustomer == null)
|
||||||
{
|
{
|
||||||
_logger.LogError("Could not find Stripe customer ({ID}) when removing payment method", organization.GatewayCustomerId);
|
logger.LogError("Could not find Stripe customer ({ID}) when removing payment method", organization.GatewayCustomerId);
|
||||||
|
|
||||||
throw ContactSupport();
|
throw ContactSupport();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (stripeCustomer.Metadata?.TryGetValue(braintreeCustomerIdKey, out var braintreeCustomerId) ?? false)
|
if (stripeCustomer.Metadata?.TryGetValue(BraintreeCustomerIdKey, out var braintreeCustomerId) ?? false)
|
||||||
{
|
{
|
||||||
await RemoveBraintreePaymentMethodAsync(braintreeCustomerId);
|
await RemoveBraintreePaymentMethodAsync(braintreeCustomerId);
|
||||||
}
|
}
|
||||||
@ -61,11 +47,11 @@ public class RemovePaymentMethodCommand : IRemovePaymentMethodCommand
|
|||||||
|
|
||||||
private async Task RemoveBraintreePaymentMethodAsync(string braintreeCustomerId)
|
private async Task RemoveBraintreePaymentMethodAsync(string braintreeCustomerId)
|
||||||
{
|
{
|
||||||
var customer = await _braintreeGateway.Customer.FindAsync(braintreeCustomerId);
|
var customer = await braintreeGateway.Customer.FindAsync(braintreeCustomerId);
|
||||||
|
|
||||||
if (customer == null)
|
if (customer == null)
|
||||||
{
|
{
|
||||||
_logger.LogError("Failed to retrieve Braintree customer ({ID}) when removing payment method", braintreeCustomerId);
|
logger.LogError("Failed to retrieve Braintree customer ({ID}) when removing payment method", braintreeCustomerId);
|
||||||
|
|
||||||
throw ContactSupport();
|
throw ContactSupport();
|
||||||
}
|
}
|
||||||
@ -74,27 +60,27 @@ public class RemovePaymentMethodCommand : IRemovePaymentMethodCommand
|
|||||||
{
|
{
|
||||||
var existingDefaultPaymentMethod = customer.DefaultPaymentMethod;
|
var existingDefaultPaymentMethod = customer.DefaultPaymentMethod;
|
||||||
|
|
||||||
var updateCustomerResult = await _braintreeGateway.Customer.UpdateAsync(
|
var updateCustomerResult = await braintreeGateway.Customer.UpdateAsync(
|
||||||
braintreeCustomerId,
|
braintreeCustomerId,
|
||||||
new CustomerRequest { DefaultPaymentMethodToken = null });
|
new CustomerRequest { DefaultPaymentMethodToken = null });
|
||||||
|
|
||||||
if (!updateCustomerResult.IsSuccess())
|
if (!updateCustomerResult.IsSuccess())
|
||||||
{
|
{
|
||||||
_logger.LogError("Failed to update payment method for Braintree customer ({ID}) | Message: {Message}",
|
logger.LogError("Failed to update payment method for Braintree customer ({ID}) | Message: {Message}",
|
||||||
braintreeCustomerId, updateCustomerResult.Message);
|
braintreeCustomerId, updateCustomerResult.Message);
|
||||||
|
|
||||||
throw ContactSupport();
|
throw ContactSupport();
|
||||||
}
|
}
|
||||||
|
|
||||||
var deletePaymentMethodResult = await _braintreeGateway.PaymentMethod.DeleteAsync(existingDefaultPaymentMethod.Token);
|
var deletePaymentMethodResult = await braintreeGateway.PaymentMethod.DeleteAsync(existingDefaultPaymentMethod.Token);
|
||||||
|
|
||||||
if (!deletePaymentMethodResult.IsSuccess())
|
if (!deletePaymentMethodResult.IsSuccess())
|
||||||
{
|
{
|
||||||
await _braintreeGateway.Customer.UpdateAsync(
|
await braintreeGateway.Customer.UpdateAsync(
|
||||||
braintreeCustomerId,
|
braintreeCustomerId,
|
||||||
new CustomerRequest { DefaultPaymentMethodToken = existingDefaultPaymentMethod.Token });
|
new CustomerRequest { DefaultPaymentMethodToken = existingDefaultPaymentMethod.Token });
|
||||||
|
|
||||||
_logger.LogError(
|
logger.LogError(
|
||||||
"Failed to delete Braintree payment method for Customer ({ID}), re-linked payment method. Message: {Message}",
|
"Failed to delete Braintree payment method for Customer ({ID}), re-linked payment method. Message: {Message}",
|
||||||
braintreeCustomerId, deletePaymentMethodResult.Message);
|
braintreeCustomerId, deletePaymentMethodResult.Message);
|
||||||
|
|
||||||
@ -103,7 +89,7 @@ public class RemovePaymentMethodCommand : IRemovePaymentMethodCommand
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
_logger.LogWarning("Tried to remove non-existent Braintree payment method for Customer ({ID})", braintreeCustomerId);
|
logger.LogWarning("Tried to remove non-existent Braintree payment method for Customer ({ID})", braintreeCustomerId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -116,25 +102,23 @@ public class RemovePaymentMethodCommand : IRemovePaymentMethodCommand
|
|||||||
switch (source)
|
switch (source)
|
||||||
{
|
{
|
||||||
case Stripe.BankAccount:
|
case Stripe.BankAccount:
|
||||||
await _stripeAdapter.BankAccountDeleteAsync(customer.Id, source.Id);
|
await stripeAdapter.BankAccountDeleteAsync(customer.Id, source.Id);
|
||||||
break;
|
break;
|
||||||
case Stripe.Card:
|
case Stripe.Card:
|
||||||
await _stripeAdapter.CardDeleteAsync(customer.Id, source.Id);
|
await stripeAdapter.CardDeleteAsync(customer.Id, source.Id);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var paymentMethods = _stripeAdapter.PaymentMethodListAutoPagingAsync(new Stripe.PaymentMethodListOptions
|
var paymentMethods = stripeAdapter.PaymentMethodListAutoPagingAsync(new Stripe.PaymentMethodListOptions
|
||||||
{
|
{
|
||||||
Customer = customer.Id
|
Customer = customer.Id
|
||||||
});
|
});
|
||||||
|
|
||||||
await foreach (var paymentMethod in paymentMethods)
|
await foreach (var paymentMethod in paymentMethods)
|
||||||
{
|
{
|
||||||
await _stripeAdapter.PaymentMethodDetachAsync(paymentMethod.Id, new Stripe.PaymentMethodDetachOptions());
|
await stripeAdapter.PaymentMethodDetachAsync(paymentMethod.Id, new Stripe.PaymentMethodDetachOptions());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static GatewayException ContactSupport() => new("Could not remove your payment method. Please contact support for assistance.");
|
|
||||||
}
|
}
|
||||||
|
@ -11,7 +11,6 @@ public class ProviderPlan : ITableObject<Guid>
|
|||||||
public PlanType PlanType { get; set; }
|
public PlanType PlanType { get; set; }
|
||||||
public int? SeatMinimum { get; set; }
|
public int? SeatMinimum { get; set; }
|
||||||
public int? PurchasedSeats { get; set; }
|
public int? PurchasedSeats { get; set; }
|
||||||
public int? AllocatedSeats { get; set; }
|
|
||||||
|
|
||||||
public void SetNewId()
|
public void SetNewId()
|
||||||
{
|
{
|
||||||
@ -20,4 +19,6 @@ public class ProviderPlan : ITableObject<Guid>
|
|||||||
Id = CoreHelpers.GenerateComb();
|
Id = CoreHelpers.GenerateComb();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public bool Configured => SeatMinimum.HasValue && PurchasedSeats.HasValue;
|
||||||
}
|
}
|
||||||
|
@ -17,6 +17,7 @@ public static class ServiceCollectionExtensions
|
|||||||
|
|
||||||
public static void AddBillingQueries(this IServiceCollection services)
|
public static void AddBillingQueries(this IServiceCollection services)
|
||||||
{
|
{
|
||||||
services.AddSingleton<IGetSubscriptionQuery, GetSubscriptionQuery>();
|
services.AddSingleton<IProviderBillingQueries, ProviderBillingQueries>();
|
||||||
|
services.AddSingleton<ISubscriberQueries, SubscriberQueries>();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
22
src/Core/Billing/Models/ConfiguredProviderPlan.cs
Normal file
22
src/Core/Billing/Models/ConfiguredProviderPlan.cs
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
using Bit.Core.Billing.Entities;
|
||||||
|
using Bit.Core.Enums;
|
||||||
|
|
||||||
|
namespace Bit.Core.Billing.Models;
|
||||||
|
|
||||||
|
public record ConfiguredProviderPlan(
|
||||||
|
Guid Id,
|
||||||
|
Guid ProviderId,
|
||||||
|
PlanType PlanType,
|
||||||
|
int SeatMinimum,
|
||||||
|
int PurchasedSeats)
|
||||||
|
{
|
||||||
|
public static ConfiguredProviderPlan From(ProviderPlan providerPlan) =>
|
||||||
|
providerPlan.Configured
|
||||||
|
? new ConfiguredProviderPlan(
|
||||||
|
providerPlan.Id,
|
||||||
|
providerPlan.ProviderId,
|
||||||
|
providerPlan.PlanType,
|
||||||
|
providerPlan.SeatMinimum.GetValueOrDefault(0),
|
||||||
|
providerPlan.PurchasedSeats.GetValueOrDefault(0))
|
||||||
|
: null;
|
||||||
|
}
|
7
src/Core/Billing/Models/ProviderSubscriptionData.cs
Normal file
7
src/Core/Billing/Models/ProviderSubscriptionData.cs
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
using Stripe;
|
||||||
|
|
||||||
|
namespace Bit.Core.Billing.Models;
|
||||||
|
|
||||||
|
public record ProviderSubscriptionData(
|
||||||
|
List<ConfiguredProviderPlan> ProviderPlans,
|
||||||
|
Subscription Subscription);
|
@ -1,18 +0,0 @@
|
|||||||
using Bit.Core.Entities;
|
|
||||||
using Bit.Core.Exceptions;
|
|
||||||
using Stripe;
|
|
||||||
|
|
||||||
namespace Bit.Core.Billing.Queries;
|
|
||||||
|
|
||||||
public interface IGetSubscriptionQuery
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Retrieves a Stripe <see cref="Subscription"/> using the <paramref name="subscriber"/>'s <see cref="ISubscriber.GatewaySubscriptionId"/> property.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="subscriber">The organization or user to retrieve the subscription for.</param>
|
|
||||||
/// <returns>A Stripe <see cref="Subscription"/>.</returns>
|
|
||||||
/// <exception cref="ArgumentNullException">Thrown when the <paramref name="subscriber"/> is <see langword="null"/>.</exception>
|
|
||||||
/// <exception cref="GatewayException">Thrown when the subscriber's <see cref="ISubscriber.GatewaySubscriptionId"/> is <see langword="null"/> or empty.</exception>
|
|
||||||
/// <exception cref="GatewayException">Thrown when the <see cref="Subscription"/> returned from Stripe's API is null.</exception>
|
|
||||||
Task<Subscription> GetSubscription(ISubscriber subscriber);
|
|
||||||
}
|
|
14
src/Core/Billing/Queries/IProviderBillingQueries.cs
Normal file
14
src/Core/Billing/Queries/IProviderBillingQueries.cs
Normal file
@ -0,0 +1,14 @@
|
|||||||
|
using Bit.Core.Billing.Models;
|
||||||
|
|
||||||
|
namespace Bit.Core.Billing.Queries;
|
||||||
|
|
||||||
|
public interface IProviderBillingQueries
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Retrieves a provider's billing subscription data.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="providerId">The ID of the provider to retrieve subscription data for.</param>
|
||||||
|
/// <returns>A <see cref="ProviderSubscriptionData"/> object containing the provider's Stripe <see cref="Stripe.Subscription"/> and their <see cref="ConfiguredProviderPlan"/>s.</returns>
|
||||||
|
/// <remarks>This method opts for returning <see langword="null"/> rather than throwing exceptions, making it ideal for surfacing data from API endpoints.</remarks>
|
||||||
|
Task<ProviderSubscriptionData> GetSubscriptionData(Guid providerId);
|
||||||
|
}
|
30
src/Core/Billing/Queries/ISubscriberQueries.cs
Normal file
30
src/Core/Billing/Queries/ISubscriberQueries.cs
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
using Bit.Core.Entities;
|
||||||
|
using Bit.Core.Exceptions;
|
||||||
|
using Stripe;
|
||||||
|
|
||||||
|
namespace Bit.Core.Billing.Queries;
|
||||||
|
|
||||||
|
public interface ISubscriberQueries
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Retrieves a Stripe <see cref="Subscription"/> using the <paramref name="subscriber"/>'s <see cref="ISubscriber.GatewaySubscriptionId"/> property.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="subscriber">The organization, provider or user to retrieve the subscription for.</param>
|
||||||
|
/// <param name="subscriptionGetOptions">Optional parameters that can be passed to Stripe to expand or modify the <see cref="Subscription"/>.</param>
|
||||||
|
/// <returns>A Stripe <see cref="Subscription"/>.</returns>
|
||||||
|
/// <exception cref="ArgumentNullException">Thrown when the <paramref name="subscriber"/> is <see langword="null"/>.</exception>
|
||||||
|
/// <remarks>This method opts for returning <see langword="null"/> rather than throwing exceptions, making it ideal for surfacing data from API endpoints.</remarks>
|
||||||
|
Task<Subscription> GetSubscription(
|
||||||
|
ISubscriber subscriber,
|
||||||
|
SubscriptionGetOptions subscriptionGetOptions = null);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Retrieves a Stripe <see cref="Subscription"/> using the <paramref name="subscriber"/>'s <see cref="ISubscriber.GatewaySubscriptionId"/> property.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="subscriber">The organization or user to retrieve the subscription for.</param>
|
||||||
|
/// <returns>A Stripe <see cref="Subscription"/>.</returns>
|
||||||
|
/// <exception cref="ArgumentNullException">Thrown when the <paramref name="subscriber"/> is <see langword="null"/>.</exception>
|
||||||
|
/// <exception cref="GatewayException">Thrown when the subscriber's <see cref="ISubscriber.GatewaySubscriptionId"/> is <see langword="null"/> or empty.</exception>
|
||||||
|
/// <exception cref="GatewayException">Thrown when the <see cref="Subscription"/> returned from Stripe's API is null.</exception>
|
||||||
|
Task<Subscription> GetSubscriptionOrThrow(ISubscriber subscriber);
|
||||||
|
}
|
@ -1,36 +0,0 @@
|
|||||||
using Bit.Core.Entities;
|
|
||||||
using Bit.Core.Services;
|
|
||||||
using Microsoft.Extensions.Logging;
|
|
||||||
using Stripe;
|
|
||||||
|
|
||||||
using static Bit.Core.Billing.Utilities;
|
|
||||||
|
|
||||||
namespace Bit.Core.Billing.Queries.Implementations;
|
|
||||||
|
|
||||||
public class GetSubscriptionQuery(
|
|
||||||
ILogger<GetSubscriptionQuery> logger,
|
|
||||||
IStripeAdapter stripeAdapter) : IGetSubscriptionQuery
|
|
||||||
{
|
|
||||||
public async Task<Subscription> GetSubscription(ISubscriber subscriber)
|
|
||||||
{
|
|
||||||
ArgumentNullException.ThrowIfNull(subscriber);
|
|
||||||
|
|
||||||
if (string.IsNullOrEmpty(subscriber.GatewaySubscriptionId))
|
|
||||||
{
|
|
||||||
logger.LogError("Cannot cancel subscription for subscriber ({ID}) with no GatewaySubscriptionId.", subscriber.Id);
|
|
||||||
|
|
||||||
throw ContactSupport();
|
|
||||||
}
|
|
||||||
|
|
||||||
var subscription = await stripeAdapter.SubscriptionGetAsync(subscriber.GatewaySubscriptionId);
|
|
||||||
|
|
||||||
if (subscription != null)
|
|
||||||
{
|
|
||||||
return subscription;
|
|
||||||
}
|
|
||||||
|
|
||||||
logger.LogError("Could not find Stripe subscription ({ID}) to cancel.", subscriber.GatewaySubscriptionId);
|
|
||||||
|
|
||||||
throw ContactSupport();
|
|
||||||
}
|
|
||||||
}
|
|
@ -0,0 +1,49 @@
|
|||||||
|
using Bit.Core.AdminConsole.Repositories;
|
||||||
|
using Bit.Core.Billing.Models;
|
||||||
|
using Bit.Core.Billing.Repositories;
|
||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
using Stripe;
|
||||||
|
|
||||||
|
namespace Bit.Core.Billing.Queries.Implementations;
|
||||||
|
|
||||||
|
public class ProviderBillingQueries(
|
||||||
|
ILogger<ProviderBillingQueries> logger,
|
||||||
|
IProviderPlanRepository providerPlanRepository,
|
||||||
|
IProviderRepository providerRepository,
|
||||||
|
ISubscriberQueries subscriberQueries) : IProviderBillingQueries
|
||||||
|
{
|
||||||
|
public async Task<ProviderSubscriptionData> GetSubscriptionData(Guid providerId)
|
||||||
|
{
|
||||||
|
var provider = await providerRepository.GetByIdAsync(providerId);
|
||||||
|
|
||||||
|
if (provider == null)
|
||||||
|
{
|
||||||
|
logger.LogError(
|
||||||
|
"Could not find provider ({ID}) when retrieving subscription data.",
|
||||||
|
providerId);
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var subscription = await subscriberQueries.GetSubscription(provider, new SubscriptionGetOptions
|
||||||
|
{
|
||||||
|
Expand = ["customer"]
|
||||||
|
});
|
||||||
|
|
||||||
|
if (subscription == null)
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var providerPlans = await providerPlanRepository.GetByProviderId(providerId);
|
||||||
|
|
||||||
|
var configuredProviderPlans = providerPlans
|
||||||
|
.Where(providerPlan => providerPlan.Configured)
|
||||||
|
.Select(ConfiguredProviderPlan.From)
|
||||||
|
.ToList();
|
||||||
|
|
||||||
|
return new ProviderSubscriptionData(
|
||||||
|
configuredProviderPlans,
|
||||||
|
subscription);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,61 @@
|
|||||||
|
using Bit.Core.Entities;
|
||||||
|
using Bit.Core.Services;
|
||||||
|
using Microsoft.Extensions.Logging;
|
||||||
|
using Stripe;
|
||||||
|
|
||||||
|
using static Bit.Core.Billing.Utilities;
|
||||||
|
|
||||||
|
namespace Bit.Core.Billing.Queries.Implementations;
|
||||||
|
|
||||||
|
public class SubscriberQueries(
|
||||||
|
ILogger<SubscriberQueries> logger,
|
||||||
|
IStripeAdapter stripeAdapter) : ISubscriberQueries
|
||||||
|
{
|
||||||
|
public async Task<Subscription> GetSubscription(
|
||||||
|
ISubscriber subscriber,
|
||||||
|
SubscriptionGetOptions subscriptionGetOptions = null)
|
||||||
|
{
|
||||||
|
ArgumentNullException.ThrowIfNull(subscriber);
|
||||||
|
|
||||||
|
if (string.IsNullOrEmpty(subscriber.GatewaySubscriptionId))
|
||||||
|
{
|
||||||
|
logger.LogError("Cannot cancel subscription for subscriber ({ID}) with no GatewaySubscriptionId.", subscriber.Id);
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var subscription = await stripeAdapter.SubscriptionGetAsync(subscriber.GatewaySubscriptionId, subscriptionGetOptions);
|
||||||
|
|
||||||
|
if (subscription != null)
|
||||||
|
{
|
||||||
|
return subscription;
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.LogError("Could not find Stripe subscription ({ID}) to cancel.", subscriber.GatewaySubscriptionId);
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<Subscription> GetSubscriptionOrThrow(ISubscriber subscriber)
|
||||||
|
{
|
||||||
|
ArgumentNullException.ThrowIfNull(subscriber);
|
||||||
|
|
||||||
|
if (string.IsNullOrEmpty(subscriber.GatewaySubscriptionId))
|
||||||
|
{
|
||||||
|
logger.LogError("Cannot cancel subscription for subscriber ({ID}) with no GatewaySubscriptionId.", subscriber.Id);
|
||||||
|
|
||||||
|
throw ContactSupport();
|
||||||
|
}
|
||||||
|
|
||||||
|
var subscription = await stripeAdapter.SubscriptionGetAsync(subscriber.GatewaySubscriptionId);
|
||||||
|
|
||||||
|
if (subscription != null)
|
||||||
|
{
|
||||||
|
return subscription;
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.LogError("Could not find Stripe subscription ({ID}) to cancel.", subscriber.GatewaySubscriptionId);
|
||||||
|
|
||||||
|
throw ContactSupport();
|
||||||
|
}
|
||||||
|
}
|
@ -5,5 +5,5 @@ namespace Bit.Core.Billing.Repositories;
|
|||||||
|
|
||||||
public interface IProviderPlanRepository : IRepository<ProviderPlan, Guid>
|
public interface IProviderPlanRepository : IRepository<ProviderPlan, Guid>
|
||||||
{
|
{
|
||||||
Task<ProviderPlan> GetByProviderId(Guid providerId);
|
Task<ICollection<ProviderPlan>> GetByProviderId(Guid providerId);
|
||||||
}
|
}
|
||||||
|
@ -1,8 +1,11 @@
|
|||||||
using Bit.Core.Exceptions;
|
namespace Bit.Core.Billing;
|
||||||
|
|
||||||
namespace Bit.Core.Billing;
|
|
||||||
|
|
||||||
public static class Utilities
|
public static class Utilities
|
||||||
{
|
{
|
||||||
public static GatewayException ContactSupport() => new("Something went wrong with your request. Please contact support.");
|
public const string BraintreeCustomerIdKey = "btCustomerId";
|
||||||
|
|
||||||
|
public static BillingException ContactSupport(
|
||||||
|
string internalMessage = null,
|
||||||
|
Exception innerException = null) => new("Something went wrong with your request. Please contact support.",
|
||||||
|
internalMessage, innerException);
|
||||||
}
|
}
|
||||||
|
@ -130,6 +130,7 @@ public static class FeatureFlagKeys
|
|||||||
public const string PM5864DollarThreshold = "PM-5864-dollar-threshold";
|
public const string PM5864DollarThreshold = "PM-5864-dollar-threshold";
|
||||||
public const string AC2101UpdateTrialInitiationEmail = "AC-2101-update-trial-initiation-email";
|
public const string AC2101UpdateTrialInitiationEmail = "AC-2101-update-trial-initiation-email";
|
||||||
public const string ShowPaymentMethodWarningBanners = "show-payment-method-warning-banners";
|
public const string ShowPaymentMethodWarningBanners = "show-payment-method-warning-banners";
|
||||||
|
public const string EnableConsolidatedBilling = "enable-consolidated-billing";
|
||||||
|
|
||||||
public static List<string> GetAllKeys()
|
public static List<string> GetAllKeys()
|
||||||
{
|
{
|
||||||
|
@ -14,7 +14,7 @@ public class ProviderPlanRepository(
|
|||||||
globalSettings.SqlServer.ConnectionString,
|
globalSettings.SqlServer.ConnectionString,
|
||||||
globalSettings.SqlServer.ReadOnlyConnectionString), IProviderPlanRepository
|
globalSettings.SqlServer.ReadOnlyConnectionString), IProviderPlanRepository
|
||||||
{
|
{
|
||||||
public async Task<ProviderPlan> GetByProviderId(Guid providerId)
|
public async Task<ICollection<ProviderPlan>> GetByProviderId(Guid providerId)
|
||||||
{
|
{
|
||||||
var sqlConnection = new SqlConnection(ConnectionString);
|
var sqlConnection = new SqlConnection(ConnectionString);
|
||||||
|
|
||||||
@ -23,6 +23,6 @@ public class ProviderPlanRepository(
|
|||||||
new { ProviderId = providerId },
|
new { ProviderId = providerId },
|
||||||
commandType: CommandType.StoredProcedure);
|
commandType: CommandType.StoredProcedure);
|
||||||
|
|
||||||
return results.FirstOrDefault();
|
return results.ToArray();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -16,14 +16,17 @@ public class ProviderPlanRepository(
|
|||||||
mapper,
|
mapper,
|
||||||
context => context.ProviderPlans), IProviderPlanRepository
|
context => context.ProviderPlans), IProviderPlanRepository
|
||||||
{
|
{
|
||||||
public async Task<ProviderPlan> GetByProviderId(Guid providerId)
|
public async Task<ICollection<ProviderPlan>> GetByProviderId(Guid providerId)
|
||||||
{
|
{
|
||||||
using var serviceScope = ServiceScopeFactory.CreateScope();
|
using var serviceScope = ServiceScopeFactory.CreateScope();
|
||||||
|
|
||||||
var databaseContext = GetDatabaseContext(serviceScope);
|
var databaseContext = GetDatabaseContext(serviceScope);
|
||||||
|
|
||||||
var query =
|
var query =
|
||||||
from providerPlan in databaseContext.ProviderPlans
|
from providerPlan in databaseContext.ProviderPlans
|
||||||
where providerPlan.ProviderId == providerId
|
where providerPlan.ProviderId == providerId
|
||||||
select providerPlan;
|
select providerPlan;
|
||||||
return await query.FirstOrDefaultAsync();
|
|
||||||
|
return await query.ToArrayAsync();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -56,7 +56,7 @@ public class OrganizationsControllerTests : IDisposable
|
|||||||
private readonly IAddSecretsManagerSubscriptionCommand _addSecretsManagerSubscriptionCommand;
|
private readonly IAddSecretsManagerSubscriptionCommand _addSecretsManagerSubscriptionCommand;
|
||||||
private readonly IPushNotificationService _pushNotificationService;
|
private readonly IPushNotificationService _pushNotificationService;
|
||||||
private readonly ICancelSubscriptionCommand _cancelSubscriptionCommand;
|
private readonly ICancelSubscriptionCommand _cancelSubscriptionCommand;
|
||||||
private readonly IGetSubscriptionQuery _getSubscriptionQuery;
|
private readonly ISubscriberQueries _subscriberQueries;
|
||||||
private readonly IReferenceEventService _referenceEventService;
|
private readonly IReferenceEventService _referenceEventService;
|
||||||
private readonly IOrganizationEnableCollectionEnhancementsCommand _organizationEnableCollectionEnhancementsCommand;
|
private readonly IOrganizationEnableCollectionEnhancementsCommand _organizationEnableCollectionEnhancementsCommand;
|
||||||
|
|
||||||
@ -86,7 +86,7 @@ public class OrganizationsControllerTests : IDisposable
|
|||||||
_addSecretsManagerSubscriptionCommand = Substitute.For<IAddSecretsManagerSubscriptionCommand>();
|
_addSecretsManagerSubscriptionCommand = Substitute.For<IAddSecretsManagerSubscriptionCommand>();
|
||||||
_pushNotificationService = Substitute.For<IPushNotificationService>();
|
_pushNotificationService = Substitute.For<IPushNotificationService>();
|
||||||
_cancelSubscriptionCommand = Substitute.For<ICancelSubscriptionCommand>();
|
_cancelSubscriptionCommand = Substitute.For<ICancelSubscriptionCommand>();
|
||||||
_getSubscriptionQuery = Substitute.For<IGetSubscriptionQuery>();
|
_subscriberQueries = Substitute.For<ISubscriberQueries>();
|
||||||
_referenceEventService = Substitute.For<IReferenceEventService>();
|
_referenceEventService = Substitute.For<IReferenceEventService>();
|
||||||
_organizationEnableCollectionEnhancementsCommand = Substitute.For<IOrganizationEnableCollectionEnhancementsCommand>();
|
_organizationEnableCollectionEnhancementsCommand = Substitute.For<IOrganizationEnableCollectionEnhancementsCommand>();
|
||||||
|
|
||||||
@ -113,7 +113,7 @@ public class OrganizationsControllerTests : IDisposable
|
|||||||
_addSecretsManagerSubscriptionCommand,
|
_addSecretsManagerSubscriptionCommand,
|
||||||
_pushNotificationService,
|
_pushNotificationService,
|
||||||
_cancelSubscriptionCommand,
|
_cancelSubscriptionCommand,
|
||||||
_getSubscriptionQuery,
|
_subscriberQueries,
|
||||||
_referenceEventService,
|
_referenceEventService,
|
||||||
_organizationEnableCollectionEnhancementsCommand);
|
_organizationEnableCollectionEnhancementsCommand);
|
||||||
}
|
}
|
||||||
|
@ -57,7 +57,7 @@ public class AccountsControllerTests : IDisposable
|
|||||||
private readonly IRotateUserKeyCommand _rotateUserKeyCommand;
|
private readonly IRotateUserKeyCommand _rotateUserKeyCommand;
|
||||||
private readonly IFeatureService _featureService;
|
private readonly IFeatureService _featureService;
|
||||||
private readonly ICancelSubscriptionCommand _cancelSubscriptionCommand;
|
private readonly ICancelSubscriptionCommand _cancelSubscriptionCommand;
|
||||||
private readonly IGetSubscriptionQuery _getSubscriptionQuery;
|
private readonly ISubscriberQueries _subscriberQueries;
|
||||||
private readonly IReferenceEventService _referenceEventService;
|
private readonly IReferenceEventService _referenceEventService;
|
||||||
private readonly ICurrentContext _currentContext;
|
private readonly ICurrentContext _currentContext;
|
||||||
|
|
||||||
@ -90,7 +90,7 @@ public class AccountsControllerTests : IDisposable
|
|||||||
_rotateUserKeyCommand = Substitute.For<IRotateUserKeyCommand>();
|
_rotateUserKeyCommand = Substitute.For<IRotateUserKeyCommand>();
|
||||||
_featureService = Substitute.For<IFeatureService>();
|
_featureService = Substitute.For<IFeatureService>();
|
||||||
_cancelSubscriptionCommand = Substitute.For<ICancelSubscriptionCommand>();
|
_cancelSubscriptionCommand = Substitute.For<ICancelSubscriptionCommand>();
|
||||||
_getSubscriptionQuery = Substitute.For<IGetSubscriptionQuery>();
|
_subscriberQueries = Substitute.For<ISubscriberQueries>();
|
||||||
_referenceEventService = Substitute.For<IReferenceEventService>();
|
_referenceEventService = Substitute.For<IReferenceEventService>();
|
||||||
_currentContext = Substitute.For<ICurrentContext>();
|
_currentContext = Substitute.For<ICurrentContext>();
|
||||||
_cipherValidator =
|
_cipherValidator =
|
||||||
@ -122,7 +122,7 @@ public class AccountsControllerTests : IDisposable
|
|||||||
_rotateUserKeyCommand,
|
_rotateUserKeyCommand,
|
||||||
_featureService,
|
_featureService,
|
||||||
_cancelSubscriptionCommand,
|
_cancelSubscriptionCommand,
|
||||||
_getSubscriptionQuery,
|
_subscriberQueries,
|
||||||
_referenceEventService,
|
_referenceEventService,
|
||||||
_currentContext,
|
_currentContext,
|
||||||
_cipherValidator,
|
_cipherValidator,
|
||||||
|
@ -1,13 +1,13 @@
|
|||||||
using Bit.Core.AdminConsole.Entities;
|
using Bit.Core.AdminConsole.Entities;
|
||||||
using Bit.Core.Billing.Commands.Implementations;
|
using Bit.Core.Billing.Commands.Implementations;
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.Exceptions;
|
|
||||||
using Bit.Core.Services;
|
using Bit.Core.Services;
|
||||||
using Bit.Test.Common.AutoFixture;
|
using Bit.Test.Common.AutoFixture;
|
||||||
using Bit.Test.Common.AutoFixture.Attributes;
|
using Bit.Test.Common.AutoFixture.Attributes;
|
||||||
using NSubstitute;
|
using NSubstitute;
|
||||||
using NSubstitute.ReturnsExtensions;
|
using NSubstitute.ReturnsExtensions;
|
||||||
using Xunit;
|
using Xunit;
|
||||||
|
using static Bit.Core.Test.Billing.Utilities;
|
||||||
using BT = Braintree;
|
using BT = Braintree;
|
||||||
using S = Stripe;
|
using S = Stripe;
|
||||||
|
|
||||||
@ -355,13 +355,4 @@ public class RemovePaymentMethodCommandTests
|
|||||||
|
|
||||||
return (braintreeGateway, customerGateway, paymentMethodGateway);
|
return (braintreeGateway, customerGateway, paymentMethodGateway);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static async Task ThrowsContactSupportAsync(Func<Task> function)
|
|
||||||
{
|
|
||||||
const string message = "Could not remove your payment method. Please contact support for assistance.";
|
|
||||||
|
|
||||||
var exception = await Assert.ThrowsAsync<GatewayException>(function);
|
|
||||||
|
|
||||||
Assert.Equal(message, exception.Message);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -1,104 +0,0 @@
|
|||||||
using Bit.Core.AdminConsole.Entities;
|
|
||||||
using Bit.Core.Billing.Queries.Implementations;
|
|
||||||
using Bit.Core.Entities;
|
|
||||||
using Bit.Core.Exceptions;
|
|
||||||
using Bit.Core.Services;
|
|
||||||
using Bit.Test.Common.AutoFixture;
|
|
||||||
using Bit.Test.Common.AutoFixture.Attributes;
|
|
||||||
using NSubstitute;
|
|
||||||
using NSubstitute.ReturnsExtensions;
|
|
||||||
using Stripe;
|
|
||||||
using Xunit;
|
|
||||||
|
|
||||||
namespace Bit.Core.Test.Billing.Queries;
|
|
||||||
|
|
||||||
[SutProviderCustomize]
|
|
||||||
public class GetSubscriptionQueryTests
|
|
||||||
{
|
|
||||||
[Theory, BitAutoData]
|
|
||||||
public async Task GetSubscription_NullSubscriber_ThrowsArgumentNullException(
|
|
||||||
SutProvider<GetSubscriptionQuery> sutProvider)
|
|
||||||
=> await Assert.ThrowsAsync<ArgumentNullException>(
|
|
||||||
async () => await sutProvider.Sut.GetSubscription(null));
|
|
||||||
|
|
||||||
[Theory, BitAutoData]
|
|
||||||
public async Task GetSubscription_Organization_NoGatewaySubscriptionId_ThrowsGatewayException(
|
|
||||||
Organization organization,
|
|
||||||
SutProvider<GetSubscriptionQuery> sutProvider)
|
|
||||||
{
|
|
||||||
organization.GatewaySubscriptionId = null;
|
|
||||||
|
|
||||||
await ThrowsContactSupportAsync(async () => await sutProvider.Sut.GetSubscription(organization));
|
|
||||||
}
|
|
||||||
|
|
||||||
[Theory, BitAutoData]
|
|
||||||
public async Task GetSubscription_Organization_NoSubscription_ThrowsGatewayException(
|
|
||||||
Organization organization,
|
|
||||||
SutProvider<GetSubscriptionQuery> sutProvider)
|
|
||||||
{
|
|
||||||
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(organization.GatewaySubscriptionId)
|
|
||||||
.ReturnsNull();
|
|
||||||
|
|
||||||
await ThrowsContactSupportAsync(async () => await sutProvider.Sut.GetSubscription(organization));
|
|
||||||
}
|
|
||||||
|
|
||||||
[Theory, BitAutoData]
|
|
||||||
public async Task GetSubscription_Organization_Succeeds(
|
|
||||||
Organization organization,
|
|
||||||
SutProvider<GetSubscriptionQuery> sutProvider)
|
|
||||||
{
|
|
||||||
var subscription = new Subscription();
|
|
||||||
|
|
||||||
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(organization.GatewaySubscriptionId)
|
|
||||||
.Returns(subscription);
|
|
||||||
|
|
||||||
var gotSubscription = await sutProvider.Sut.GetSubscription(organization);
|
|
||||||
|
|
||||||
Assert.Equivalent(subscription, gotSubscription);
|
|
||||||
}
|
|
||||||
|
|
||||||
[Theory, BitAutoData]
|
|
||||||
public async Task GetSubscription_User_NoGatewaySubscriptionId_ThrowsGatewayException(
|
|
||||||
User user,
|
|
||||||
SutProvider<GetSubscriptionQuery> sutProvider)
|
|
||||||
{
|
|
||||||
user.GatewaySubscriptionId = null;
|
|
||||||
|
|
||||||
await ThrowsContactSupportAsync(async () => await sutProvider.Sut.GetSubscription(user));
|
|
||||||
}
|
|
||||||
|
|
||||||
[Theory, BitAutoData]
|
|
||||||
public async Task GetSubscription_User_NoSubscription_ThrowsGatewayException(
|
|
||||||
User user,
|
|
||||||
SutProvider<GetSubscriptionQuery> sutProvider)
|
|
||||||
{
|
|
||||||
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(user.GatewaySubscriptionId)
|
|
||||||
.ReturnsNull();
|
|
||||||
|
|
||||||
await ThrowsContactSupportAsync(async () => await sutProvider.Sut.GetSubscription(user));
|
|
||||||
}
|
|
||||||
|
|
||||||
[Theory, BitAutoData]
|
|
||||||
public async Task GetSubscription_User_Succeeds(
|
|
||||||
User user,
|
|
||||||
SutProvider<GetSubscriptionQuery> sutProvider)
|
|
||||||
{
|
|
||||||
var subscription = new Subscription();
|
|
||||||
|
|
||||||
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(user.GatewaySubscriptionId)
|
|
||||||
.Returns(subscription);
|
|
||||||
|
|
||||||
var gotSubscription = await sutProvider.Sut.GetSubscription(user);
|
|
||||||
|
|
||||||
Assert.Equivalent(subscription, gotSubscription);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static async Task ThrowsContactSupportAsync(Func<Task> function)
|
|
||||||
{
|
|
||||||
const string message = "Something went wrong with your request. Please contact support.";
|
|
||||||
|
|
||||||
var exception = await Assert.ThrowsAsync<GatewayException>(function);
|
|
||||||
|
|
||||||
Assert.Equal(message, exception.Message);
|
|
||||||
}
|
|
||||||
}
|
|
151
test/Core.Test/Billing/Queries/ProviderBillingQueriesTests.cs
Normal file
151
test/Core.Test/Billing/Queries/ProviderBillingQueriesTests.cs
Normal file
@ -0,0 +1,151 @@
|
|||||||
|
using Bit.Core.AdminConsole.Entities.Provider;
|
||||||
|
using Bit.Core.AdminConsole.Repositories;
|
||||||
|
using Bit.Core.Billing.Entities;
|
||||||
|
using Bit.Core.Billing.Models;
|
||||||
|
using Bit.Core.Billing.Queries;
|
||||||
|
using Bit.Core.Billing.Queries.Implementations;
|
||||||
|
using Bit.Core.Billing.Repositories;
|
||||||
|
using Bit.Core.Enums;
|
||||||
|
using Bit.Test.Common.AutoFixture;
|
||||||
|
using Bit.Test.Common.AutoFixture.Attributes;
|
||||||
|
using NSubstitute;
|
||||||
|
using NSubstitute.ReturnsExtensions;
|
||||||
|
using Stripe;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
namespace Bit.Core.Test.Billing.Queries;
|
||||||
|
|
||||||
|
[SutProviderCustomize]
|
||||||
|
public class ProviderBillingQueriesTests
|
||||||
|
{
|
||||||
|
#region GetSubscriptionData
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscriptionData_NullProvider_ReturnsNull(
|
||||||
|
SutProvider<ProviderBillingQueries> sutProvider,
|
||||||
|
Guid providerId)
|
||||||
|
{
|
||||||
|
var providerRepository = sutProvider.GetDependency<IProviderRepository>();
|
||||||
|
|
||||||
|
providerRepository.GetByIdAsync(providerId).ReturnsNull();
|
||||||
|
|
||||||
|
var subscriptionData = await sutProvider.Sut.GetSubscriptionData(providerId);
|
||||||
|
|
||||||
|
Assert.Null(subscriptionData);
|
||||||
|
|
||||||
|
await providerRepository.Received(1).GetByIdAsync(providerId);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscriptionData_NullSubscription_ReturnsNull(
|
||||||
|
SutProvider<ProviderBillingQueries> sutProvider,
|
||||||
|
Guid providerId,
|
||||||
|
Provider provider)
|
||||||
|
{
|
||||||
|
var providerRepository = sutProvider.GetDependency<IProviderRepository>();
|
||||||
|
|
||||||
|
providerRepository.GetByIdAsync(providerId).Returns(provider);
|
||||||
|
|
||||||
|
var subscriberQueries = sutProvider.GetDependency<ISubscriberQueries>();
|
||||||
|
|
||||||
|
subscriberQueries.GetSubscription(provider).ReturnsNull();
|
||||||
|
|
||||||
|
var subscriptionData = await sutProvider.Sut.GetSubscriptionData(providerId);
|
||||||
|
|
||||||
|
Assert.Null(subscriptionData);
|
||||||
|
|
||||||
|
await providerRepository.Received(1).GetByIdAsync(providerId);
|
||||||
|
|
||||||
|
await subscriberQueries.Received(1).GetSubscription(
|
||||||
|
provider,
|
||||||
|
Arg.Is<SubscriptionGetOptions>(
|
||||||
|
options => options.Expand.Count == 1 && options.Expand.First() == "customer"));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscriptionData_Success(
|
||||||
|
SutProvider<ProviderBillingQueries> sutProvider,
|
||||||
|
Guid providerId,
|
||||||
|
Provider provider)
|
||||||
|
{
|
||||||
|
var providerRepository = sutProvider.GetDependency<IProviderRepository>();
|
||||||
|
|
||||||
|
providerRepository.GetByIdAsync(providerId).Returns(provider);
|
||||||
|
|
||||||
|
var subscriberQueries = sutProvider.GetDependency<ISubscriberQueries>();
|
||||||
|
|
||||||
|
var subscription = new Subscription();
|
||||||
|
|
||||||
|
subscriberQueries.GetSubscription(provider, Arg.Is<SubscriptionGetOptions>(
|
||||||
|
options => options.Expand.Count == 1 && options.Expand.First() == "customer")).Returns(subscription);
|
||||||
|
|
||||||
|
var providerPlanRepository = sutProvider.GetDependency<IProviderPlanRepository>();
|
||||||
|
|
||||||
|
var enterprisePlan = new ProviderPlan
|
||||||
|
{
|
||||||
|
Id = Guid.NewGuid(),
|
||||||
|
ProviderId = providerId,
|
||||||
|
PlanType = PlanType.EnterpriseMonthly,
|
||||||
|
SeatMinimum = 100,
|
||||||
|
PurchasedSeats = 0
|
||||||
|
};
|
||||||
|
|
||||||
|
var teamsPlan = new ProviderPlan
|
||||||
|
{
|
||||||
|
Id = Guid.NewGuid(),
|
||||||
|
ProviderId = providerId,
|
||||||
|
PlanType = PlanType.TeamsMonthly,
|
||||||
|
SeatMinimum = 50,
|
||||||
|
PurchasedSeats = 10
|
||||||
|
};
|
||||||
|
|
||||||
|
var providerPlans = new List<ProviderPlan>
|
||||||
|
{
|
||||||
|
enterprisePlan,
|
||||||
|
teamsPlan,
|
||||||
|
};
|
||||||
|
|
||||||
|
providerPlanRepository.GetByProviderId(providerId).Returns(providerPlans);
|
||||||
|
|
||||||
|
var subscriptionData = await sutProvider.Sut.GetSubscriptionData(providerId);
|
||||||
|
|
||||||
|
Assert.NotNull(subscriptionData);
|
||||||
|
|
||||||
|
Assert.Equivalent(subscriptionData.Subscription, subscription);
|
||||||
|
|
||||||
|
Assert.Equal(2, subscriptionData.ProviderPlans.Count);
|
||||||
|
|
||||||
|
var configuredEnterprisePlan =
|
||||||
|
subscriptionData.ProviderPlans.FirstOrDefault(configuredPlan =>
|
||||||
|
configuredPlan.PlanType == PlanType.EnterpriseMonthly);
|
||||||
|
|
||||||
|
var configuredTeamsPlan =
|
||||||
|
subscriptionData.ProviderPlans.FirstOrDefault(configuredPlan =>
|
||||||
|
configuredPlan.PlanType == PlanType.TeamsMonthly);
|
||||||
|
|
||||||
|
Compare(enterprisePlan, configuredEnterprisePlan);
|
||||||
|
|
||||||
|
Compare(teamsPlan, configuredTeamsPlan);
|
||||||
|
|
||||||
|
await providerRepository.Received(1).GetByIdAsync(providerId);
|
||||||
|
|
||||||
|
await subscriberQueries.Received(1).GetSubscription(
|
||||||
|
provider,
|
||||||
|
Arg.Is<SubscriptionGetOptions>(
|
||||||
|
options => options.Expand.Count == 1 && options.Expand.First() == "customer"));
|
||||||
|
|
||||||
|
await providerPlanRepository.Received(1).GetByProviderId(providerId);
|
||||||
|
|
||||||
|
return;
|
||||||
|
|
||||||
|
void Compare(ProviderPlan providerPlan, ConfiguredProviderPlan configuredProviderPlan)
|
||||||
|
{
|
||||||
|
Assert.NotNull(configuredProviderPlan);
|
||||||
|
Assert.Equal(providerPlan.Id, configuredProviderPlan.Id);
|
||||||
|
Assert.Equal(providerPlan.ProviderId, configuredProviderPlan.ProviderId);
|
||||||
|
Assert.Equal(providerPlan.SeatMinimum!.Value, configuredProviderPlan.SeatMinimum);
|
||||||
|
Assert.Equal(providerPlan.PurchasedSeats!.Value, configuredProviderPlan.PurchasedSeats);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endregion
|
||||||
|
}
|
263
test/Core.Test/Billing/Queries/SubscriberQueriesTests.cs
Normal file
263
test/Core.Test/Billing/Queries/SubscriberQueriesTests.cs
Normal file
@ -0,0 +1,263 @@
|
|||||||
|
using Bit.Core.AdminConsole.Entities;
|
||||||
|
using Bit.Core.AdminConsole.Entities.Provider;
|
||||||
|
using Bit.Core.Billing.Queries.Implementations;
|
||||||
|
using Bit.Core.Entities;
|
||||||
|
using Bit.Core.Services;
|
||||||
|
using Bit.Test.Common.AutoFixture;
|
||||||
|
using Bit.Test.Common.AutoFixture.Attributes;
|
||||||
|
using NSubstitute;
|
||||||
|
using NSubstitute.ReturnsExtensions;
|
||||||
|
using Stripe;
|
||||||
|
using Xunit;
|
||||||
|
|
||||||
|
using static Bit.Core.Test.Billing.Utilities;
|
||||||
|
|
||||||
|
namespace Bit.Core.Test.Billing.Queries;
|
||||||
|
|
||||||
|
[SutProviderCustomize]
|
||||||
|
public class SubscriberQueriesTests
|
||||||
|
{
|
||||||
|
#region GetSubscription
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscription_NullSubscriber_ThrowsArgumentNullException(
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
=> await Assert.ThrowsAsync<ArgumentNullException>(
|
||||||
|
async () => await sutProvider.Sut.GetSubscription(null));
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscription_Organization_NoGatewaySubscriptionId_ReturnsNull(
|
||||||
|
Organization organization,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
organization.GatewaySubscriptionId = null;
|
||||||
|
|
||||||
|
var gotSubscription = await sutProvider.Sut.GetSubscription(organization);
|
||||||
|
|
||||||
|
Assert.Null(gotSubscription);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscription_Organization_NoSubscription_ReturnsNull(
|
||||||
|
Organization organization,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(organization.GatewaySubscriptionId)
|
||||||
|
.ReturnsNull();
|
||||||
|
|
||||||
|
var gotSubscription = await sutProvider.Sut.GetSubscription(organization);
|
||||||
|
|
||||||
|
Assert.Null(gotSubscription);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscription_Organization_Succeeds(
|
||||||
|
Organization organization,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription();
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(organization.GatewaySubscriptionId)
|
||||||
|
.Returns(subscription);
|
||||||
|
|
||||||
|
var gotSubscription = await sutProvider.Sut.GetSubscription(organization);
|
||||||
|
|
||||||
|
Assert.Equivalent(subscription, gotSubscription);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscription_User_NoGatewaySubscriptionId_ReturnsNull(
|
||||||
|
User user,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
user.GatewaySubscriptionId = null;
|
||||||
|
|
||||||
|
var gotSubscription = await sutProvider.Sut.GetSubscription(user);
|
||||||
|
|
||||||
|
Assert.Null(gotSubscription);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscription_User_NoSubscription_ReturnsNull(
|
||||||
|
User user,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(user.GatewaySubscriptionId)
|
||||||
|
.ReturnsNull();
|
||||||
|
|
||||||
|
var gotSubscription = await sutProvider.Sut.GetSubscription(user);
|
||||||
|
|
||||||
|
Assert.Null(gotSubscription);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscription_User_Succeeds(
|
||||||
|
User user,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription();
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(user.GatewaySubscriptionId)
|
||||||
|
.Returns(subscription);
|
||||||
|
|
||||||
|
var gotSubscription = await sutProvider.Sut.GetSubscription(user);
|
||||||
|
|
||||||
|
Assert.Equivalent(subscription, gotSubscription);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscription_Provider_NoGatewaySubscriptionId_ReturnsNull(
|
||||||
|
Provider provider,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
provider.GatewaySubscriptionId = null;
|
||||||
|
|
||||||
|
var gotSubscription = await sutProvider.Sut.GetSubscription(provider);
|
||||||
|
|
||||||
|
Assert.Null(gotSubscription);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscription_Provider_NoSubscription_ReturnsNull(
|
||||||
|
Provider provider,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(provider.GatewaySubscriptionId)
|
||||||
|
.ReturnsNull();
|
||||||
|
|
||||||
|
var gotSubscription = await sutProvider.Sut.GetSubscription(provider);
|
||||||
|
|
||||||
|
Assert.Null(gotSubscription);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscription_Provider_Succeeds(
|
||||||
|
Provider provider,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription();
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(provider.GatewaySubscriptionId)
|
||||||
|
.Returns(subscription);
|
||||||
|
|
||||||
|
var gotSubscription = await sutProvider.Sut.GetSubscription(provider);
|
||||||
|
|
||||||
|
Assert.Equivalent(subscription, gotSubscription);
|
||||||
|
}
|
||||||
|
#endregion
|
||||||
|
|
||||||
|
#region GetSubscriptionOrThrow
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscriptionOrThrow_NullSubscriber_ThrowsArgumentNullException(
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
=> await Assert.ThrowsAsync<ArgumentNullException>(
|
||||||
|
async () => await sutProvider.Sut.GetSubscriptionOrThrow(null));
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscriptionOrThrow_Organization_NoGatewaySubscriptionId_ThrowsGatewayException(
|
||||||
|
Organization organization,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
organization.GatewaySubscriptionId = null;
|
||||||
|
|
||||||
|
await ThrowsContactSupportAsync(async () => await sutProvider.Sut.GetSubscriptionOrThrow(organization));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscriptionOrThrow_Organization_NoSubscription_ThrowsGatewayException(
|
||||||
|
Organization organization,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(organization.GatewaySubscriptionId)
|
||||||
|
.ReturnsNull();
|
||||||
|
|
||||||
|
await ThrowsContactSupportAsync(async () => await sutProvider.Sut.GetSubscriptionOrThrow(organization));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscriptionOrThrow_Organization_Succeeds(
|
||||||
|
Organization organization,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription();
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(organization.GatewaySubscriptionId)
|
||||||
|
.Returns(subscription);
|
||||||
|
|
||||||
|
var gotSubscription = await sutProvider.Sut.GetSubscriptionOrThrow(organization);
|
||||||
|
|
||||||
|
Assert.Equivalent(subscription, gotSubscription);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscriptionOrThrow_User_NoGatewaySubscriptionId_ThrowsGatewayException(
|
||||||
|
User user,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
user.GatewaySubscriptionId = null;
|
||||||
|
|
||||||
|
await ThrowsContactSupportAsync(async () => await sutProvider.Sut.GetSubscriptionOrThrow(user));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscriptionOrThrow_User_NoSubscription_ThrowsGatewayException(
|
||||||
|
User user,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(user.GatewaySubscriptionId)
|
||||||
|
.ReturnsNull();
|
||||||
|
|
||||||
|
await ThrowsContactSupportAsync(async () => await sutProvider.Sut.GetSubscriptionOrThrow(user));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscriptionOrThrow_User_Succeeds(
|
||||||
|
User user,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription();
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(user.GatewaySubscriptionId)
|
||||||
|
.Returns(subscription);
|
||||||
|
|
||||||
|
var gotSubscription = await sutProvider.Sut.GetSubscriptionOrThrow(user);
|
||||||
|
|
||||||
|
Assert.Equivalent(subscription, gotSubscription);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscriptionOrThrow_Provider_NoGatewaySubscriptionId_ThrowsGatewayException(
|
||||||
|
Provider provider,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
provider.GatewaySubscriptionId = null;
|
||||||
|
|
||||||
|
await ThrowsContactSupportAsync(async () => await sutProvider.Sut.GetSubscriptionOrThrow(provider));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscriptionOrThrow_Provider_NoSubscription_ThrowsGatewayException(
|
||||||
|
Provider provider,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(provider.GatewaySubscriptionId)
|
||||||
|
.ReturnsNull();
|
||||||
|
|
||||||
|
await ThrowsContactSupportAsync(async () => await sutProvider.Sut.GetSubscriptionOrThrow(provider));
|
||||||
|
}
|
||||||
|
|
||||||
|
[Theory, BitAutoData]
|
||||||
|
public async Task GetSubscriptionOrThrow_Provider_Succeeds(
|
||||||
|
Provider provider,
|
||||||
|
SutProvider<SubscriberQueries> sutProvider)
|
||||||
|
{
|
||||||
|
var subscription = new Subscription();
|
||||||
|
|
||||||
|
sutProvider.GetDependency<IStripeAdapter>().SubscriptionGetAsync(provider.GatewaySubscriptionId)
|
||||||
|
.Returns(subscription);
|
||||||
|
|
||||||
|
var gotSubscription = await sutProvider.Sut.GetSubscriptionOrThrow(provider);
|
||||||
|
|
||||||
|
Assert.Equivalent(subscription, gotSubscription);
|
||||||
|
}
|
||||||
|
#endregion
|
||||||
|
}
|
@ -1,4 +1,4 @@
|
|||||||
using Bit.Core.Exceptions;
|
using Bit.Core.Billing;
|
||||||
using Xunit;
|
using Xunit;
|
||||||
|
|
||||||
using static Bit.Core.Billing.Utilities;
|
using static Bit.Core.Billing.Utilities;
|
||||||
@ -11,7 +11,7 @@ public static class Utilities
|
|||||||
{
|
{
|
||||||
var contactSupport = ContactSupport();
|
var contactSupport = ContactSupport();
|
||||||
|
|
||||||
var exception = await Assert.ThrowsAsync<GatewayException>(function);
|
var exception = await Assert.ThrowsAsync<BillingException>(function);
|
||||||
|
|
||||||
Assert.Equal(contactSupport.Message, exception.Message);
|
Assert.Equal(contactSupport.Message, exception.Message);
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user