diff --git a/src/Admin/Controllers/ToolsController.cs b/src/Admin/Controllers/ToolsController.cs new file mode 100644 index 0000000000..eaf7254903 --- /dev/null +++ b/src/Admin/Controllers/ToolsController.cs @@ -0,0 +1,79 @@ +using System; +using System.Collections.Generic; +using System.Threading.Tasks; +using Bit.Admin.Models; +using Bit.Core; +using Bit.Core.Utilities; +using Microsoft.AspNetCore.Authorization; +using Microsoft.AspNetCore.Mvc; + +namespace Bit.Admin.Controllers +{ + [Authorize] + [SelfHosted(NotSelfHostedOnly = true)] + public class ToolsController : Controller + { + private readonly GlobalSettings _globalSettings; + + public ToolsController(GlobalSettings globalSettings) + { + _globalSettings = globalSettings; + } + + public async Task ChargeBraintree() + { + return View(new ChargeBraintreeModel()); + } + + [HttpPost] + public async Task ChargeBraintree(ChargeBraintreeModel model) + { + if(!ModelState.IsValid) + { + return View(model); + } + + var btGateway = new Braintree.BraintreeGateway + { + Environment = _globalSettings.Braintree.Production ? + Braintree.Environment.PRODUCTION : Braintree.Environment.SANDBOX, + MerchantId = _globalSettings.Braintree.MerchantId, + PublicKey = _globalSettings.Braintree.PublicKey, + PrivateKey = _globalSettings.Braintree.PrivateKey + }; + + var btObjIdField = model.Id[0] == 'o' ? "organization_id" : "user_id"; + var btObjId = new Guid(model.Id.Substring(1, 32)); + + var transactionResult = await btGateway.Transaction.SaleAsync( + new Braintree.TransactionRequest + { + Amount = model.Amount.Value, + CustomerId = model.Id, + Options = new Braintree.TransactionOptionsRequest + { + SubmitForSettlement = true, + PayPal = new Braintree.TransactionOptionsPayPalRequest + { + CustomField = $"{btObjIdField}:{btObjId}" + } + }, + CustomFields = new Dictionary + { + [btObjIdField] = btObjId.ToString() + } + }); + + if(!transactionResult.IsSuccess()) + { + ModelState.AddModelError(string.Empty, "Charge failed."); + } + else + { + model.TransactionId = transactionResult.Target.Id; + model.PayPalTransactionId = transactionResult.Target?.PayPalDetails?.CaptureId; + } + return View(model); + } + } +} diff --git a/src/Admin/Models/ChargeBraintreeModel.cs b/src/Admin/Models/ChargeBraintreeModel.cs new file mode 100644 index 0000000000..a9d45a9ac2 --- /dev/null +++ b/src/Admin/Models/ChargeBraintreeModel.cs @@ -0,0 +1,30 @@ +using System; +using System.Collections.Generic; +using System.ComponentModel.DataAnnotations; + +namespace Bit.Admin.Models +{ + public class ChargeBraintreeModel : IValidatableObject + { + [Required] + [Display(Name = "Braintree Customer Id")] + public string Id { get; set; } + [Required] + [Display(Name = "Amount")] + public decimal? Amount { get; set; } + public string TransactionId { get; set; } + public string PayPalTransactionId { get; set; } + + public IEnumerable Validate(ValidationContext validationContext) + { + if(Id != null) + { + if(Id.Length != 36 || (Id[0] != 'o' && Id[0] != 'u') || + !Guid.TryParse(Id.Substring(1, 32), out var guid)) + { + yield return new ValidationResult("Customer Id is not a valid format."); + } + } + } + } +} diff --git a/src/Admin/Sass/site.scss b/src/Admin/Sass/site.scss index 70899fb282..38676f337d 100644 --- a/src/Admin/Sass/site.scss +++ b/src/Admin/Sass/site.scss @@ -28,3 +28,7 @@ h2 { h3 { text-transform: uppercase; } + +.validation-summary-valid { + display: none; +} diff --git a/src/Admin/Views/Shared/_Layout.cshtml b/src/Admin/Views/Shared/_Layout.cshtml index e8320bb0b6..0344b52767 100644 --- a/src/Admin/Views/Shared/_Layout.cshtml +++ b/src/Admin/Views/Shared/_Layout.cshtml @@ -40,6 +40,17 @@ @if(!GlobalSettings.SelfHosted) { + diff --git a/src/Admin/Views/Tools/ChargeBraintree.cshtml b/src/Admin/Views/Tools/ChargeBraintree.cshtml new file mode 100644 index 0000000000..d9081ac1fa --- /dev/null +++ b/src/Admin/Views/Tools/ChargeBraintree.cshtml @@ -0,0 +1,44 @@ +@model ChargeBraintreeModel +@{ + ViewData["Title"] = "Charge Braintree Customer"; +} + +

Charge Braintree Customer

+ +@if(!string.IsNullOrWhiteSpace(Model.TransactionId)) +{ + + Charge Another Customer +} +else +{ +
+
+
+
+
+ + +
+
+
+
+ +
+
+ $ +
+ +
+
+
+
+ +
+}