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

* Copy PatchGroupCommand to vNext and refactor * Detect duplicate add requests and return early * Update read repository method to use HA replica * Add new write repository method
171 lines
6.7 KiB
C#
171 lines
6.7 KiB
C#
using System.Text.Json;
|
|
using Bit.Core.AdminConsole.Entities;
|
|
using Bit.Core.AdminConsole.OrganizationFeatures.Groups.Interfaces;
|
|
using Bit.Core.AdminConsole.Repositories;
|
|
using Bit.Core.AdminConsole.Services;
|
|
using Bit.Core.Enums;
|
|
using Bit.Core.Exceptions;
|
|
using Bit.Core.Repositories;
|
|
using Bit.Scim.Groups.Interfaces;
|
|
using Bit.Scim.Models;
|
|
using Bit.Scim.Utilities;
|
|
|
|
namespace Bit.Scim.Groups;
|
|
|
|
public class PatchGroupCommandvNext : IPatchGroupCommandvNext
|
|
{
|
|
private readonly IGroupRepository _groupRepository;
|
|
private readonly IGroupService _groupService;
|
|
private readonly IUpdateGroupCommand _updateGroupCommand;
|
|
private readonly ILogger<PatchGroupCommandvNext> _logger;
|
|
private readonly IOrganizationRepository _organizationRepository;
|
|
|
|
public PatchGroupCommandvNext(
|
|
IGroupRepository groupRepository,
|
|
IGroupService groupService,
|
|
IUpdateGroupCommand updateGroupCommand,
|
|
ILogger<PatchGroupCommandvNext> logger,
|
|
IOrganizationRepository organizationRepository)
|
|
{
|
|
_groupRepository = groupRepository;
|
|
_groupService = groupService;
|
|
_updateGroupCommand = updateGroupCommand;
|
|
_logger = logger;
|
|
_organizationRepository = organizationRepository;
|
|
}
|
|
|
|
public async Task PatchGroupAsync(Group group, ScimPatchModel model)
|
|
{
|
|
foreach (var operation in model.Operations)
|
|
{
|
|
await HandleOperationAsync(group, operation);
|
|
}
|
|
}
|
|
|
|
private async Task HandleOperationAsync(Group group, ScimPatchModel.OperationModel operation)
|
|
{
|
|
switch (operation.Op?.ToLowerInvariant())
|
|
{
|
|
// Replace a list of members
|
|
case PatchOps.Replace when operation.Path?.ToLowerInvariant() == PatchPaths.Members:
|
|
{
|
|
var ids = GetOperationValueIds(operation.Value);
|
|
await _groupRepository.UpdateUsersAsync(group.Id, ids);
|
|
break;
|
|
}
|
|
|
|
// Replace group name from path
|
|
case PatchOps.Replace when operation.Path?.ToLowerInvariant() == PatchPaths.DisplayName:
|
|
{
|
|
group.Name = operation.Value.GetString();
|
|
var organization = await _organizationRepository.GetByIdAsync(group.OrganizationId);
|
|
if (organization == null)
|
|
{
|
|
throw new NotFoundException();
|
|
}
|
|
await _updateGroupCommand.UpdateGroupAsync(group, organization, EventSystemUser.SCIM);
|
|
break;
|
|
}
|
|
|
|
// Replace group name from value object
|
|
case PatchOps.Replace when
|
|
string.IsNullOrWhiteSpace(operation.Path) &&
|
|
operation.Value.TryGetProperty("displayName", out var displayNameProperty):
|
|
{
|
|
group.Name = displayNameProperty.GetString();
|
|
var organization = await _organizationRepository.GetByIdAsync(group.OrganizationId);
|
|
if (organization == null)
|
|
{
|
|
throw new NotFoundException();
|
|
}
|
|
await _updateGroupCommand.UpdateGroupAsync(group, organization, EventSystemUser.SCIM);
|
|
break;
|
|
}
|
|
|
|
// Add a single member
|
|
case PatchOps.Add when
|
|
!string.IsNullOrWhiteSpace(operation.Path) &&
|
|
operation.Path.StartsWith("members[value eq ", StringComparison.OrdinalIgnoreCase) &&
|
|
TryGetOperationPathId(operation.Path, out var addId):
|
|
{
|
|
await AddMembersAsync(group, [addId]);
|
|
break;
|
|
}
|
|
|
|
// Add a list of members
|
|
case PatchOps.Add when
|
|
operation.Path?.ToLowerInvariant() == PatchPaths.Members:
|
|
{
|
|
await AddMembersAsync(group, GetOperationValueIds(operation.Value));
|
|
break;
|
|
}
|
|
|
|
// Remove a single member
|
|
case PatchOps.Remove when
|
|
!string.IsNullOrWhiteSpace(operation.Path) &&
|
|
operation.Path.StartsWith("members[value eq ", StringComparison.OrdinalIgnoreCase) &&
|
|
TryGetOperationPathId(operation.Path, out var removeId):
|
|
{
|
|
await _groupService.DeleteUserAsync(group, removeId, EventSystemUser.SCIM);
|
|
break;
|
|
}
|
|
|
|
// Remove a list of members
|
|
case PatchOps.Remove when
|
|
operation.Path?.ToLowerInvariant() == PatchPaths.Members:
|
|
{
|
|
var orgUserIds = (await _groupRepository.GetManyUserIdsByIdAsync(group.Id)).ToHashSet();
|
|
foreach (var v in GetOperationValueIds(operation.Value))
|
|
{
|
|
orgUserIds.Remove(v);
|
|
}
|
|
await _groupRepository.UpdateUsersAsync(group.Id, orgUserIds);
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
_logger.LogWarning("Group patch operation not handled: {OperationOp}:{OperationPath}", operation.Op, operation.Path);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
private async Task AddMembersAsync(Group group, HashSet<Guid> usersToAdd)
|
|
{
|
|
// Azure Entra ID is known to send redundant "add" requests for each existing member every time any member
|
|
// is removed. To avoid excessive load on the database, we check against the high availability replica and
|
|
// return early if they already exist.
|
|
var groupMembers = await _groupRepository.GetManyUserIdsByIdAsync(group.Id, useReadOnlyReplica: true);
|
|
if (usersToAdd.IsSubsetOf(groupMembers))
|
|
{
|
|
_logger.LogDebug("Ignoring duplicate SCIM request to add members {Members} to group {Group}", usersToAdd, group.Id);
|
|
return;
|
|
}
|
|
|
|
await _groupRepository.AddGroupUsersByIdAsync(group.Id, usersToAdd);
|
|
}
|
|
|
|
private static HashSet<Guid> GetOperationValueIds(JsonElement objArray)
|
|
{
|
|
var ids = new HashSet<Guid>();
|
|
foreach (var obj in objArray.EnumerateArray())
|
|
{
|
|
if (obj.TryGetProperty("value", out var valueProperty))
|
|
{
|
|
if (valueProperty.TryGetGuid(out var guid))
|
|
{
|
|
ids.Add(guid);
|
|
}
|
|
}
|
|
}
|
|
return ids;
|
|
}
|
|
|
|
private static bool TryGetOperationPathId(string path, out Guid pathId)
|
|
{
|
|
// Parse Guid from string like: members[value eq "{GUID}"}]
|
|
return Guid.TryParse(path.Substring(18).Replace("\"]", string.Empty), out pathId);
|
|
}
|
|
}
|