diff --git a/src/Admin/Controllers/ProvidersController.cs b/src/Admin/Controllers/ProvidersController.cs new file mode 100644 index 0000000000..b97f50a9f1 --- /dev/null +++ b/src/Admin/Controllers/ProvidersController.cs @@ -0,0 +1,118 @@ +using System; +using System.Collections.Generic; +using System.Threading.Tasks; +using Bit.Admin.Models; +using Bit.Core.Models.Table.Provider; +using Bit.Core.Repositories; +using Bit.Core.Services; +using Bit.Core.Settings; +using Bit.Core.Utilities; +using Microsoft.AspNetCore.Authorization; +using Microsoft.AspNetCore.Mvc; + +namespace Bit.Admin.Controllers +{ + [Authorize] + [SelfHosted(NotSelfHostedOnly = true)] + public class ProvidersController : Controller + { + private readonly IProviderRepository _providerRepository; + private readonly IProviderUserRepository _providerUserRepository; + private readonly GlobalSettings _globalSettings; + private readonly IProviderService _providerService; + + public ProvidersController(IProviderRepository providerRepository, IProviderUserRepository providerUserRepository, + IProviderService providerService, GlobalSettings globalSettings) + { + _providerRepository = providerRepository; + _providerUserRepository = providerUserRepository; + _providerService = providerService; + _globalSettings = globalSettings; + } + + public async Task Index(string name = null, string userEmail = null, int page = 1, int count = 25) + { + if (page < 1) + { + page = 1; + } + + if (count < 1) + { + count = 1; + } + + var skip = (page - 1) * count; + var providers = await _providerRepository.SearchAsync(name, userEmail, skip, count); + return View(new ProvidersModel + { + Items = providers as List, + Name = string.IsNullOrWhiteSpace(name) ? null : name, + UserEmail = string.IsNullOrWhiteSpace(userEmail) ? null : userEmail, + Page = page, + Count = count, + Action = _globalSettings.SelfHosted ? "View" : "Edit", + SelfHosted = _globalSettings.SelfHosted + }); + } + + public IActionResult Create(string ownerEmail = null) + { + return View(new CreateProviderModel + { + OwnerEmail = ownerEmail + }); + } + + [HttpPost] + public async Task Create(CreateProviderModel model) + { + if (!ModelState.IsValid) + { + return View(model); + } + + await _providerService.CreateAsync(model.OwnerEmail); + + return RedirectToAction("Index"); + } + + public async Task View(Guid id) + { + var provider = await _providerRepository.GetByIdAsync(id); + if (provider == null) + { + return RedirectToAction("Index"); + } + + var users = await _providerUserRepository.GetManyByProviderAsync(id); + return View(new ProviderViewModel(provider, users)); + } + + [SelfHosted(NotSelfHostedOnly = true)] + public async Task Edit(Guid id) + { + var provider = await _providerRepository.GetByIdAsync(id); + if (provider == null) + { + return RedirectToAction("Index"); + } + + var users = await _providerUserRepository.GetManyByProviderAsync(id); + return View(new ProviderEditModel(provider, users)); + } + + [HttpPost] + [ValidateAntiForgeryToken] + public async Task Delete(Guid id) + { + var provider = await _providerRepository.GetByIdAsync(id); + if (provider != null) + { + await _providerRepository.DeleteAsync(provider); + } + + return RedirectToAction("Index"); + } + } +} diff --git a/src/Admin/Models/CreateProviderModel.cs b/src/Admin/Models/CreateProviderModel.cs new file mode 100644 index 0000000000..ad4c547f42 --- /dev/null +++ b/src/Admin/Models/CreateProviderModel.cs @@ -0,0 +1,13 @@ +using System.ComponentModel.DataAnnotations; + +namespace Bit.Admin.Models +{ + public class CreateProviderModel + { + public CreateProviderModel() { } + + [Display(Name = "Owner Email")] + [Required] + public string OwnerEmail { get; set; } + } +} diff --git a/src/Admin/Models/ProviderEditModel.cs b/src/Admin/Models/ProviderEditModel.cs new file mode 100644 index 0000000000..652234e845 --- /dev/null +++ b/src/Admin/Models/ProviderEditModel.cs @@ -0,0 +1,29 @@ +using System.Collections.Generic; +using System.Linq; +using Bit.Core.Enums.Provider; +using Bit.Core.Models.Table.Provider; + +namespace Bit.Admin.Models +{ + public class ProviderEditModel : ProviderViewModel + { + public ProviderEditModel(Provider provider, IEnumerable providerUsers) + : base(provider, providerUsers) + { + Name = provider.Name; + BusinessName = provider.BusinessName; + BillingEmail = provider.BillingEmail; + Enabled = provider.Enabled; + } + + public string Administrators { get; set; } + + public bool Enabled { get; set; } + + public string BillingEmail { get; set; } + + public string BusinessName { get; set; } + + public string Name { get; set; } + } +} diff --git a/src/Admin/Models/ProviderViewModel.cs b/src/Admin/Models/ProviderViewModel.cs new file mode 100644 index 0000000000..9f77e41971 --- /dev/null +++ b/src/Admin/Models/ProviderViewModel.cs @@ -0,0 +1,27 @@ +using System.Collections.Generic; +using System.Linq; +using Bit.Core.Enums.Provider; +using Bit.Core.Models.Table.Provider; + +namespace Bit.Admin.Models +{ + public class ProviderViewModel + { + public ProviderViewModel(Provider provider, IEnumerable providerUsers) + { + Provider = provider; + UserCount = providerUsers.Count(); + + ProviderAdmins = string.Join(", ", + providerUsers + .Where(u => u.Type == ProviderUserType.ProviderAdmin && u.Status == ProviderUserStatusType.Confirmed) + .Select(u => u.Email)); + } + + public int UserCount { get; set; } + + public Provider Provider { get; set; } + + public string ProviderAdmins { get; set; } + } +} diff --git a/src/Admin/Models/ProvidersModel.cs b/src/Admin/Models/ProvidersModel.cs new file mode 100644 index 0000000000..dc2d1e87be --- /dev/null +++ b/src/Admin/Models/ProvidersModel.cs @@ -0,0 +1,14 @@ +using Bit.Core.Models.Table; +using Bit.Core.Models.Table.Provider; + +namespace Bit.Admin.Models +{ + public class ProvidersModel : PagedModel + { + public string Name { get; set; } + public string UserEmail { get; set; } + public bool? Paid { get; set; } + public string Action { get; set; } + public bool SelfHosted { get; set; } + } +} diff --git a/src/Admin/Views/Providers/Create.cshtml b/src/Admin/Views/Providers/Create.cshtml new file mode 100644 index 0000000000..0369ca0179 --- /dev/null +++ b/src/Admin/Views/Providers/Create.cshtml @@ -0,0 +1,17 @@ +@model CreateProviderModel +@{ + ViewData["Title"] = "Create Provider"; +} + +

Create Provider

+ +
+
+ +
+ + +
+ + +
diff --git a/src/Admin/Views/Providers/Edit.cshtml b/src/Admin/Views/Providers/Edit.cshtml new file mode 100644 index 0000000000..2e75614aaa --- /dev/null +++ b/src/Admin/Views/Providers/Edit.cshtml @@ -0,0 +1,51 @@ +@model ProviderEditModel +@{ + ViewData["Title"] = "Provider: " + Model.Provider.Name; +} + +

Provider @Model.Provider.Name

+ +

Provider Information

+@await Html.PartialAsync("_ViewInformation", Model) +
+

General

+
+
+
+ + +
+
+
+
+ + +
+

Business Information

+
+
+
+ + +
+
+
+

Billing

+
+
+
+ + +
+
+
+
+
+ +
+
+ +
+
+
diff --git a/src/Admin/Views/Providers/Index.cshtml b/src/Admin/Views/Providers/Index.cshtml new file mode 100644 index 0000000000..b1d9a7a6b2 --- /dev/null +++ b/src/Admin/Views/Providers/Index.cshtml @@ -0,0 +1,91 @@ +@model ProvidersModel +@{ + ViewData["Title"] = "Providers"; +} + +

Providers

+ +
+
+
+ + + + + +
+
+ +
+ +
+ + + + + + + + + + @if(!Model.Items.Any()) + { + + + + } + else + { + @foreach(var provider in Model.Items) + { + + + + + + } + } + +
NameStatusCreated
No results to list.
+ @(provider.Name ?? "Pending") + @provider.Status + + @provider.CreationDate.ToShortDateString() + +
+
+ + diff --git a/src/Admin/Views/Providers/View.cshtml b/src/Admin/Views/Providers/View.cshtml new file mode 100644 index 0000000000..14bb9ed6cd --- /dev/null +++ b/src/Admin/Views/Providers/View.cshtml @@ -0,0 +1,13 @@ +@model ProviderViewModel +@{ + ViewData["Title"] = "Provider: " + Model.Provider.Name; +} + +

Provider @Model.Provider.Name

+ +

Information

+@await Html.PartialAsync("_ViewInformation", Model) +
+ +
diff --git a/src/Admin/Views/Providers/_ViewInformation.cshtml b/src/Admin/Views/Providers/_ViewInformation.cshtml new file mode 100644 index 0000000000..89e5f5aced --- /dev/null +++ b/src/Admin/Views/Providers/_ViewInformation.cshtml @@ -0,0 +1,20 @@ +@model ProviderViewModel +
+
Id
+
@Model.Provider.Id
+ +
Status
+
@Model.Provider.Status
+ +
Users
+
@Model.UserCount
+ +
ProviderAdmins
+
@(string.IsNullOrWhiteSpace(Model.ProviderAdmins) ? "None" : Model.ProviderAdmins)
+ +
Created
+
@Model.Provider.CreationDate.ToString()
+ +
Modified
+
@Model.Provider.RevisionDate.ToString()
+
diff --git a/src/Admin/Views/Shared/_Layout.cshtml b/src/Admin/Views/Shared/_Layout.cshtml index 635ab4bc43..ef06018b4c 100644 --- a/src/Admin/Views/Shared/_Layout.cshtml +++ b/src/Admin/Views/Shared/_Layout.cshtml @@ -40,6 +40,9 @@ @if(!GlobalSettings.SelfHosted) { +