1
0
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:
Kyle Spearrin 2019-02-19 17:13:21 -05:00
parent de85dbc67c
commit a9336a0e02
12 changed files with 91 additions and 55 deletions

View File

@ -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")]

View File

@ -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")]

View File

@ -5,6 +5,6 @@
Card = 0, Card = 0,
BankAccount = 1, BankAccount = 1,
PayPal = 2, PayPal = 2,
Bitcoin = 3 BitPay = 3,
} }
} }

View File

@ -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; }

View File

@ -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),

View File

@ -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; }
} }

View File

@ -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; }
} }

View File

@ -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);

View File

@ -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);

View File

@ -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);
} }

View File

@ -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

View File

@ -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);