mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 21:18:13 -05:00
Early return zero or negative amount invoices (#1595)
Stripe handles these by immediately finalizing as paid and crediting their account the appropriate amount.
This commit is contained in:
parent
dac3b3e893
commit
3d74f514ad
@ -747,6 +747,19 @@ namespace Bit.Core.Services
|
|||||||
|
|
||||||
var subResponse = await subscriptionService.UpdateAsync(sub.Id, subUpdateOptions);
|
var subResponse = await subscriptionService.UpdateAsync(sub.Id, subUpdateOptions);
|
||||||
|
|
||||||
|
var invoiceService = new InvoiceService();
|
||||||
|
var invoice = await invoiceService.GetAsync(subResponse?.LatestInvoiceId, new InvoiceGetOptions());
|
||||||
|
if (invoice == null)
|
||||||
|
{
|
||||||
|
throw new BadRequestException("Unable to locate draft invoice for subscription update.");
|
||||||
|
}
|
||||||
|
|
||||||
|
// If no amount due, invoice is autofinalized, we're done
|
||||||
|
if (invoice.AmountDue <= 0)
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
string paymentIntentClientSecret = null;
|
string paymentIntentClientSecret = null;
|
||||||
if (updatedItemOptions.Quantity > 0)
|
if (updatedItemOptions.Quantity > 0)
|
||||||
{
|
{
|
||||||
@ -755,12 +768,12 @@ namespace Bit.Core.Services
|
|||||||
if (chargeNow)
|
if (chargeNow)
|
||||||
{
|
{
|
||||||
paymentIntentClientSecret = await PayInvoiceAfterSubscriptionChangeAsync(
|
paymentIntentClientSecret = await PayInvoiceAfterSubscriptionChangeAsync(
|
||||||
storableSubscriber, subResponse?.LatestInvoiceId);
|
storableSubscriber, invoice);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
var invoiceService = new InvoiceService();
|
|
||||||
var invoice = await invoiceService.FinalizeInvoiceAsync(subResponse.LatestInvoiceId, new InvoiceFinalizeOptions
|
invoice = await invoiceService.FinalizeInvoiceAsync(subResponse.LatestInvoiceId, new InvoiceFinalizeOptions
|
||||||
{
|
{
|
||||||
AutoAdvance = false,
|
AutoAdvance = false,
|
||||||
});
|
});
|
||||||
@ -867,7 +880,7 @@ namespace Bit.Core.Services
|
|||||||
await customerService.DeleteAsync(subscriber.GatewayCustomerId);
|
await customerService.DeleteAsync(subscriber.GatewayCustomerId);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task<string> PayInvoiceAfterSubscriptionChangeAsync(ISubscriber subscriber, string invoiceId)
|
public async Task<string> PayInvoiceAfterSubscriptionChangeAsync(ISubscriber subscriber, Invoice invoice)
|
||||||
{
|
{
|
||||||
var customerService = new CustomerService();
|
var customerService = new CustomerService();
|
||||||
var customerOptions = new CustomerGetOptions();
|
var customerOptions = new CustomerGetOptions();
|
||||||
@ -884,16 +897,10 @@ namespace Bit.Core.Services
|
|||||||
var invoiceService = new InvoiceService();
|
var invoiceService = new InvoiceService();
|
||||||
string paymentIntentClientSecret = null;
|
string paymentIntentClientSecret = null;
|
||||||
|
|
||||||
var invoice = await invoiceService.GetAsync(invoiceId, new InvoiceGetOptions());
|
|
||||||
if (invoice == null)
|
|
||||||
{
|
|
||||||
throw new BadRequestException("Unable to locate draft invoice for subscription update.");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Invoice them and pay now instead of waiting until Stripe does this automatically.
|
// Invoice them and pay now instead of waiting until Stripe does this automatically.
|
||||||
|
|
||||||
string cardPaymentMethodId = null;
|
string cardPaymentMethodId = null;
|
||||||
if (invoice?.AmountDue > 0 && !customer.Metadata.ContainsKey("btCustomerId"))
|
if (!customer.Metadata.ContainsKey("btCustomerId"))
|
||||||
{
|
{
|
||||||
var hasDefaultCardPaymentMethod = customer.InvoiceSettings?.DefaultPaymentMethod?.Type == "card";
|
var hasDefaultCardPaymentMethod = customer.InvoiceSettings?.DefaultPaymentMethod?.Type == "card";
|
||||||
var hasDefaultValidSource = customer.DefaultSource != null &&
|
var hasDefaultValidSource = customer.DefaultSource != null &&
|
||||||
@ -934,48 +941,45 @@ namespace Bit.Core.Services
|
|||||||
{
|
{
|
||||||
PaymentMethod = cardPaymentMethodId,
|
PaymentMethod = cardPaymentMethodId,
|
||||||
};
|
};
|
||||||
if (invoice.AmountDue > 0)
|
if (customer?.Metadata?.ContainsKey("btCustomerId") ?? false)
|
||||||
{
|
{
|
||||||
if (customer?.Metadata?.ContainsKey("btCustomerId") ?? false)
|
invoicePayOptions.PaidOutOfBand = true;
|
||||||
{
|
var btInvoiceAmount = (invoice.AmountDue / 100M);
|
||||||
invoicePayOptions.PaidOutOfBand = true;
|
var transactionResult = await _btGateway.Transaction.SaleAsync(
|
||||||
var btInvoiceAmount = (invoice.AmountDue / 100M);
|
new Braintree.TransactionRequest
|
||||||
var transactionResult = await _btGateway.Transaction.SaleAsync(
|
{
|
||||||
new Braintree.TransactionRequest
|
Amount = btInvoiceAmount,
|
||||||
|
CustomerId = customer.Metadata["btCustomerId"],
|
||||||
|
Options = new Braintree.TransactionOptionsRequest
|
||||||
{
|
{
|
||||||
Amount = btInvoiceAmount,
|
SubmitForSettlement = true,
|
||||||
CustomerId = customer.Metadata["btCustomerId"],
|
PayPal = new Braintree.TransactionOptionsPayPalRequest
|
||||||
Options = new Braintree.TransactionOptionsRequest
|
|
||||||
{
|
{
|
||||||
SubmitForSettlement = true,
|
CustomField = $"{subscriber.BraintreeIdField()}:{subscriber.Id}"
|
||||||
PayPal = new Braintree.TransactionOptionsPayPalRequest
|
|
||||||
{
|
|
||||||
CustomField = $"{subscriber.BraintreeIdField()}:{subscriber.Id}"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
CustomFields = new Dictionary<string, string>
|
|
||||||
{
|
|
||||||
[subscriber.BraintreeIdField()] = subscriber.Id.ToString()
|
|
||||||
}
|
}
|
||||||
});
|
|
||||||
|
|
||||||
if (!transactionResult.IsSuccess())
|
|
||||||
{
|
|
||||||
throw new GatewayException("Failed to charge PayPal customer.");
|
|
||||||
}
|
|
||||||
|
|
||||||
braintreeTransaction = transactionResult.Target;
|
|
||||||
invoice = await invoiceService.UpdateAsync(invoice.Id, new InvoiceUpdateOptions
|
|
||||||
{
|
|
||||||
Metadata = new Dictionary<string, string>
|
|
||||||
{
|
|
||||||
["btTransactionId"] = braintreeTransaction.Id,
|
|
||||||
["btPayPalTransactionId"] =
|
|
||||||
braintreeTransaction.PayPalDetails.AuthorizationId
|
|
||||||
},
|
},
|
||||||
|
CustomFields = new Dictionary<string, string>
|
||||||
|
{
|
||||||
|
[subscriber.BraintreeIdField()] = subscriber.Id.ToString()
|
||||||
|
}
|
||||||
});
|
});
|
||||||
invoicePayOptions.PaidOutOfBand = true;
|
|
||||||
|
if (!transactionResult.IsSuccess())
|
||||||
|
{
|
||||||
|
throw new GatewayException("Failed to charge PayPal customer.");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
braintreeTransaction = transactionResult.Target;
|
||||||
|
invoice = await invoiceService.UpdateAsync(invoice.Id, new InvoiceUpdateOptions
|
||||||
|
{
|
||||||
|
Metadata = new Dictionary<string, string>
|
||||||
|
{
|
||||||
|
["btTransactionId"] = braintreeTransaction.Id,
|
||||||
|
["btPayPalTransactionId"] =
|
||||||
|
braintreeTransaction.PayPalDetails.AuthorizationId
|
||||||
|
},
|
||||||
|
});
|
||||||
|
invoicePayOptions.PaidOutOfBand = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
try
|
try
|
||||||
|
Loading…
x
Reference in New Issue
Block a user