mirror of
https://github.com/bitwarden/server.git
synced 2025-04-06 05:28:15 -05:00
add support for passing payment method type
This commit is contained in:
parent
de85dbc67c
commit
a9336a0e02
@ -463,7 +463,7 @@ namespace Bit.Api.Controllers
|
|||||||
throw new BadRequestException("Invalid license.");
|
throw new BadRequestException("Invalid license.");
|
||||||
}
|
}
|
||||||
|
|
||||||
await _userService.SignUpPremiumAsync(user, model.PaymentToken,
|
await _userService.SignUpPremiumAsync(user, model.PaymentToken, model.PaymentMethodType,
|
||||||
model.AdditionalStorageGb.GetValueOrDefault(0), license);
|
model.AdditionalStorageGb.GetValueOrDefault(0), license);
|
||||||
return new ProfileResponseModel(user, null, await _userService.TwoFactorIsEnabledAsync(user));
|
return new ProfileResponseModel(user, null, await _userService.TwoFactorIsEnabledAsync(user));
|
||||||
}
|
}
|
||||||
@ -518,7 +518,7 @@ namespace Bit.Api.Controllers
|
|||||||
throw new UnauthorizedAccessException();
|
throw new UnauthorizedAccessException();
|
||||||
}
|
}
|
||||||
|
|
||||||
await _userService.ReplacePaymentMethodAsync(user, model.PaymentToken);
|
await _userService.ReplacePaymentMethodAsync(user, model.PaymentToken, model.PaymentMethodType);
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpPost("storage")]
|
[HttpPost("storage")]
|
||||||
|
@ -209,7 +209,8 @@ namespace Bit.Api.Controllers
|
|||||||
throw new NotFoundException();
|
throw new NotFoundException();
|
||||||
}
|
}
|
||||||
|
|
||||||
await _organizationService.ReplacePaymentMethodAsync(orgIdGuid, model.PaymentToken);
|
await _organizationService.ReplacePaymentMethodAsync(orgIdGuid, model.PaymentToken,
|
||||||
|
model.PaymentMethodType);
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpPost("{id}/upgrade")]
|
[HttpPost("{id}/upgrade")]
|
||||||
|
@ -5,6 +5,6 @@
|
|||||||
Card = 0,
|
Card = 0,
|
||||||
BankAccount = 1,
|
BankAccount = 1,
|
||||||
PayPal = 2,
|
PayPal = 2,
|
||||||
Bitcoin = 3
|
BitPay = 3,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,11 +1,14 @@
|
|||||||
using Microsoft.AspNetCore.Http;
|
using Microsoft.AspNetCore.Http;
|
||||||
using System.ComponentModel.DataAnnotations;
|
using System.ComponentModel.DataAnnotations;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
using Bit.Core.Enums;
|
||||||
|
|
||||||
namespace Bit.Core.Models.Api
|
namespace Bit.Core.Models.Api
|
||||||
{
|
{
|
||||||
public class PremiumRequestModel : IValidatableObject
|
public class PremiumRequestModel : IValidatableObject
|
||||||
{
|
{
|
||||||
|
// TODO: Required in future
|
||||||
|
public PaymentMethodType? PaymentMethodType { get; set; }
|
||||||
public string PaymentToken { get; set; }
|
public string PaymentToken { get; set; }
|
||||||
[Range(0, 99)]
|
[Range(0, 99)]
|
||||||
public short? AdditionalStorageGb { get; set; }
|
public short? AdditionalStorageGb { get; set; }
|
||||||
|
@ -21,6 +21,8 @@ namespace Bit.Core.Models.Api
|
|||||||
public PlanType PlanType { get; set; }
|
public PlanType PlanType { get; set; }
|
||||||
[Required]
|
[Required]
|
||||||
public string Key { get; set; }
|
public string Key { get; set; }
|
||||||
|
// TODO: Required in future if not free plan
|
||||||
|
public PaymentMethodType? PaymentMethodType { get; set; }
|
||||||
public string PaymentToken { get; set; }
|
public string PaymentToken { get; set; }
|
||||||
[Range(0, double.MaxValue)]
|
[Range(0, double.MaxValue)]
|
||||||
public short AdditionalSeats { get; set; }
|
public short AdditionalSeats { get; set; }
|
||||||
@ -39,6 +41,7 @@ namespace Bit.Core.Models.Api
|
|||||||
OwnerKey = Key,
|
OwnerKey = Key,
|
||||||
Name = Name,
|
Name = Name,
|
||||||
Plan = PlanType,
|
Plan = PlanType,
|
||||||
|
PaymentMethodType = PaymentMethodType,
|
||||||
PaymentToken = PaymentToken,
|
PaymentToken = PaymentToken,
|
||||||
AdditionalSeats = AdditionalSeats,
|
AdditionalSeats = AdditionalSeats,
|
||||||
AdditionalStorageGb = AdditionalStorageGb.GetValueOrDefault(0),
|
AdditionalStorageGb = AdditionalStorageGb.GetValueOrDefault(0),
|
||||||
|
@ -1,9 +1,12 @@
|
|||||||
using System.ComponentModel.DataAnnotations;
|
using System.ComponentModel.DataAnnotations;
|
||||||
|
using Bit.Core.Enums;
|
||||||
|
|
||||||
namespace Bit.Core.Models.Api
|
namespace Bit.Core.Models.Api
|
||||||
{
|
{
|
||||||
public class PaymentRequestModel
|
public class PaymentRequestModel
|
||||||
{
|
{
|
||||||
|
// TODO: Required in future
|
||||||
|
public PaymentMethodType? PaymentMethodType { get; set; }
|
||||||
[Required]
|
[Required]
|
||||||
public string PaymentToken { get; set; }
|
public string PaymentToken { get; set; }
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
using Bit.Core.Models.Table;
|
using Bit.Core.Enums;
|
||||||
|
using Bit.Core.Models.Table;
|
||||||
|
|
||||||
namespace Bit.Core.Models.Business
|
namespace Bit.Core.Models.Business
|
||||||
{
|
{
|
||||||
@ -9,10 +10,11 @@ namespace Bit.Core.Models.Business
|
|||||||
public string BillingEmail { get; set; }
|
public string BillingEmail { get; set; }
|
||||||
public User Owner { get; set; }
|
public User Owner { get; set; }
|
||||||
public string OwnerKey { get; set; }
|
public string OwnerKey { get; set; }
|
||||||
public Enums.PlanType Plan { get; set; }
|
public PlanType Plan { get; set; }
|
||||||
public short AdditionalSeats { get; set; }
|
public short AdditionalSeats { get; set; }
|
||||||
public short AdditionalStorageGb { get; set; }
|
public short AdditionalStorageGb { get; set; }
|
||||||
public bool PremiumAccessAddon { get; set; }
|
public bool PremiumAccessAddon { get; set; }
|
||||||
|
public PaymentMethodType? PaymentMethodType { get; set; }
|
||||||
public string PaymentToken { get; set; }
|
public string PaymentToken { get; set; }
|
||||||
public string CollectionName { get; set; }
|
public string CollectionName { get; set; }
|
||||||
}
|
}
|
||||||
|
@ -10,7 +10,7 @@ namespace Bit.Core.Services
|
|||||||
{
|
{
|
||||||
public interface IOrganizationService
|
public interface IOrganizationService
|
||||||
{
|
{
|
||||||
Task ReplacePaymentMethodAsync(Guid organizationId, string paymentToken);
|
Task ReplacePaymentMethodAsync(Guid organizationId, string paymentToken, PaymentMethodType? paymentMethodType);
|
||||||
Task CancelSubscriptionAsync(Guid organizationId, bool? endOfPeriod = null);
|
Task CancelSubscriptionAsync(Guid organizationId, bool? endOfPeriod = null);
|
||||||
Task ReinstateSubscriptionAsync(Guid organizationId);
|
Task ReinstateSubscriptionAsync(Guid organizationId);
|
||||||
Task UpgradePlanAsync(Guid organizationId, PlanType plan, int additionalSeats);
|
Task UpgradePlanAsync(Guid organizationId, PlanType plan, int additionalSeats);
|
||||||
|
@ -43,10 +43,11 @@ namespace Bit.Core.Services
|
|||||||
Task<IdentityResult> DeleteAsync(User user);
|
Task<IdentityResult> DeleteAsync(User user);
|
||||||
Task<IdentityResult> DeleteAsync(User user, string token);
|
Task<IdentityResult> DeleteAsync(User user, string token);
|
||||||
Task SendDeleteConfirmationAsync(string email);
|
Task SendDeleteConfirmationAsync(string email);
|
||||||
Task SignUpPremiumAsync(User user, string paymentToken, short additionalStorageGb, UserLicense license);
|
Task SignUpPremiumAsync(User user, string paymentToken, PaymentMethodType? paymentMethodType,
|
||||||
|
short additionalStorageGb, UserLicense license);
|
||||||
Task UpdateLicenseAsync(User user, UserLicense license);
|
Task UpdateLicenseAsync(User user, UserLicense license);
|
||||||
Task AdjustStorageAsync(User user, short storageAdjustmentGb);
|
Task AdjustStorageAsync(User user, short storageAdjustmentGb);
|
||||||
Task ReplacePaymentMethodAsync(User user, string paymentToken);
|
Task ReplacePaymentMethodAsync(User user, string paymentToken, PaymentMethodType? paymentMethodType);
|
||||||
Task CancelPremiumAsync(User user, bool? endOfPeriod = null);
|
Task CancelPremiumAsync(User user, bool? endOfPeriod = null);
|
||||||
Task ReinstatePremiumAsync(User user);
|
Task ReinstatePremiumAsync(User user);
|
||||||
Task DisablePremiumAsync(Guid userId, DateTime? expirationDate);
|
Task DisablePremiumAsync(Guid userId, DateTime? expirationDate);
|
||||||
|
@ -71,7 +71,8 @@ namespace Bit.Core.Services
|
|||||||
_globalSettings = globalSettings;
|
_globalSettings = globalSettings;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task ReplacePaymentMethodAsync(Guid organizationId, string paymentToken)
|
public async Task ReplacePaymentMethodAsync(Guid organizationId, string paymentToken,
|
||||||
|
PaymentMethodType? paymentMethodType)
|
||||||
{
|
{
|
||||||
var organization = await GetOrgById(organizationId);
|
var organization = await GetOrgById(organizationId);
|
||||||
if(organization == null)
|
if(organization == null)
|
||||||
@ -79,22 +80,24 @@ namespace Bit.Core.Services
|
|||||||
throw new NotFoundException();
|
throw new NotFoundException();
|
||||||
}
|
}
|
||||||
|
|
||||||
PaymentMethodType paymentMethodType;
|
if(!paymentMethodType.HasValue)
|
||||||
if(paymentToken.StartsWith("btok_"))
|
|
||||||
{
|
{
|
||||||
paymentMethodType = PaymentMethodType.BankAccount;
|
if(paymentToken.StartsWith("tok_"))
|
||||||
}
|
{
|
||||||
else if(paymentToken.StartsWith("tok_"))
|
paymentMethodType = PaymentMethodType.Card;
|
||||||
{
|
}
|
||||||
paymentMethodType = PaymentMethodType.Card;
|
else if(paymentToken.StartsWith("btok_"))
|
||||||
}
|
{
|
||||||
else
|
paymentMethodType = PaymentMethodType.BankAccount;
|
||||||
{
|
}
|
||||||
paymentMethodType = PaymentMethodType.PayPal;
|
else
|
||||||
|
{
|
||||||
|
paymentMethodType = PaymentMethodType.PayPal;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var updated = await _paymentService.UpdatePaymentMethodAsync(organization,
|
var updated = await _paymentService.UpdatePaymentMethodAsync(organization,
|
||||||
paymentMethodType, paymentToken);
|
paymentMethodType.Value, paymentToken);
|
||||||
if(updated)
|
if(updated)
|
||||||
{
|
{
|
||||||
await ReplaceAndUpdateCache(organization);
|
await ReplaceAndUpdateCache(organization);
|
||||||
@ -547,21 +550,23 @@ namespace Bit.Core.Services
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
PaymentMethodType paymentMethodType;
|
if(!signup.PaymentMethodType.HasValue)
|
||||||
if(signup.PaymentToken.StartsWith("btok_"))
|
|
||||||
{
|
{
|
||||||
paymentMethodType = PaymentMethodType.BankAccount;
|
if(signup.PaymentToken.StartsWith("btok_"))
|
||||||
}
|
{
|
||||||
else if(signup.PaymentToken.StartsWith("tok_"))
|
signup.PaymentMethodType = PaymentMethodType.BankAccount;
|
||||||
{
|
}
|
||||||
paymentMethodType = PaymentMethodType.Card;
|
else if(signup.PaymentToken.StartsWith("tok_"))
|
||||||
}
|
{
|
||||||
else
|
signup.PaymentMethodType = PaymentMethodType.Card;
|
||||||
{
|
}
|
||||||
paymentMethodType = PaymentMethodType.PayPal;
|
else
|
||||||
|
{
|
||||||
|
signup.PaymentMethodType = PaymentMethodType.PayPal;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
await _paymentService.PurchaseOrganizationAsync(organization, paymentMethodType,
|
await _paymentService.PurchaseOrganizationAsync(organization, signup.PaymentMethodType.Value,
|
||||||
signup.PaymentToken, plan, signup.AdditionalStorageGb, signup.AdditionalSeats,
|
signup.PaymentToken, plan, signup.AdditionalStorageGb, signup.AdditionalSeats,
|
||||||
signup.PremiumAccessAddon);
|
signup.PremiumAccessAddon);
|
||||||
}
|
}
|
||||||
@ -1205,7 +1210,7 @@ namespace Bit.Core.Services
|
|||||||
{
|
{
|
||||||
throw new BadRequestException("Invalid installation id");
|
throw new BadRequestException("Invalid installation id");
|
||||||
}
|
}
|
||||||
|
|
||||||
var subInfo = await _paymentService.GetSubscriptionAsync(organization);
|
var subInfo = await _paymentService.GetSubscriptionAsync(organization);
|
||||||
return new OrganizationLicense(organization, subInfo, installationId, _licensingService);
|
return new OrganizationLicense(organization, subInfo, installationId, _licensingService);
|
||||||
}
|
}
|
||||||
|
@ -171,6 +171,11 @@ namespace Bit.Core.Services
|
|||||||
var stripePaymentMethod = paymentMethodType == PaymentMethodType.Card ||
|
var stripePaymentMethod = paymentMethodType == PaymentMethodType.Card ||
|
||||||
paymentMethodType == PaymentMethodType.BankAccount;
|
paymentMethodType == PaymentMethodType.BankAccount;
|
||||||
|
|
||||||
|
if(paymentMethodType == PaymentMethodType.BankAccount)
|
||||||
|
{
|
||||||
|
throw new GatewayException("Bank account payment method is not supported at this time.");
|
||||||
|
}
|
||||||
|
|
||||||
if(user.Gateway == GatewayType.Stripe && !string.IsNullOrWhiteSpace(user.GatewayCustomerId))
|
if(user.Gateway == GatewayType.Stripe && !string.IsNullOrWhiteSpace(user.GatewayCustomerId))
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
|
@ -678,7 +678,8 @@ namespace Bit.Core.Services
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task SignUpPremiumAsync(User user, string paymentToken, short additionalStorageGb, UserLicense license)
|
public async Task SignUpPremiumAsync(User user, string paymentToken, PaymentMethodType? paymentMethodType,
|
||||||
|
short additionalStorageGb, UserLicense license)
|
||||||
{
|
{
|
||||||
if(user.Premium)
|
if(user.Premium)
|
||||||
{
|
{
|
||||||
@ -709,18 +710,23 @@ namespace Bit.Core.Services
|
|||||||
}
|
}
|
||||||
else if(!string.IsNullOrWhiteSpace(paymentToken))
|
else if(!string.IsNullOrWhiteSpace(paymentToken))
|
||||||
{
|
{
|
||||||
if(paymentToken.StartsWith("btok_"))
|
if(!paymentMethodType.HasValue)
|
||||||
{
|
{
|
||||||
throw new BadRequestException("Invalid token.");
|
if(paymentToken.StartsWith("tok_"))
|
||||||
|
{
|
||||||
|
paymentMethodType = PaymentMethodType.Card;
|
||||||
|
}
|
||||||
|
else if(paymentToken.StartsWith("btok_"))
|
||||||
|
{
|
||||||
|
paymentMethodType = PaymentMethodType.BankAccount;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
paymentMethodType = PaymentMethodType.PayPal;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var paymentMethodType = PaymentMethodType.Card;
|
await _paymentService.PurchasePremiumAsync(user, paymentMethodType.Value,
|
||||||
if(!paymentToken.StartsWith("tok_"))
|
|
||||||
{
|
|
||||||
paymentMethodType = PaymentMethodType.PayPal;
|
|
||||||
}
|
|
||||||
|
|
||||||
await _paymentService.PurchasePremiumAsync(user, paymentMethodType,
|
|
||||||
paymentToken, additionalStorageGb);
|
paymentToken, additionalStorageGb);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -795,29 +801,36 @@ namespace Bit.Core.Services
|
|||||||
{
|
{
|
||||||
throw new BadRequestException("Not a premium user.");
|
throw new BadRequestException("Not a premium user.");
|
||||||
}
|
}
|
||||||
|
|
||||||
await BillingHelpers.AdjustStorageAsync(_paymentService, user, storageAdjustmentGb, StoragePlanId);
|
await BillingHelpers.AdjustStorageAsync(_paymentService, user, storageAdjustmentGb, StoragePlanId);
|
||||||
await SaveUserAsync(user);
|
await SaveUserAsync(user);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task ReplacePaymentMethodAsync(User user, string paymentToken)
|
public async Task ReplacePaymentMethodAsync(User user, string paymentToken,
|
||||||
|
PaymentMethodType? paymentMethodType)
|
||||||
{
|
{
|
||||||
if(paymentToken.StartsWith("btok_"))
|
if(paymentToken.StartsWith("btok_"))
|
||||||
{
|
{
|
||||||
throw new BadRequestException("Invalid token.");
|
throw new BadRequestException("Invalid token.");
|
||||||
}
|
}
|
||||||
|
|
||||||
PaymentMethodType paymentMethodType;
|
if(!paymentMethodType.HasValue)
|
||||||
if(paymentToken.StartsWith("tok_"))
|
|
||||||
{
|
{
|
||||||
paymentMethodType = PaymentMethodType.Card;
|
if(paymentToken.StartsWith("tok_"))
|
||||||
}
|
{
|
||||||
else
|
paymentMethodType = PaymentMethodType.Card;
|
||||||
{
|
}
|
||||||
paymentMethodType = PaymentMethodType.PayPal;
|
else if(paymentToken.StartsWith("btok_"))
|
||||||
|
{
|
||||||
|
paymentMethodType = PaymentMethodType.BankAccount;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
paymentMethodType = PaymentMethodType.PayPal;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var updated = await _paymentService.UpdatePaymentMethodAsync(user, paymentMethodType, paymentToken);
|
var updated = await _paymentService.UpdatePaymentMethodAsync(user, paymentMethodType.Value, paymentToken);
|
||||||
if(updated)
|
if(updated)
|
||||||
{
|
{
|
||||||
await SaveUserAsync(user);
|
await SaveUserAsync(user);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user