mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 05:00:19 -05:00
[NO LOGIC] Rename MultiOrganizationEnterprise to BusinessUnit
This commit is contained in:
parent
39ac93326d
commit
06c0c4cfc1
@ -48,7 +48,7 @@ public class CreateProviderCommand : ICreateProviderCommand
|
|||||||
await ProviderRepositoryCreateAsync(provider, ProviderStatusType.Created);
|
await ProviderRepositoryCreateAsync(provider, ProviderStatusType.Created);
|
||||||
}
|
}
|
||||||
|
|
||||||
public async Task CreateMultiOrganizationEnterpriseAsync(Provider provider, string ownerEmail, PlanType plan, int minimumSeats)
|
public async Task CreateBusinessUnitAsync(Provider provider, string ownerEmail, PlanType plan, int minimumSeats)
|
||||||
{
|
{
|
||||||
var providerId = await CreateProviderAsync(provider, ownerEmail);
|
var providerId = await CreateProviderAsync(provider, ownerEmail);
|
||||||
|
|
||||||
|
@ -692,10 +692,10 @@ public class ProviderService : IProviderService
|
|||||||
throw new BadRequestException($"Managed Service Providers cannot manage organizations with the plan type {requestedType}. Only Teams (Monthly) and Enterprise (Monthly) are allowed.");
|
throw new BadRequestException($"Managed Service Providers cannot manage organizations with the plan type {requestedType}. Only Teams (Monthly) and Enterprise (Monthly) are allowed.");
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case ProviderType.MultiOrganizationEnterprise:
|
case ProviderType.BusinessUnit:
|
||||||
if (requestedType is not (PlanType.EnterpriseMonthly or PlanType.EnterpriseAnnually))
|
if (requestedType is not (PlanType.EnterpriseMonthly or PlanType.EnterpriseAnnually))
|
||||||
{
|
{
|
||||||
throw new BadRequestException($"Multi-organization Enterprise Providers cannot manage organizations with the plan type {requestedType}. Only Enterprise (Monthly) and Enterprise (Annually) are allowed.");
|
throw new BadRequestException($"Business Unit Providers cannot manage organizations with the plan type {requestedType}. Only Enterprise (Monthly) and Enterprise (Annually) are allowed.");
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case ProviderType.Reseller:
|
case ProviderType.Reseller:
|
||||||
|
@ -791,7 +791,7 @@ public class ProviderBillingService(
|
|||||||
Provider provider,
|
Provider provider,
|
||||||
Organization organization)
|
Organization organization)
|
||||||
{
|
{
|
||||||
if (provider.Type == ProviderType.MultiOrganizationEnterprise)
|
if (provider.Type == ProviderType.BusinessUnit)
|
||||||
{
|
{
|
||||||
return (await providerPlanRepository.GetByProviderId(provider.Id)).First().PlanType;
|
return (await providerPlanRepository.GetByProviderId(provider.Id)).First().PlanType;
|
||||||
}
|
}
|
||||||
|
@ -51,7 +51,7 @@ public static class ProviderPriceAdapter
|
|||||||
/// <param name="subscription">The provider's subscription.</param>
|
/// <param name="subscription">The provider's subscription.</param>
|
||||||
/// <param name="planType">The plan type correlating to the desired Stripe price ID.</param>
|
/// <param name="planType">The plan type correlating to the desired Stripe price ID.</param>
|
||||||
/// <returns>A Stripe <see cref="Stripe.Price"/> ID.</returns>
|
/// <returns>A Stripe <see cref="Stripe.Price"/> ID.</returns>
|
||||||
/// <exception cref="BillingException">Thrown when the provider's type is not <see cref="ProviderType.Msp"/> or <see cref="ProviderType.MultiOrganizationEnterprise"/>.</exception>
|
/// <exception cref="BillingException">Thrown when the provider's type is not <see cref="ProviderType.Msp"/> or <see cref="ProviderType.BusinessUnit"/>.</exception>
|
||||||
/// <exception cref="BillingException">Thrown when the provided <see cref="planType"/> does not relate to a Stripe price ID.</exception>
|
/// <exception cref="BillingException">Thrown when the provided <see cref="planType"/> does not relate to a Stripe price ID.</exception>
|
||||||
public static string GetPriceId(
|
public static string GetPriceId(
|
||||||
Provider provider,
|
Provider provider,
|
||||||
@ -78,7 +78,7 @@ public static class ProviderPriceAdapter
|
|||||||
PlanType.EnterpriseMonthly => MSP.Active.Enterprise,
|
PlanType.EnterpriseMonthly => MSP.Active.Enterprise,
|
||||||
_ => throw invalidPlanType
|
_ => throw invalidPlanType
|
||||||
},
|
},
|
||||||
ProviderType.MultiOrganizationEnterprise => BusinessUnit.Legacy.List.Intersect(priceIds).Any()
|
ProviderType.BusinessUnit => BusinessUnit.Legacy.List.Intersect(priceIds).Any()
|
||||||
? planType switch
|
? planType switch
|
||||||
{
|
{
|
||||||
PlanType.EnterpriseAnnually => BusinessUnit.Legacy.Annually,
|
PlanType.EnterpriseAnnually => BusinessUnit.Legacy.Annually,
|
||||||
@ -103,7 +103,7 @@ public static class ProviderPriceAdapter
|
|||||||
/// <param name="provider">The provider to get the Stripe price ID for.</param>
|
/// <param name="provider">The provider to get the Stripe price ID for.</param>
|
||||||
/// <param name="planType">The plan type correlating to the desired Stripe price ID.</param>
|
/// <param name="planType">The plan type correlating to the desired Stripe price ID.</param>
|
||||||
/// <returns>A Stripe <see cref="Stripe.Price"/> ID.</returns>
|
/// <returns>A Stripe <see cref="Stripe.Price"/> ID.</returns>
|
||||||
/// <exception cref="BillingException">Thrown when the provider's type is not <see cref="ProviderType.Msp"/> or <see cref="ProviderType.MultiOrganizationEnterprise"/>.</exception>
|
/// <exception cref="BillingException">Thrown when the provider's type is not <see cref="ProviderType.Msp"/> or <see cref="ProviderType.BusinessUnit"/>.</exception>
|
||||||
/// <exception cref="BillingException">Thrown when the provided <see cref="planType"/> does not relate to a Stripe price ID.</exception>
|
/// <exception cref="BillingException">Thrown when the provided <see cref="planType"/> does not relate to a Stripe price ID.</exception>
|
||||||
public static string GetActivePriceId(
|
public static string GetActivePriceId(
|
||||||
Provider provider,
|
Provider provider,
|
||||||
@ -120,7 +120,7 @@ public static class ProviderPriceAdapter
|
|||||||
PlanType.EnterpriseMonthly => MSP.Active.Enterprise,
|
PlanType.EnterpriseMonthly => MSP.Active.Enterprise,
|
||||||
_ => throw invalidPlanType
|
_ => throw invalidPlanType
|
||||||
},
|
},
|
||||||
ProviderType.MultiOrganizationEnterprise => planType switch
|
ProviderType.BusinessUnit => planType switch
|
||||||
{
|
{
|
||||||
PlanType.EnterpriseAnnually => BusinessUnit.Active.Annually,
|
PlanType.EnterpriseAnnually => BusinessUnit.Active.Annually,
|
||||||
PlanType.EnterpriseMonthly => BusinessUnit.Active.Monthly,
|
PlanType.EnterpriseMonthly => BusinessUnit.Active.Monthly,
|
||||||
|
@ -63,7 +63,7 @@ public class CreateProviderCommandTests
|
|||||||
}
|
}
|
||||||
|
|
||||||
[Theory, BitAutoData]
|
[Theory, BitAutoData]
|
||||||
public async Task CreateMultiOrganizationEnterpriseAsync_Success(
|
public async Task CreateBusinessUnitAsync_Success(
|
||||||
Provider provider,
|
Provider provider,
|
||||||
User user,
|
User user,
|
||||||
PlanType plan,
|
PlanType plan,
|
||||||
@ -71,13 +71,13 @@ public class CreateProviderCommandTests
|
|||||||
SutProvider<CreateProviderCommand> sutProvider)
|
SutProvider<CreateProviderCommand> sutProvider)
|
||||||
{
|
{
|
||||||
// Arrange
|
// Arrange
|
||||||
provider.Type = ProviderType.MultiOrganizationEnterprise;
|
provider.Type = ProviderType.BusinessUnit;
|
||||||
|
|
||||||
var userRepository = sutProvider.GetDependency<IUserRepository>();
|
var userRepository = sutProvider.GetDependency<IUserRepository>();
|
||||||
userRepository.GetByEmailAsync(user.Email).Returns(user);
|
userRepository.GetByEmailAsync(user.Email).Returns(user);
|
||||||
|
|
||||||
// Act
|
// Act
|
||||||
await sutProvider.Sut.CreateMultiOrganizationEnterpriseAsync(provider, user.Email, plan, minimumSeats);
|
await sutProvider.Sut.CreateBusinessUnitAsync(provider, user.Email, plan, minimumSeats);
|
||||||
|
|
||||||
// Assert
|
// Assert
|
||||||
await sutProvider.GetDependency<IProviderRepository>().ReceivedWithAnyArgs().CreateAsync(provider);
|
await sutProvider.GetDependency<IProviderRepository>().ReceivedWithAnyArgs().CreateAsync(provider);
|
||||||
@ -85,7 +85,7 @@ public class CreateProviderCommandTests
|
|||||||
}
|
}
|
||||||
|
|
||||||
[Theory, BitAutoData]
|
[Theory, BitAutoData]
|
||||||
public async Task CreateMultiOrganizationEnterpriseAsync_UserIdIsInvalid_Throws(
|
public async Task CreateBusinessUnitAsync_UserIdIsInvalid_Throws(
|
||||||
Provider provider,
|
Provider provider,
|
||||||
SutProvider<CreateProviderCommand> sutProvider)
|
SutProvider<CreateProviderCommand> sutProvider)
|
||||||
{
|
{
|
||||||
@ -94,7 +94,7 @@ public class CreateProviderCommandTests
|
|||||||
|
|
||||||
// Act
|
// Act
|
||||||
var exception = await Assert.ThrowsAsync<BadRequestException>(
|
var exception = await Assert.ThrowsAsync<BadRequestException>(
|
||||||
() => sutProvider.Sut.CreateMultiOrganizationEnterpriseAsync(provider, default, default, default));
|
() => sutProvider.Sut.CreateBusinessUnitAsync(provider, default, default, default));
|
||||||
|
|
||||||
// Assert
|
// Assert
|
||||||
Assert.Contains("Invalid owner.", exception.Message);
|
Assert.Contains("Invalid owner.", exception.Message);
|
||||||
|
@ -116,7 +116,7 @@ public class ProviderBillingServiceTests
|
|||||||
SutProvider<ProviderBillingService> sutProvider)
|
SutProvider<ProviderBillingService> sutProvider)
|
||||||
{
|
{
|
||||||
// Arrange
|
// Arrange
|
||||||
provider.Type = ProviderType.MultiOrganizationEnterprise;
|
provider.Type = ProviderType.BusinessUnit;
|
||||||
|
|
||||||
var providerPlanRepository = sutProvider.GetDependency<IProviderPlanRepository>();
|
var providerPlanRepository = sutProvider.GetDependency<IProviderPlanRepository>();
|
||||||
var existingPlan = new ProviderPlan
|
var existingPlan = new ProviderPlan
|
||||||
|
@ -71,7 +71,7 @@ public class ProviderPriceAdapterTests
|
|||||||
var provider = new Provider
|
var provider = new Provider
|
||||||
{
|
{
|
||||||
Id = Guid.NewGuid(),
|
Id = Guid.NewGuid(),
|
||||||
Type = ProviderType.MultiOrganizationEnterprise
|
Type = ProviderType.BusinessUnit
|
||||||
};
|
};
|
||||||
|
|
||||||
var subscription = new Subscription
|
var subscription = new Subscription
|
||||||
@ -98,7 +98,7 @@ public class ProviderPriceAdapterTests
|
|||||||
var provider = new Provider
|
var provider = new Provider
|
||||||
{
|
{
|
||||||
Id = Guid.NewGuid(),
|
Id = Guid.NewGuid(),
|
||||||
Type = ProviderType.MultiOrganizationEnterprise
|
Type = ProviderType.BusinessUnit
|
||||||
};
|
};
|
||||||
|
|
||||||
var subscription = new Subscription
|
var subscription = new Subscription
|
||||||
@ -141,7 +141,7 @@ public class ProviderPriceAdapterTests
|
|||||||
var provider = new Provider
|
var provider = new Provider
|
||||||
{
|
{
|
||||||
Id = Guid.NewGuid(),
|
Id = Guid.NewGuid(),
|
||||||
Type = ProviderType.MultiOrganizationEnterprise
|
Type = ProviderType.BusinessUnit
|
||||||
};
|
};
|
||||||
|
|
||||||
var result = ProviderPriceAdapter.GetActivePriceId(provider, planType);
|
var result = ProviderPriceAdapter.GetActivePriceId(provider, planType);
|
||||||
|
@ -133,10 +133,10 @@ public class ProvidersController : Controller
|
|||||||
return View(new CreateResellerProviderModel());
|
return View(new CreateResellerProviderModel());
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpGet("providers/create/multi-organization-enterprise")]
|
[HttpGet("providers/create/business-unit")]
|
||||||
public IActionResult CreateMultiOrganizationEnterprise(int enterpriseMinimumSeats, string ownerEmail = null)
|
public IActionResult CreateBusinessUnit(int enterpriseMinimumSeats, string ownerEmail = null)
|
||||||
{
|
{
|
||||||
return View(new CreateMultiOrganizationEnterpriseProviderModel
|
return View(new CreateBusinessUnitProviderModel
|
||||||
{
|
{
|
||||||
OwnerEmail = ownerEmail,
|
OwnerEmail = ownerEmail,
|
||||||
EnterpriseSeatMinimum = enterpriseMinimumSeats
|
EnterpriseSeatMinimum = enterpriseMinimumSeats
|
||||||
@ -157,7 +157,7 @@ public class ProvidersController : Controller
|
|||||||
{
|
{
|
||||||
ProviderType.Msp => RedirectToAction("CreateMsp"),
|
ProviderType.Msp => RedirectToAction("CreateMsp"),
|
||||||
ProviderType.Reseller => RedirectToAction("CreateReseller"),
|
ProviderType.Reseller => RedirectToAction("CreateReseller"),
|
||||||
ProviderType.MultiOrganizationEnterprise => RedirectToAction("CreateMultiOrganizationEnterprise"),
|
ProviderType.BusinessUnit => RedirectToAction("CreateBusinessUnit"),
|
||||||
_ => View(model)
|
_ => View(model)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@ -198,10 +198,10 @@ public class ProvidersController : Controller
|
|||||||
return RedirectToAction("Edit", new { id = provider.Id });
|
return RedirectToAction("Edit", new { id = provider.Id });
|
||||||
}
|
}
|
||||||
|
|
||||||
[HttpPost("providers/create/multi-organization-enterprise")]
|
[HttpPost("providers/create/business-unit")]
|
||||||
[ValidateAntiForgeryToken]
|
[ValidateAntiForgeryToken]
|
||||||
[RequirePermission(Permission.Provider_Create)]
|
[RequirePermission(Permission.Provider_Create)]
|
||||||
public async Task<IActionResult> CreateMultiOrganizationEnterprise(CreateMultiOrganizationEnterpriseProviderModel model)
|
public async Task<IActionResult> CreateBusinessUnit(CreateBusinessUnitProviderModel model)
|
||||||
{
|
{
|
||||||
if (!ModelState.IsValid)
|
if (!ModelState.IsValid)
|
||||||
{
|
{
|
||||||
@ -209,7 +209,7 @@ public class ProvidersController : Controller
|
|||||||
}
|
}
|
||||||
var provider = model.ToProvider();
|
var provider = model.ToProvider();
|
||||||
|
|
||||||
await _createProviderCommand.CreateMultiOrganizationEnterpriseAsync(
|
await _createProviderCommand.CreateBusinessUnitAsync(
|
||||||
provider,
|
provider,
|
||||||
model.OwnerEmail,
|
model.OwnerEmail,
|
||||||
model.Plan.Value,
|
model.Plan.Value,
|
||||||
@ -307,7 +307,7 @@ public class ProvidersController : Controller
|
|||||||
]);
|
]);
|
||||||
await _providerBillingService.UpdateSeatMinimums(updateMspSeatMinimumsCommand);
|
await _providerBillingService.UpdateSeatMinimums(updateMspSeatMinimumsCommand);
|
||||||
break;
|
break;
|
||||||
case ProviderType.MultiOrganizationEnterprise:
|
case ProviderType.BusinessUnit:
|
||||||
{
|
{
|
||||||
var existingMoePlan = providerPlans.Single();
|
var existingMoePlan = providerPlans.Single();
|
||||||
|
|
||||||
|
@ -6,7 +6,7 @@ using Bit.SharedWeb.Utilities;
|
|||||||
|
|
||||||
namespace Bit.Admin.AdminConsole.Models;
|
namespace Bit.Admin.AdminConsole.Models;
|
||||||
|
|
||||||
public class CreateMultiOrganizationEnterpriseProviderModel : IValidatableObject
|
public class CreateBusinessUnitProviderModel : IValidatableObject
|
||||||
{
|
{
|
||||||
[Display(Name = "Owner Email")]
|
[Display(Name = "Owner Email")]
|
||||||
public string OwnerEmail { get; set; }
|
public string OwnerEmail { get; set; }
|
||||||
@ -22,7 +22,7 @@ public class CreateMultiOrganizationEnterpriseProviderModel : IValidatableObject
|
|||||||
{
|
{
|
||||||
return new Provider
|
return new Provider
|
||||||
{
|
{
|
||||||
Type = ProviderType.MultiOrganizationEnterprise
|
Type = ProviderType.BusinessUnit
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -30,17 +30,17 @@ public class CreateMultiOrganizationEnterpriseProviderModel : IValidatableObject
|
|||||||
{
|
{
|
||||||
if (string.IsNullOrWhiteSpace(OwnerEmail))
|
if (string.IsNullOrWhiteSpace(OwnerEmail))
|
||||||
{
|
{
|
||||||
var ownerEmailDisplayName = nameof(OwnerEmail).GetDisplayAttribute<CreateMultiOrganizationEnterpriseProviderModel>()?.GetName() ?? nameof(OwnerEmail);
|
var ownerEmailDisplayName = nameof(OwnerEmail).GetDisplayAttribute<CreateBusinessUnitProviderModel>()?.GetName() ?? nameof(OwnerEmail);
|
||||||
yield return new ValidationResult($"The {ownerEmailDisplayName} field is required.");
|
yield return new ValidationResult($"The {ownerEmailDisplayName} field is required.");
|
||||||
}
|
}
|
||||||
if (EnterpriseSeatMinimum < 0)
|
if (EnterpriseSeatMinimum < 0)
|
||||||
{
|
{
|
||||||
var enterpriseSeatMinimumDisplayName = nameof(EnterpriseSeatMinimum).GetDisplayAttribute<CreateMultiOrganizationEnterpriseProviderModel>()?.GetName() ?? nameof(EnterpriseSeatMinimum);
|
var enterpriseSeatMinimumDisplayName = nameof(EnterpriseSeatMinimum).GetDisplayAttribute<CreateBusinessUnitProviderModel>()?.GetName() ?? nameof(EnterpriseSeatMinimum);
|
||||||
yield return new ValidationResult($"The {enterpriseSeatMinimumDisplayName} field can not be negative.");
|
yield return new ValidationResult($"The {enterpriseSeatMinimumDisplayName} field can not be negative.");
|
||||||
}
|
}
|
||||||
if (Plan != PlanType.EnterpriseAnnually && Plan != PlanType.EnterpriseMonthly)
|
if (Plan != PlanType.EnterpriseAnnually && Plan != PlanType.EnterpriseMonthly)
|
||||||
{
|
{
|
||||||
var planDisplayName = nameof(Plan).GetDisplayAttribute<CreateMultiOrganizationEnterpriseProviderModel>()?.GetName() ?? nameof(Plan);
|
var planDisplayName = nameof(Plan).GetDisplayAttribute<CreateBusinessUnitProviderModel>()?.GetName() ?? nameof(Plan);
|
||||||
yield return new ValidationResult($"The {planDisplayName} field must be set to Enterprise Annually or Enterprise Monthly.");
|
yield return new ValidationResult($"The {planDisplayName} field must be set to Enterprise Annually or Enterprise Monthly.");
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -34,7 +34,7 @@ public class ProviderEditModel : ProviderViewModel, IValidatableObject
|
|||||||
GatewaySubscriptionUrl = gatewaySubscriptionUrl;
|
GatewaySubscriptionUrl = gatewaySubscriptionUrl;
|
||||||
Type = provider.Type;
|
Type = provider.Type;
|
||||||
|
|
||||||
if (Type == ProviderType.MultiOrganizationEnterprise)
|
if (Type == ProviderType.BusinessUnit)
|
||||||
{
|
{
|
||||||
var plan = providerPlans.SingleOrDefault();
|
var plan = providerPlans.SingleOrDefault();
|
||||||
EnterpriseMinimumSeats = plan?.SeatMinimum ?? 0;
|
EnterpriseMinimumSeats = plan?.SeatMinimum ?? 0;
|
||||||
@ -100,7 +100,7 @@ public class ProviderEditModel : ProviderViewModel, IValidatableObject
|
|||||||
yield return new ValidationResult($"The {billingEmailDisplayName} field is required.");
|
yield return new ValidationResult($"The {billingEmailDisplayName} field is required.");
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case ProviderType.MultiOrganizationEnterprise:
|
case ProviderType.BusinessUnit:
|
||||||
if (Plan == null)
|
if (Plan == null)
|
||||||
{
|
{
|
||||||
var displayName = nameof(Plan).GetDisplayAttribute<CreateProviderModel>()?.GetName() ?? nameof(Plan);
|
var displayName = nameof(Plan).GetDisplayAttribute<CreateProviderModel>()?.GetName() ?? nameof(Plan);
|
||||||
|
@ -40,7 +40,7 @@ public class ProviderViewModel
|
|||||||
ProviderPlanViewModels.Add(new ProviderPlanViewModel("Enterprise (Monthly) Subscription", enterpriseProviderPlan, usedEnterpriseSeats));
|
ProviderPlanViewModels.Add(new ProviderPlanViewModel("Enterprise (Monthly) Subscription", enterpriseProviderPlan, usedEnterpriseSeats));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (Provider.Type == ProviderType.MultiOrganizationEnterprise)
|
else if (Provider.Type == ProviderType.BusinessUnit)
|
||||||
{
|
{
|
||||||
var usedEnterpriseSeats = ProviderOrganizations.Where(po => po.PlanType == PlanType.EnterpriseMonthly)
|
var usedEnterpriseSeats = ProviderOrganizations.Where(po => po.PlanType == PlanType.EnterpriseMonthly)
|
||||||
.Sum(po => po.OccupiedSeats).GetValueOrDefault(0);
|
.Sum(po => po.OccupiedSeats).GetValueOrDefault(0);
|
||||||
|
@ -1,15 +1,15 @@
|
|||||||
@using Bit.Core.Billing.Enums
|
@using Bit.Core.Billing.Enums
|
||||||
@using Microsoft.AspNetCore.Mvc.TagHelpers
|
@using Microsoft.AspNetCore.Mvc.TagHelpers
|
||||||
|
|
||||||
@model CreateMultiOrganizationEnterpriseProviderModel
|
@model CreateBusinessUnitProviderModel
|
||||||
|
|
||||||
@{
|
@{
|
||||||
ViewData["Title"] = "Create Multi-organization Enterprise Provider";
|
ViewData["Title"] = "Create Business Unit Provider";
|
||||||
}
|
}
|
||||||
|
|
||||||
<h1 class="mb-4">Create Multi-organization Enterprise Provider</h1>
|
<h1 class="mb-4">Create Business Unit Provider</h1>
|
||||||
<div>
|
<div>
|
||||||
<form method="post" asp-action="CreateMultiOrganizationEnterprise">
|
<form method="post" asp-action="CreateBusinessUnit">
|
||||||
<div asp-validation-summary="All" class="alert alert-danger"></div>
|
<div asp-validation-summary="All" class="alert alert-danger"></div>
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
<label asp-for="OwnerEmail" class="form-label"></label>
|
<label asp-for="OwnerEmail" class="form-label"></label>
|
||||||
@ -19,14 +19,14 @@
|
|||||||
<div class="col-sm">
|
<div class="col-sm">
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
@{
|
@{
|
||||||
var multiOrgPlans = new List<PlanType>
|
var businessUnitPlanTypes = new List<PlanType>
|
||||||
{
|
{
|
||||||
PlanType.EnterpriseAnnually,
|
PlanType.EnterpriseAnnually,
|
||||||
PlanType.EnterpriseMonthly
|
PlanType.EnterpriseMonthly
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
<label asp-for="Plan" class="form-label"></label>
|
<label asp-for="Plan" class="form-label"></label>
|
||||||
<select class="form-select" asp-for="Plan" asp-items="Html.GetEnumSelectList(multiOrgPlans)">
|
<select class="form-select" asp-for="Plan" asp-items="Html.GetEnumSelectList(businessUnitPlanTypes)">
|
||||||
<option value="">--</option>
|
<option value="">--</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
@ -74,20 +74,20 @@
|
|||||||
</div>
|
</div>
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case ProviderType.MultiOrganizationEnterprise:
|
case ProviderType.BusinessUnit:
|
||||||
{
|
{
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<div class="col-sm">
|
<div class="col-sm">
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
@{
|
@{
|
||||||
var multiOrgPlans = new List<PlanType>
|
var businessUnitPlanTypes = new List<PlanType>
|
||||||
{
|
{
|
||||||
PlanType.EnterpriseAnnually,
|
PlanType.EnterpriseAnnually,
|
||||||
PlanType.EnterpriseMonthly
|
PlanType.EnterpriseMonthly
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
<label asp-for="Plan" class="form-label"></label>
|
<label asp-for="Plan" class="form-label"></label>
|
||||||
<select class="form-control" asp-for="Plan" asp-items="Html.GetEnumSelectList(multiOrgPlans)">
|
<select class="form-control" asp-for="Plan" asp-items="Html.GetEnumSelectList(businessUnitPlanTypes)">
|
||||||
<option value="">--</option>
|
<option value="">--</option>
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
|
@ -8,6 +8,6 @@ public enum ProviderType : byte
|
|||||||
Msp = 0,
|
Msp = 0,
|
||||||
[Display(ShortName = "Reseller", Name = "Reseller", Description = "Creates Bitwarden Portal page for client organization billing management", Order = 1000)]
|
[Display(ShortName = "Reseller", Name = "Reseller", Description = "Creates Bitwarden Portal page for client organization billing management", Order = 1000)]
|
||||||
Reseller = 1,
|
Reseller = 1,
|
||||||
[Display(ShortName = "MOE", Name = "Multi-organization Enterprises", Description = "Creates provider portal for multi-organization management", Order = 1)]
|
[Display(ShortName = "Business Unit", Name = "Business Unit", Description = "Creates provider portal for business unit management", Order = 1)]
|
||||||
MultiOrganizationEnterprise = 2,
|
BusinessUnit = 2,
|
||||||
}
|
}
|
||||||
|
@ -7,5 +7,5 @@ public interface ICreateProviderCommand
|
|||||||
{
|
{
|
||||||
Task CreateMspAsync(Provider provider, string ownerEmail, int teamsMinimumSeats, int enterpriseMinimumSeats);
|
Task CreateMspAsync(Provider provider, string ownerEmail, int teamsMinimumSeats, int enterpriseMinimumSeats);
|
||||||
Task CreateResellerAsync(Provider provider);
|
Task CreateResellerAsync(Provider provider);
|
||||||
Task CreateMultiOrganizationEnterpriseAsync(Provider provider, string ownerEmail, PlanType plan, int minimumSeats);
|
Task CreateBusinessUnitAsync(Provider provider, string ownerEmail, PlanType plan, int minimumSeats);
|
||||||
}
|
}
|
||||||
|
@ -24,12 +24,12 @@ public static class BillingExtensions
|
|||||||
public static bool IsBillable(this Provider provider) =>
|
public static bool IsBillable(this Provider provider) =>
|
||||||
provider is
|
provider is
|
||||||
{
|
{
|
||||||
Type: ProviderType.Msp or ProviderType.MultiOrganizationEnterprise,
|
Type: ProviderType.Msp or ProviderType.BusinessUnit,
|
||||||
Status: ProviderStatusType.Billable
|
Status: ProviderStatusType.Billable
|
||||||
};
|
};
|
||||||
|
|
||||||
public static bool SupportsConsolidatedBilling(this ProviderType providerType)
|
public static bool SupportsConsolidatedBilling(this ProviderType providerType)
|
||||||
=> providerType is ProviderType.Msp or ProviderType.MultiOrganizationEnterprise;
|
=> providerType is ProviderType.Msp or ProviderType.BusinessUnit;
|
||||||
|
|
||||||
public static bool IsValidClient(this Organization organization)
|
public static bool IsValidClient(this Organization organization)
|
||||||
=> organization is
|
=> organization is
|
||||||
|
@ -330,7 +330,7 @@ public class OrganizationRepository : Repository<Core.AdminConsole.Entities.Orga
|
|||||||
var planTypes = providerType switch
|
var planTypes = providerType switch
|
||||||
{
|
{
|
||||||
ProviderType.Msp => PlanConstants.EnterprisePlanTypes.Concat(PlanConstants.TeamsPlanTypes),
|
ProviderType.Msp => PlanConstants.EnterprisePlanTypes.Concat(PlanConstants.TeamsPlanTypes),
|
||||||
ProviderType.MultiOrganizationEnterprise => PlanConstants.EnterprisePlanTypes,
|
ProviderType.BusinessUnit => PlanConstants.EnterprisePlanTypes,
|
||||||
_ => []
|
_ => []
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -75,25 +75,25 @@ public class ProvidersControllerTests
|
|||||||
}
|
}
|
||||||
#endregion
|
#endregion
|
||||||
|
|
||||||
#region CreateMultiOrganizationEnterpriseAsync
|
#region CreateBusinessUnitAsync
|
||||||
[BitAutoData]
|
[BitAutoData]
|
||||||
[SutProviderCustomize]
|
[SutProviderCustomize]
|
||||||
[Theory]
|
[Theory]
|
||||||
public async Task CreateMultiOrganizationEnterpriseAsync_WithValidModel_CreatesProvider(
|
public async Task CreateBusinessUnitAsync_WithValidModel_CreatesProvider(
|
||||||
CreateMultiOrganizationEnterpriseProviderModel model,
|
CreateBusinessUnitProviderModel model,
|
||||||
SutProvider<ProvidersController> sutProvider)
|
SutProvider<ProvidersController> sutProvider)
|
||||||
{
|
{
|
||||||
// Arrange
|
// Arrange
|
||||||
|
|
||||||
// Act
|
// Act
|
||||||
var actual = await sutProvider.Sut.CreateMultiOrganizationEnterprise(model);
|
var actual = await sutProvider.Sut.CreateBusinessUnit(model);
|
||||||
|
|
||||||
// Assert
|
// Assert
|
||||||
Assert.NotNull(actual);
|
Assert.NotNull(actual);
|
||||||
await sutProvider.GetDependency<ICreateProviderCommand>()
|
await sutProvider.GetDependency<ICreateProviderCommand>()
|
||||||
.Received(Quantity.Exactly(1))
|
.Received(Quantity.Exactly(1))
|
||||||
.CreateMultiOrganizationEnterpriseAsync(
|
.CreateBusinessUnitAsync(
|
||||||
Arg.Is<Provider>(x => x.Type == ProviderType.MultiOrganizationEnterprise),
|
Arg.Is<Provider>(x => x.Type == ProviderType.BusinessUnit),
|
||||||
model.OwnerEmail,
|
model.OwnerEmail,
|
||||||
Arg.Is<PlanType>(y => y == model.Plan),
|
Arg.Is<PlanType>(y => y == model.Plan),
|
||||||
model.EnterpriseSeatMinimum);
|
model.EnterpriseSeatMinimum);
|
||||||
@ -102,16 +102,16 @@ public class ProvidersControllerTests
|
|||||||
[BitAutoData]
|
[BitAutoData]
|
||||||
[SutProviderCustomize]
|
[SutProviderCustomize]
|
||||||
[Theory]
|
[Theory]
|
||||||
public async Task CreateMultiOrganizationEnterpriseAsync_RedirectsToExpectedPage_AfterCreatingProvider(
|
public async Task CreateBusinessUnitAsync_RedirectsToExpectedPage_AfterCreatingProvider(
|
||||||
CreateMultiOrganizationEnterpriseProviderModel model,
|
CreateBusinessUnitProviderModel model,
|
||||||
Guid expectedProviderId,
|
Guid expectedProviderId,
|
||||||
SutProvider<ProvidersController> sutProvider)
|
SutProvider<ProvidersController> sutProvider)
|
||||||
{
|
{
|
||||||
// Arrange
|
// Arrange
|
||||||
sutProvider.GetDependency<ICreateProviderCommand>()
|
sutProvider.GetDependency<ICreateProviderCommand>()
|
||||||
.When(x =>
|
.When(x =>
|
||||||
x.CreateMultiOrganizationEnterpriseAsync(
|
x.CreateBusinessUnitAsync(
|
||||||
Arg.Is<Provider>(y => y.Type == ProviderType.MultiOrganizationEnterprise),
|
Arg.Is<Provider>(y => y.Type == ProviderType.BusinessUnit),
|
||||||
model.OwnerEmail,
|
model.OwnerEmail,
|
||||||
Arg.Is<PlanType>(y => y == model.Plan),
|
Arg.Is<PlanType>(y => y == model.Plan),
|
||||||
model.EnterpriseSeatMinimum))
|
model.EnterpriseSeatMinimum))
|
||||||
@ -122,7 +122,7 @@ public class ProvidersControllerTests
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Act
|
// Act
|
||||||
var actual = await sutProvider.Sut.CreateMultiOrganizationEnterprise(model);
|
var actual = await sutProvider.Sut.CreateBusinessUnit(model);
|
||||||
|
|
||||||
// Assert
|
// Assert
|
||||||
Assert.NotNull(actual);
|
Assert.NotNull(actual);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user