mirror of
https://github.com/bitwarden/server.git
synced 2025-04-05 21:18:13 -05:00
properly handle patch operations with path values (#2190)
This commit is contained in:
parent
a89bfdfe2b
commit
62f29efb00
@ -165,35 +165,40 @@ namespace Bit.Scim.Controllers.v2
|
|||||||
}
|
}
|
||||||
|
|
||||||
var operationHandled = false;
|
var operationHandled = false;
|
||||||
|
foreach (var operation in model.Operations)
|
||||||
var replaceOp = model.Operations?.FirstOrDefault(o =>
|
{
|
||||||
o.Op?.ToLowerInvariant() == "replace");
|
// Replace operations
|
||||||
if (replaceOp != null)
|
if (operation.Op?.ToLowerInvariant() == "replace")
|
||||||
{
|
{
|
||||||
// Replace a list of members
|
// Replace a list of members
|
||||||
if (replaceOp.Path?.ToLowerInvariant() == "members")
|
if (operation.Path?.ToLowerInvariant() == "members")
|
||||||
{
|
{
|
||||||
var ids = GetOperationValueIds(replaceOp.Value);
|
var ids = GetOperationValueIds(operation.Value);
|
||||||
await _groupRepository.UpdateUsersAsync(group.Id, ids);
|
await _groupRepository.UpdateUsersAsync(group.Id, ids);
|
||||||
operationHandled = true;
|
operationHandled = true;
|
||||||
}
|
}
|
||||||
// Replace group name
|
// Replace group name from path
|
||||||
else if (replaceOp.Value.TryGetProperty("displayName", out var displayNameProperty))
|
else if (operation.Path?.ToLowerInvariant() == "displayname")
|
||||||
|
{
|
||||||
|
group.Name = operation.Value.GetString();
|
||||||
|
await _groupService.SaveAsync(group);
|
||||||
|
operationHandled = true;
|
||||||
|
}
|
||||||
|
// Replace group name from value object
|
||||||
|
else if (string.IsNullOrWhiteSpace(operation.Path) &&
|
||||||
|
operation.Value.TryGetProperty("displayName", out var displayNameProperty))
|
||||||
{
|
{
|
||||||
group.Name = displayNameProperty.GetString();
|
group.Name = displayNameProperty.GetString();
|
||||||
await _groupService.SaveAsync(group);
|
await _groupService.SaveAsync(group);
|
||||||
operationHandled = true;
|
operationHandled = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add a single member
|
// Add a single member
|
||||||
var addMemberOp = model.Operations?.FirstOrDefault(
|
else if (operation.Op?.ToLowerInvariant() == "add" &&
|
||||||
o => o.Op?.ToLowerInvariant() == "add" &&
|
!string.IsNullOrWhiteSpace(operation.Path) &&
|
||||||
!string.IsNullOrWhiteSpace(o.Path) &&
|
operation.Path.ToLowerInvariant().StartsWith("members[value eq "))
|
||||||
o.Path.ToLowerInvariant().StartsWith("members[value eq "));
|
|
||||||
if (addMemberOp != null)
|
|
||||||
{
|
{
|
||||||
var addId = GetOperationPathId(addMemberOp.Path);
|
var addId = GetOperationPathId(operation.Path);
|
||||||
if (addId.HasValue)
|
if (addId.HasValue)
|
||||||
{
|
{
|
||||||
var orgUserIds = (await _groupRepository.GetManyUserIdsByIdAsync(group.Id)).ToHashSet();
|
var orgUserIds = (await _groupRepository.GetManyUserIdsByIdAsync(group.Id)).ToHashSet();
|
||||||
@ -202,51 +207,43 @@ namespace Bit.Scim.Controllers.v2
|
|||||||
operationHandled = true;
|
operationHandled = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add a list of members
|
// Add a list of members
|
||||||
var addMembersOp = model.Operations?.FirstOrDefault(o =>
|
else if (operation.Op?.ToLowerInvariant() == "add" &&
|
||||||
o.Op?.ToLowerInvariant() == "add" &&
|
operation.Path?.ToLowerInvariant() == "members")
|
||||||
o.Path?.ToLowerInvariant() == "members");
|
|
||||||
if (addMembersOp != null)
|
|
||||||
{
|
{
|
||||||
var orgUserIds = (await _groupRepository.GetManyUserIdsByIdAsync(group.Id)).ToHashSet();
|
var orgUserIds = (await _groupRepository.GetManyUserIdsByIdAsync(group.Id)).ToHashSet();
|
||||||
foreach (var v in GetOperationValueIds(addMembersOp.Value))
|
foreach (var v in GetOperationValueIds(operation.Value))
|
||||||
{
|
{
|
||||||
orgUserIds.Add(v);
|
orgUserIds.Add(v);
|
||||||
}
|
}
|
||||||
await _groupRepository.UpdateUsersAsync(group.Id, orgUserIds);
|
await _groupRepository.UpdateUsersAsync(group.Id, orgUserIds);
|
||||||
operationHandled = true;
|
operationHandled = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove a single member
|
// Remove a single member
|
||||||
var removeMemberOp = model.Operations?.FirstOrDefault(
|
else if (operation.Op?.ToLowerInvariant() == "remove" &&
|
||||||
o => o.Op?.ToLowerInvariant() == "remove" &&
|
!string.IsNullOrWhiteSpace(operation.Path) &&
|
||||||
!string.IsNullOrWhiteSpace(o.Path) &&
|
operation.Path.ToLowerInvariant().StartsWith("members[value eq "))
|
||||||
o.Path.ToLowerInvariant().StartsWith("members[value eq "));
|
|
||||||
if (removeMemberOp != null)
|
|
||||||
{
|
{
|
||||||
var removeId = GetOperationPathId(removeMemberOp.Path);
|
var removeId = GetOperationPathId(operation.Path);
|
||||||
if (removeId.HasValue)
|
if (removeId.HasValue)
|
||||||
{
|
{
|
||||||
await _groupService.DeleteUserAsync(group, removeId.Value);
|
await _groupService.DeleteUserAsync(group, removeId.Value);
|
||||||
operationHandled = true;
|
operationHandled = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove a list of members
|
// Remove a list of members
|
||||||
var removeMembersOp = model.Operations?.FirstOrDefault(o =>
|
else if (operation.Op?.ToLowerInvariant() == "remove" &&
|
||||||
o.Op?.ToLowerInvariant() == "remove" &&
|
operation.Path?.ToLowerInvariant() == "members")
|
||||||
o.Path?.ToLowerInvariant() == "members");
|
|
||||||
if (removeMembersOp != null)
|
|
||||||
{
|
{
|
||||||
var orgUserIds = (await _groupRepository.GetManyUserIdsByIdAsync(group.Id)).ToHashSet();
|
var orgUserIds = (await _groupRepository.GetManyUserIdsByIdAsync(group.Id)).ToHashSet();
|
||||||
foreach (var v in GetOperationValueIds(removeMembersOp.Value))
|
foreach (var v in GetOperationValueIds(operation.Value))
|
||||||
{
|
{
|
||||||
orgUserIds.Remove(v);
|
orgUserIds.Remove(v);
|
||||||
}
|
}
|
||||||
await _groupRepository.UpdateUsersAsync(group.Id, orgUserIds);
|
await _groupRepository.UpdateUsersAsync(group.Id, orgUserIds);
|
||||||
operationHandled = true;
|
operationHandled = true;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (!operationHandled)
|
if (!operationHandled)
|
||||||
{
|
{
|
||||||
|
@ -224,23 +224,29 @@ namespace Bit.Scim.Controllers.v2
|
|||||||
}
|
}
|
||||||
|
|
||||||
var operationHandled = false;
|
var operationHandled = false;
|
||||||
|
foreach (var operation in model.Operations)
|
||||||
var replaceOp = model.Operations?.FirstOrDefault(o =>
|
|
||||||
o.Op?.ToLowerInvariant() == "replace");
|
|
||||||
if (replaceOp != null)
|
|
||||||
{
|
{
|
||||||
if (replaceOp.Value.TryGetProperty("active", out var activeProperty))
|
// Replace operations
|
||||||
|
if (operation.Op?.ToLowerInvariant() == "replace")
|
||||||
{
|
{
|
||||||
var active = activeProperty.GetBoolean();
|
// Active from path
|
||||||
if (active && orgUser.Status == OrganizationUserStatusType.Revoked)
|
if (operation.Path?.ToLowerInvariant() == "active")
|
||||||
{
|
{
|
||||||
await _organizationService.RestoreUserAsync(orgUser, null, _userService);
|
var handled = await HandleActiveOperationAsync(orgUser, operation.Value.GetBoolean());
|
||||||
operationHandled = true;
|
if (!operationHandled)
|
||||||
|
{
|
||||||
|
operationHandled = handled;
|
||||||
}
|
}
|
||||||
else if (!active && orgUser.Status != OrganizationUserStatusType.Revoked)
|
}
|
||||||
|
// Active from value object
|
||||||
|
else if (string.IsNullOrWhiteSpace(operation.Path) &&
|
||||||
|
operation.Value.TryGetProperty("active", out var activeProperty))
|
||||||
{
|
{
|
||||||
await _organizationService.RevokeUserAsync(orgUser, null);
|
var handled = await HandleActiveOperationAsync(orgUser, activeProperty.GetBoolean());
|
||||||
operationHandled = true;
|
if (!operationHandled)
|
||||||
|
{
|
||||||
|
operationHandled = handled;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -269,5 +275,20 @@ namespace Bit.Scim.Controllers.v2
|
|||||||
await _organizationService.DeleteUserAsync(organizationId, id, null);
|
await _organizationService.DeleteUserAsync(organizationId, id, null);
|
||||||
return new NoContentResult();
|
return new NoContentResult();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async Task<bool> HandleActiveOperationAsync(Core.Entities.OrganizationUser orgUser, bool active)
|
||||||
|
{
|
||||||
|
if (active && orgUser.Status == OrganizationUserStatusType.Revoked)
|
||||||
|
{
|
||||||
|
await _organizationService.RestoreUserAsync(orgUser, null, _userService);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else if (!active && orgUser.Status != OrganizationUserStatusType.Revoked)
|
||||||
|
{
|
||||||
|
await _organizationService.RevokeUserAsync(orgUser, null);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user