mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 05:00:19 -05:00
[Admin] Add BusinessUnitConverterController
This commit is contained in:
parent
f478c71019
commit
582cd7eb44
@ -14,9 +14,6 @@
|
|||||||
<ProjectReference Include="..\Core\Core.csproj" />
|
<ProjectReference Include="..\Core\Core.csproj" />
|
||||||
<ProjectReference Include="..\..\util\SqliteMigrations\SqliteMigrations.csproj" />
|
<ProjectReference Include="..\..\util\SqliteMigrations\SqliteMigrations.csproj" />
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
|
||||||
<Folder Include="Billing\Controllers\" />
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
<Choose>
|
<Choose>
|
||||||
<When Condition="!$(DefineConstants.Contains('OSS'))">
|
<When Condition="!$(DefineConstants.Contains('OSS'))">
|
||||||
|
@ -0,0 +1,183 @@
|
|||||||
|
#nullable enable
|
||||||
|
using Bit.Admin.Billing.Models;
|
||||||
|
using Bit.Admin.Enums;
|
||||||
|
using Bit.Admin.Utilities;
|
||||||
|
using Bit.Core.AdminConsole.Entities;
|
||||||
|
using Bit.Core.AdminConsole.Entities.Provider;
|
||||||
|
using Bit.Core.AdminConsole.Enums.Provider;
|
||||||
|
using Bit.Core.AdminConsole.Repositories;
|
||||||
|
using Bit.Core.Billing.Services;
|
||||||
|
using Bit.Core.Exceptions;
|
||||||
|
using Bit.Core.Repositories;
|
||||||
|
using Bit.Core.Utilities;
|
||||||
|
using Microsoft.AspNetCore.Authorization;
|
||||||
|
using Microsoft.AspNetCore.Mvc;
|
||||||
|
|
||||||
|
namespace Bit.Admin.Billing.Controllers;
|
||||||
|
|
||||||
|
[Authorize]
|
||||||
|
[Route("organizations/billing/{organizationId:guid}/business-unit")]
|
||||||
|
public class BusinessUnitConversionController(
|
||||||
|
IBusinessUnitConverter businessUnitConverter,
|
||||||
|
IOrganizationRepository organizationRepository,
|
||||||
|
IProviderRepository providerRepository,
|
||||||
|
IProviderUserRepository providerUserRepository) : Controller
|
||||||
|
{
|
||||||
|
[HttpGet]
|
||||||
|
[RequirePermission(Permission.Org_Billing_ConvertToBusinessUnit)]
|
||||||
|
[SelfHosted(NotSelfHostedOnly = true)]
|
||||||
|
public async Task<IActionResult> IndexAsync([FromRoute] Guid organizationId)
|
||||||
|
{
|
||||||
|
var organization = await organizationRepository.GetByIdAsync(organizationId);
|
||||||
|
|
||||||
|
if (organization == null)
|
||||||
|
{
|
||||||
|
throw new NotFoundException();
|
||||||
|
}
|
||||||
|
|
||||||
|
var model = new BusinessUnitConversionModel { Organization = organization };
|
||||||
|
|
||||||
|
var invitedProviderAdmin = await GetInvitedProviderAdminAsync(organization);
|
||||||
|
|
||||||
|
if (invitedProviderAdmin != null)
|
||||||
|
{
|
||||||
|
model.ProviderAdminEmail = invitedProviderAdmin.Email;
|
||||||
|
model.ProviderId = invitedProviderAdmin.ProviderId;
|
||||||
|
}
|
||||||
|
|
||||||
|
var success = ReadSuccessMessage();
|
||||||
|
|
||||||
|
if (!string.IsNullOrEmpty(success))
|
||||||
|
{
|
||||||
|
model.Success = success;
|
||||||
|
}
|
||||||
|
|
||||||
|
var errors = ReadErrorMessages();
|
||||||
|
|
||||||
|
if (errors is { Count: > 0 })
|
||||||
|
{
|
||||||
|
model.Errors = errors;
|
||||||
|
}
|
||||||
|
|
||||||
|
return View(model);
|
||||||
|
}
|
||||||
|
|
||||||
|
[HttpPost]
|
||||||
|
[RequirePermission(Permission.Org_Billing_ConvertToBusinessUnit)]
|
||||||
|
[SelfHosted(NotSelfHostedOnly = true)]
|
||||||
|
public async Task<IActionResult> InitiateAsync(
|
||||||
|
[FromRoute] Guid organizationId,
|
||||||
|
BusinessUnitConversionModel model)
|
||||||
|
{
|
||||||
|
var organization = await organizationRepository.GetByIdAsync(organizationId);
|
||||||
|
|
||||||
|
if (organization == null)
|
||||||
|
{
|
||||||
|
throw new NotFoundException();
|
||||||
|
}
|
||||||
|
|
||||||
|
var result = await businessUnitConverter.InitiateConversion(
|
||||||
|
organization,
|
||||||
|
model.ProviderAdminEmail!);
|
||||||
|
|
||||||
|
return result.Match(
|
||||||
|
providerId => RedirectToAction("Edit", "Providers", new { id = providerId }),
|
||||||
|
errors =>
|
||||||
|
{
|
||||||
|
PersistErrorMessages(errors);
|
||||||
|
return RedirectToAction("Index", new { organizationId });
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
[HttpPost("reset")]
|
||||||
|
[RequirePermission(Permission.Org_Billing_ConvertToBusinessUnit)]
|
||||||
|
[SelfHosted(NotSelfHostedOnly = true)]
|
||||||
|
public async Task<IActionResult> ResetAsync(
|
||||||
|
[FromRoute] Guid organizationId,
|
||||||
|
BusinessUnitConversionModel model)
|
||||||
|
{
|
||||||
|
var organization = await organizationRepository.GetByIdAsync(organizationId);
|
||||||
|
|
||||||
|
if (organization == null)
|
||||||
|
{
|
||||||
|
throw new NotFoundException();
|
||||||
|
}
|
||||||
|
|
||||||
|
await businessUnitConverter.ResetConversion(organization, model.ProviderAdminEmail!);
|
||||||
|
|
||||||
|
PersistSuccessMessage("Business unit conversion was successfully reset.");
|
||||||
|
|
||||||
|
return RedirectToAction("Index", new { organizationId });
|
||||||
|
}
|
||||||
|
|
||||||
|
[HttpPost("resend-invite")]
|
||||||
|
[RequirePermission(Permission.Org_Billing_ConvertToBusinessUnit)]
|
||||||
|
[SelfHosted(NotSelfHostedOnly = true)]
|
||||||
|
public async Task<IActionResult> ResendInviteAsync(
|
||||||
|
[FromRoute] Guid organizationId,
|
||||||
|
BusinessUnitConversionModel model)
|
||||||
|
{
|
||||||
|
var organization = await organizationRepository.GetByIdAsync(organizationId);
|
||||||
|
|
||||||
|
if (organization == null)
|
||||||
|
{
|
||||||
|
throw new NotFoundException();
|
||||||
|
}
|
||||||
|
|
||||||
|
await businessUnitConverter.ResendConversionInvite(organization, model.ProviderAdminEmail!);
|
||||||
|
|
||||||
|
PersistSuccessMessage($"Invite was successfully resent to {model.ProviderAdminEmail}.");
|
||||||
|
|
||||||
|
return RedirectToAction("Index", new { organizationId });
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task<ProviderUser?> GetInvitedProviderAdminAsync(
|
||||||
|
Organization organization)
|
||||||
|
{
|
||||||
|
var provider = await providerRepository.GetByOrganizationIdAsync(organization.Id);
|
||||||
|
|
||||||
|
if (provider is not
|
||||||
|
{
|
||||||
|
Type: ProviderType.BusinessUnit,
|
||||||
|
Status: ProviderStatusType.Pending
|
||||||
|
})
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var providerUsers =
|
||||||
|
await providerUserRepository.GetManyByProviderAsync(provider.Id, ProviderUserType.ProviderAdmin);
|
||||||
|
|
||||||
|
if (providerUsers.Count != 1)
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
var providerUser = providerUsers.First();
|
||||||
|
|
||||||
|
return providerUser is
|
||||||
|
{
|
||||||
|
Type: ProviderUserType.ProviderAdmin,
|
||||||
|
Status: ProviderUserStatusType.Invited,
|
||||||
|
UserId: not null
|
||||||
|
} ? providerUser : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private const string _errors = "errors";
|
||||||
|
private const string _success = "Success";
|
||||||
|
|
||||||
|
private void PersistSuccessMessage(string message) => TempData[_success] = message;
|
||||||
|
private void PersistErrorMessages(List<string> errors)
|
||||||
|
{
|
||||||
|
var input = string.Join("|", errors);
|
||||||
|
TempData[_errors] = input;
|
||||||
|
}
|
||||||
|
private string? ReadSuccessMessage() => ReadTempData<string>(_success);
|
||||||
|
private List<string>? ReadErrorMessages()
|
||||||
|
{
|
||||||
|
var output = ReadTempData<string>(_errors);
|
||||||
|
return string.IsNullOrEmpty(output) ? null : output.Split('|').ToList();
|
||||||
|
}
|
||||||
|
|
||||||
|
private T? ReadTempData<T>(string key) => TempData.TryGetValue(key, out var obj) && obj is T value ? value : default;
|
||||||
|
}
|
25
src/Admin/Billing/Models/BusinessUnitConversionModel.cs
Normal file
25
src/Admin/Billing/Models/BusinessUnitConversionModel.cs
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
#nullable enable
|
||||||
|
using System.ComponentModel.DataAnnotations;
|
||||||
|
using Bit.Core.AdminConsole.Entities;
|
||||||
|
using Microsoft.AspNetCore.Mvc.ModelBinding;
|
||||||
|
|
||||||
|
namespace Bit.Admin.Billing.Models;
|
||||||
|
|
||||||
|
public class BusinessUnitConversionModel
|
||||||
|
{
|
||||||
|
[Required]
|
||||||
|
[EmailAddress]
|
||||||
|
[Display(Name = "Provider Admin Email")]
|
||||||
|
public string? ProviderAdminEmail { get; set; }
|
||||||
|
|
||||||
|
[BindNever]
|
||||||
|
public required Organization Organization { get; set; }
|
||||||
|
|
||||||
|
[BindNever]
|
||||||
|
public Guid? ProviderId { get; set; }
|
||||||
|
|
||||||
|
[BindNever]
|
||||||
|
public string? Success { get; set; }
|
||||||
|
|
||||||
|
[BindNever] public List<string>? Errors { get; set; } = [];
|
||||||
|
}
|
75
src/Admin/Billing/Views/BusinessUnitConversion/Index.cshtml
Normal file
75
src/Admin/Billing/Views/BusinessUnitConversion/Index.cshtml
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
@model Bit.Admin.Billing.Models.BusinessUnitConversionModel
|
||||||
|
|
||||||
|
@{
|
||||||
|
ViewData["Title"] = "Convert Organization to Business Unit";
|
||||||
|
}
|
||||||
|
|
||||||
|
@if (!string.IsNullOrEmpty(Model.ProviderAdminEmail))
|
||||||
|
{
|
||||||
|
<h1>Convert @Model.Organization.Name to Business Unit</h1>
|
||||||
|
@if (!string.IsNullOrEmpty(Model.Success))
|
||||||
|
{
|
||||||
|
<div class="alert alert-success alert-dismissible fade show mb-3" role="alert">
|
||||||
|
@Model.Success
|
||||||
|
<button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
@if (Model.Errors?.Any() ?? false)
|
||||||
|
{
|
||||||
|
@foreach (var error in Model.Errors)
|
||||||
|
{
|
||||||
|
<div class="alert alert-danger alert-dismissible fade show mb-3" role="alert">
|
||||||
|
@error
|
||||||
|
<button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
<p>This organization has a business unit conversion in progress.</p>
|
||||||
|
|
||||||
|
<div class="mb-3">
|
||||||
|
<label asp-for="ProviderAdminEmail" class="form-label"></label>
|
||||||
|
<input type="email" class="form-control" asp-for="ProviderAdminEmail" disabled></input>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="d-flex gap-2">
|
||||||
|
<form method="post" asp-controller="BusinessUnitConversion" asp-action="ResendInvite" asp-route-organizationId="@Model.Organization.Id">
|
||||||
|
<input type="hidden" asp-for="ProviderAdminEmail" />
|
||||||
|
<button type="submit" class="btn btn-primary mb-2">Resend Invite</button>
|
||||||
|
</form>
|
||||||
|
<form method="post" asp-controller="BusinessUnitConversion" asp-action="Reset" asp-route-organizationId="@Model.Organization.Id">
|
||||||
|
<input type="hidden" asp-for="ProviderAdminEmail" />
|
||||||
|
<button type="submit" class="btn btn-danger mb-2">Reset Conversion</button>
|
||||||
|
</form>
|
||||||
|
@if (Model.ProviderId.HasValue)
|
||||||
|
{
|
||||||
|
<a asp-controller="Providers"
|
||||||
|
asp-action="Edit"
|
||||||
|
asp-route-id="@Model.ProviderId"
|
||||||
|
class="btn btn-secondary mb-2">
|
||||||
|
Go to Provider
|
||||||
|
</a>
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<h1>Convert @Model.Organization.Name to Business Unit</h1>
|
||||||
|
@if (Model.Errors?.Any() ?? false)
|
||||||
|
{
|
||||||
|
@foreach (var error in Model.Errors)
|
||||||
|
{
|
||||||
|
<div class="alert alert-danger alert-dismissible fade show mb-3" role="alert">
|
||||||
|
@error
|
||||||
|
<button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
}
|
||||||
|
<form method="post" asp-controller="BusinessUnitConversion" asp-action="Initiate" asp-route-organizationId="@Model.Organization.Id">
|
||||||
|
<div asp-validation-summary="All" class="alert alert-danger"></div>
|
||||||
|
<div class="mb-3">
|
||||||
|
<label asp-for="ProviderAdminEmail" class="form-label"></label>
|
||||||
|
<input type="email" class="form-control" asp-for="ProviderAdminEmail" />
|
||||||
|
</div>
|
||||||
|
<button type="submit" class="btn btn-primary mb-2">Convert</button>
|
||||||
|
</form>
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user