mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 05:00:19 -05:00

* Add auto-scale support to standalone SM for SCIM * Mark users for SM when using SM Stadalone with Directory Connector
101 lines
3.5 KiB
C#
101 lines
3.5 KiB
C#
using Bit.Core.AdminConsole.Enums;
|
|
using Bit.Core.Enums;
|
|
using Bit.Core.Exceptions;
|
|
using Bit.Core.Models.Data;
|
|
using Bit.Core.Models.Data.Organizations.OrganizationUsers;
|
|
using Bit.Core.Repositories;
|
|
using Bit.Core.Services;
|
|
using Bit.Core.Utilities;
|
|
using Bit.Scim.Context;
|
|
using Bit.Scim.Models;
|
|
using Bit.Scim.Users.Interfaces;
|
|
|
|
namespace Bit.Scim.Users;
|
|
|
|
public class PostUserCommand : IPostUserCommand
|
|
{
|
|
private readonly IOrganizationRepository _organizationRepository;
|
|
private readonly IOrganizationUserRepository _organizationUserRepository;
|
|
private readonly IOrganizationService _organizationService;
|
|
private readonly IPaymentService _paymentService;
|
|
private readonly IScimContext _scimContext;
|
|
|
|
public PostUserCommand(
|
|
IOrganizationRepository organizationRepository,
|
|
IOrganizationUserRepository organizationUserRepository,
|
|
IOrganizationService organizationService,
|
|
IPaymentService paymentService,
|
|
IScimContext scimContext)
|
|
{
|
|
_organizationRepository = organizationRepository;
|
|
_organizationUserRepository = organizationUserRepository;
|
|
_organizationService = organizationService;
|
|
_paymentService = paymentService;
|
|
_scimContext = scimContext;
|
|
}
|
|
|
|
public async Task<OrganizationUserUserDetails> PostUserAsync(Guid organizationId, ScimUserRequestModel model)
|
|
{
|
|
var email = model.PrimaryEmail?.ToLowerInvariant();
|
|
if (string.IsNullOrWhiteSpace(email))
|
|
{
|
|
switch (_scimContext.RequestScimProvider)
|
|
{
|
|
case ScimProviderType.AzureAd:
|
|
email = model.UserName?.ToLowerInvariant();
|
|
break;
|
|
default:
|
|
email = model.WorkEmail?.ToLowerInvariant();
|
|
if (string.IsNullOrWhiteSpace(email))
|
|
{
|
|
email = model.Emails?.FirstOrDefault()?.Value?.ToLowerInvariant();
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (string.IsNullOrWhiteSpace(email) || !model.Active)
|
|
{
|
|
throw new BadRequestException();
|
|
}
|
|
|
|
var orgUsers = await _organizationUserRepository.GetManyDetailsByOrganizationAsync(organizationId);
|
|
var orgUserByEmail = orgUsers.FirstOrDefault(ou => ou.Email?.ToLowerInvariant() == email);
|
|
if (orgUserByEmail != null)
|
|
{
|
|
throw new ConflictException();
|
|
}
|
|
|
|
string externalId = null;
|
|
if (!string.IsNullOrWhiteSpace(model.ExternalId))
|
|
{
|
|
externalId = model.ExternalId;
|
|
}
|
|
else if (!string.IsNullOrWhiteSpace(model.UserName))
|
|
{
|
|
externalId = model.UserName;
|
|
}
|
|
else
|
|
{
|
|
externalId = CoreHelpers.RandomString(15);
|
|
}
|
|
|
|
var orgUserByExternalId = orgUsers.FirstOrDefault(ou => ou.ExternalId == externalId);
|
|
if (orgUserByExternalId != null)
|
|
{
|
|
throw new ConflictException();
|
|
}
|
|
|
|
var organization = await _organizationRepository.GetByIdAsync(organizationId);
|
|
|
|
var hasStandaloneSecretsManager = await _paymentService.HasSecretsManagerStandalone(organization);
|
|
|
|
var invitedOrgUser = await _organizationService.InviteUserAsync(organizationId, EventSystemUser.SCIM, email,
|
|
OrganizationUserType.User, false, externalId, new List<CollectionAccessSelection>(), new List<Guid>(), hasStandaloneSecretsManager);
|
|
|
|
var orgUser = await _organizationUserRepository.GetDetailsByIdAsync(invitedOrgUser.Id);
|
|
|
|
return orgUser;
|
|
}
|
|
}
|