mirror of
https://github.com/bitwarden/server
synced 2026-01-02 00:23:40 +00:00
* move billing services+tests to billing namespaces * reorganized methods in file and added comment headers * renamed StripeAdapter methods for better clarity * clean up redundant qualifiers * Upgrade Stripe.net to v48.4.0 * Update PreviewTaxAmountCommand * Remove unused UpcomingInvoiceOptionExtensions * Added SubscriptionExtensions with GetCurrentPeriodEnd * Update PremiumUserBillingService * Update OrganizationBillingService * Update GetOrganizationWarningsQuery * Update BillingHistoryInfo * Update SubscriptionInfo * Remove unused Sql Billing folder * Update StripeAdapter * Update StripePaymentService * Update InvoiceCreatedHandler * Update PaymentFailedHandler * Update PaymentSucceededHandler * Update ProviderEventService * Update StripeEventUtilityService * Update SubscriptionDeletedHandler * Update SubscriptionUpdatedHandler * Update UpcomingInvoiceHandler * Update ProviderSubscriptionResponse * Remove unused Stripe Subscriptions Admin Tool * Update RemoveOrganizationFromProviderCommand * Update ProviderBillingService * Update RemoveOrganizatinoFromProviderCommandTests * Update PreviewTaxAmountCommandTests * Update GetCloudOrganizationLicenseQueryTests * Update GetOrganizationWarningsQueryTests * Update StripePaymentServiceTests * Update ProviderBillingControllerTests * Update ProviderEventServiceTests * Update SubscriptionDeletedHandlerTests * Update SubscriptionUpdatedHandlerTests * Resolve Billing test failures I completely removed tests for the StripeEventService as they were using a system I setup a while back that read JSON files of the Stripe event structure. I did not anticipate how frequently these structures would change with each API version and the cost of trying to update these specific JSON files to test a very static data retrieval service far outweigh the benefit. * Resolve Core test failures * Run dotnet format * Remove unused provider migration * Fixed failing tests * Run dotnet format * Replace the old webhook secret key with new one (#6223) * Fix compilation failures in additions * Run dotnet format * Bump Stripe API version * Fix recent addition: CreatePremiumCloudHostedSubscriptionCommand * Fix new code in main according to Stripe update * Fix InvoiceExtensions * Bump SDK version to match API Version * cleanup * fixing items missed after the merge * use expression body for all simple returns * forgot fixes, format, and pr feedback * claude pr feedback * pr feedback and cleanup * more claude feedback --------- Co-authored-by: Alex Morask <amorask@bitwarden.com> Co-authored-by: cyprain-okeke <108260115+cyprain-okeke@users.noreply.github.com>
97 lines
3.0 KiB
C#
97 lines
3.0 KiB
C#
// FIXME: Update this file to be null safe and then delete the line below
|
|
#nullable disable
|
|
|
|
using Bit.Core.Billing.Models;
|
|
using Bit.Core.Billing.Services;
|
|
using Bit.Core.Billing.Tax.Models;
|
|
using Stripe;
|
|
|
|
namespace Bit.Core.Billing;
|
|
|
|
public static class Utilities
|
|
{
|
|
public const string BraintreeCustomerIdKey = "btCustomerId";
|
|
public const string BraintreeCustomerIdOldKey = "btCustomerId_old";
|
|
|
|
public static async Task<SubscriptionSuspension> GetSubscriptionSuspensionAsync(
|
|
IStripeAdapter stripeAdapter,
|
|
Subscription subscription)
|
|
{
|
|
if (subscription.Status is not "past_due" && subscription.Status is not "unpaid")
|
|
{
|
|
return null;
|
|
}
|
|
|
|
var openInvoices = await stripeAdapter.SearchInvoiceAsync(new InvoiceSearchOptions
|
|
{
|
|
Query = $"subscription:'{subscription.Id}' status:'open'"
|
|
});
|
|
|
|
if (openInvoices.Count == 0)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
var currentDate = subscription.TestClock?.FrozenTime ?? DateTime.UtcNow;
|
|
|
|
switch (subscription.CollectionMethod)
|
|
{
|
|
case "charge_automatically":
|
|
{
|
|
var firstOverdueInvoice = openInvoices
|
|
.Where(invoice => invoice.PeriodEnd < currentDate && invoice.Attempted)
|
|
.MinBy(invoice => invoice.Created);
|
|
|
|
if (firstOverdueInvoice == null)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
const int gracePeriod = 14;
|
|
|
|
return new SubscriptionSuspension(
|
|
firstOverdueInvoice.Created.AddDays(gracePeriod),
|
|
firstOverdueInvoice.PeriodEnd,
|
|
gracePeriod);
|
|
}
|
|
case "send_invoice":
|
|
{
|
|
var firstOverdueInvoice = openInvoices
|
|
.Where(invoice => invoice.DueDate < currentDate)
|
|
.MinBy(invoice => invoice.Created);
|
|
|
|
if (firstOverdueInvoice?.DueDate == null)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
const int gracePeriod = 30;
|
|
|
|
return new SubscriptionSuspension(
|
|
firstOverdueInvoice.DueDate.Value.AddDays(gracePeriod),
|
|
firstOverdueInvoice.PeriodEnd,
|
|
gracePeriod);
|
|
}
|
|
default: return null;
|
|
}
|
|
}
|
|
|
|
public static TaxInformation GetTaxInformation(Customer customer)
|
|
{
|
|
if (customer.Address == null)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
return new TaxInformation(
|
|
customer.Address.Country,
|
|
customer.Address.PostalCode,
|
|
customer.TaxIds?.FirstOrDefault()?.Value,
|
|
customer.TaxIds?.FirstOrDefault()?.Type,
|
|
customer.Address.Line1,
|
|
customer.Address.Line2,
|
|
customer.Address.City,
|
|
customer.Address.State);
|
|
}
|
|
}
|