mirror of
https://github.com/bitwarden/server.git
synced 2025-04-06 05:28:15 -05:00
Added bulk procedure for saving users, collections and groups from inviting. Added test to validate Ef and Sproc
This commit is contained in:
parent
fcaa449f83
commit
926e786f82
@ -5,6 +5,7 @@ using Bit.Core.Enums;
|
|||||||
using Bit.Core.Models.Commands;
|
using Bit.Core.Models.Commands;
|
||||||
using Bit.Core.Repositories;
|
using Bit.Core.Repositories;
|
||||||
using Bit.Core.Services;
|
using Bit.Core.Services;
|
||||||
|
using static Bit.Core.AdminConsole.OrganizationFeatures.OrganizationUsers.InviteUsers.Models.CreateOrganizationUserExtensions;
|
||||||
|
|
||||||
namespace Bit.Core.AdminConsole.OrganizationFeatures.OrganizationUsers.InviteUsers;
|
namespace Bit.Core.AdminConsole.OrganizationFeatures.OrganizationUsers.InviteUsers;
|
||||||
|
|
||||||
@ -22,6 +23,11 @@ public class InviteOrganizationUsersCommand(IEventService eventService,
|
|||||||
{
|
{
|
||||||
var result = await InviteOrganizationUsersAsync(InviteOrganizationUsersRequest.Create(request));
|
var result = await InviteOrganizationUsersAsync(InviteOrganizationUsersRequest.Create(request));
|
||||||
|
|
||||||
|
if (result is Failure<IEnumerable<OrganizationUser>> failure)
|
||||||
|
{
|
||||||
|
return new Failure<OrganizationUser>(failure.ErrorMessage);
|
||||||
|
}
|
||||||
|
|
||||||
if (result.Value.Any())
|
if (result.Value.Any())
|
||||||
{
|
{
|
||||||
(OrganizationUser User, EventType type, EventSystemUser system, DateTime performedAt) log = (result.Value.First(), EventType.OrganizationUser_Invited, EventSystemUser.SCIM, request.PerformedAt.UtcDateTime);
|
(OrganizationUser User, EventType type, EventSystemUser system, DateTime performedAt) log = (result.Value.First(), EventType.OrganizationUser_Invited, EventSystemUser.SCIM, request.PerformedAt.UtcDateTime);
|
||||||
@ -29,7 +35,7 @@ public class InviteOrganizationUsersCommand(IEventService eventService,
|
|||||||
await eventService.LogOrganizationUserEventsAsync([log]);
|
await eventService.LogOrganizationUserEventsAsync([log]);
|
||||||
}
|
}
|
||||||
|
|
||||||
return new CommandResult<OrganizationUser>(result.Value.FirstOrDefault());
|
return new Success<OrganizationUser>(result.Value.FirstOrDefault());
|
||||||
}
|
}
|
||||||
|
|
||||||
private async Task<CommandResult<IEnumerable<OrganizationUser>>> InviteOrganizationUsersAsync(InviteOrganizationUsersRequest request)
|
private async Task<CommandResult<IEnumerable<OrganizationUser>>> InviteOrganizationUsersAsync(InviteOrganizationUsersRequest request)
|
||||||
@ -41,7 +47,7 @@ public class InviteOrganizationUsersCommand(IEventService eventService,
|
|||||||
var invitesToSend = request.Invites
|
var invitesToSend = request.Invites
|
||||||
.SelectMany(invite => invite.Emails
|
.SelectMany(invite => invite.Emails
|
||||||
.Where(email => !existingEmails.Contains(email))
|
.Where(email => !existingEmails.Contains(email))
|
||||||
.Select(email => OrganizationUserInviteDto.Create(email, invite))
|
.Select(email => OrganizationUserInviteDto.Create(email, invite, request.Organization.OrganizationId))
|
||||||
);
|
);
|
||||||
|
|
||||||
// Validate we can add those seats
|
// Validate we can add those seats
|
||||||
@ -55,8 +61,17 @@ public class InviteOrganizationUsersCommand(IEventService eventService,
|
|||||||
OccupiedSmSeats = await organizationUserRepository.GetOccupiedSmSeatCountByOrganizationIdAsync(request.Organization.OrganizationId)
|
OccupiedSmSeats = await organizationUserRepository.GetOccupiedSmSeatCountByOrganizationIdAsync(request.Organization.OrganizationId)
|
||||||
});
|
});
|
||||||
|
|
||||||
|
if (!validationResult.IsValid)
|
||||||
|
{
|
||||||
|
return new Failure<IEnumerable<OrganizationUser>>(validationResult.ErrorMessageString);
|
||||||
|
}
|
||||||
|
|
||||||
|
var organizationUserCollection = invitesToSend
|
||||||
|
.Select(MapToDataModel(request.PerformedAt));
|
||||||
|
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
|
await organizationUserRepository.CreateManyAsync(organizationUserCollection);
|
||||||
// save organization users
|
// save organization users
|
||||||
// org users
|
// org users
|
||||||
// collections
|
// collections
|
||||||
|
@ -0,0 +1,35 @@
|
|||||||
|
using Bit.Core.Entities;
|
||||||
|
using Bit.Core.Enums;
|
||||||
|
using Bit.Core.Models.Data;
|
||||||
|
using Bit.Core.Utilities;
|
||||||
|
|
||||||
|
namespace Bit.Core.AdminConsole.OrganizationFeatures.OrganizationUsers.InviteUsers.Models;
|
||||||
|
|
||||||
|
public class CreateOrganizationUser
|
||||||
|
{
|
||||||
|
public OrganizationUser User { get; set; }
|
||||||
|
public CollectionAccessSelection[] Collections { get; set; } = [];
|
||||||
|
public Guid[] Groups { get; set; } = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
public static class CreateOrganizationUserExtensions
|
||||||
|
{
|
||||||
|
public static Func<OrganizationUserInviteDto, CreateOrganizationUser> MapToDataModel(DateTimeOffset performedAt) =>
|
||||||
|
o => new CreateOrganizationUser
|
||||||
|
{
|
||||||
|
User = new OrganizationUser
|
||||||
|
{
|
||||||
|
Id = CoreHelpers.GenerateComb(),
|
||||||
|
OrganizationId = o.OrganizationId,
|
||||||
|
Email = o.Email.ToLowerInvariant(),
|
||||||
|
Type = o.Type,
|
||||||
|
Status = OrganizationUserStatusType.Invited,
|
||||||
|
AccessSecretsManager = o.AccessSecretsManager,
|
||||||
|
ExternalId = string.IsNullOrWhiteSpace(o.ExternalId) ? null : o.ExternalId,
|
||||||
|
CreationDate = performedAt.UtcDateTime,
|
||||||
|
RevisionDate = performedAt.UtcDateTime
|
||||||
|
},
|
||||||
|
Collections = o.AccessibleCollections,
|
||||||
|
Groups = o.Groups
|
||||||
|
};
|
||||||
|
}
|
@ -10,12 +10,13 @@ namespace Bit.Core.AdminConsole.OrganizationFeatures.OrganizationUsers.InviteUse
|
|||||||
public class OrganizationUserInvite
|
public class OrganizationUserInvite
|
||||||
{
|
{
|
||||||
public string[] Emails { get; private init; } = [];
|
public string[] Emails { get; private init; } = [];
|
||||||
public Guid[] AccessibleCollections { get; private init; } = [];
|
public CollectionAccessSelection[] AccessibleCollections { get; private init; } = [];
|
||||||
public OrganizationUserType Type { get; private init; } = OrganizationUserType.User;
|
public OrganizationUserType Type { get; private init; } = OrganizationUserType.User;
|
||||||
|
|
||||||
public Permissions Permissions { get; private init; } = new();
|
public Permissions Permissions { get; private init; } = new();
|
||||||
public string ExternalId { get; private init; } = string.Empty;
|
public string ExternalId { get; private init; } = string.Empty;
|
||||||
public bool AccessSecretsManager { get; private init; }
|
public bool AccessSecretsManager { get; private init; }
|
||||||
|
public Guid[] Groups { get; private init; } = [];
|
||||||
|
|
||||||
public static OrganizationUserInvite Create(string[] emails,
|
public static OrganizationUserInvite Create(string[] emails,
|
||||||
IEnumerable<CollectionAccessSelection> accessibleCollections,
|
IEnumerable<CollectionAccessSelection> accessibleCollections,
|
||||||
@ -24,26 +25,13 @@ public class OrganizationUserInvite
|
|||||||
string externalId,
|
string externalId,
|
||||||
bool accessSecretsManager)
|
bool accessSecretsManager)
|
||||||
{
|
{
|
||||||
|
ValidateEmailAddresses(emails);
|
||||||
|
|
||||||
if (accessibleCollections?.Any(ValidateCollectionConfiguration) ?? false)
|
if (accessibleCollections?.Any(ValidateCollectionConfiguration) ?? false)
|
||||||
{
|
{
|
||||||
throw new BadRequestException(InvalidCollectionConfigurationErrorMessage);
|
throw new BadRequestException(InvalidCollectionConfigurationErrorMessage);
|
||||||
}
|
}
|
||||||
|
|
||||||
return Create(emails, accessibleCollections?.Select(x => x.Id), type, permissions, externalId, accessSecretsManager);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static OrganizationUserInvite Create(OrganizationUserSingleEmailInvite invite, string externalId) =>
|
|
||||||
Create([invite.Email],
|
|
||||||
invite.AccessibleCollections,
|
|
||||||
invite.Type,
|
|
||||||
invite.Permissions,
|
|
||||||
externalId,
|
|
||||||
invite.AccessSecretsManager);
|
|
||||||
|
|
||||||
private static OrganizationUserInvite Create(string[] emails, IEnumerable<Guid> accessibleCollections, OrganizationUserType type, Permissions permissions, string externalId, bool accessSecretsManager)
|
|
||||||
{
|
|
||||||
ValidateEmailAddresses(emails);
|
|
||||||
|
|
||||||
return new OrganizationUserInvite
|
return new OrganizationUserInvite
|
||||||
{
|
{
|
||||||
Emails = emails,
|
Emails = emails,
|
||||||
@ -55,6 +43,14 @@ public class OrganizationUserInvite
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static OrganizationUserInvite Create(OrganizationUserSingleEmailInvite invite, string externalId) =>
|
||||||
|
Create([invite.Email],
|
||||||
|
invite.AccessibleCollections,
|
||||||
|
invite.Type,
|
||||||
|
invite.Permissions,
|
||||||
|
externalId,
|
||||||
|
invite.AccessSecretsManager);
|
||||||
|
|
||||||
private static void ValidateEmailAddresses(string[] emails)
|
private static void ValidateEmailAddresses(string[] emails)
|
||||||
{
|
{
|
||||||
foreach (var email in emails)
|
foreach (var email in emails)
|
||||||
|
@ -6,13 +6,15 @@ namespace Bit.Core.AdminConsole.OrganizationFeatures.OrganizationUsers.InviteUse
|
|||||||
public class OrganizationUserInviteDto
|
public class OrganizationUserInviteDto
|
||||||
{
|
{
|
||||||
public string Email { get; private init; } = string.Empty;
|
public string Email { get; private init; } = string.Empty;
|
||||||
public Guid[] AccessibleCollections { get; private init; } = [];
|
public CollectionAccessSelection[] AccessibleCollections { get; private init; } = [];
|
||||||
public string ExternalId { get; private init; } = string.Empty;
|
public string ExternalId { get; private init; } = string.Empty;
|
||||||
public Permissions Permissions { get; private init; } = new();
|
public Permissions Permissions { get; private init; } = new();
|
||||||
public OrganizationUserType Type { get; private init; } = OrganizationUserType.User;
|
public OrganizationUserType Type { get; private init; } = OrganizationUserType.User;
|
||||||
public bool AccessSecretsManager { get; private init; }
|
public bool AccessSecretsManager { get; private init; }
|
||||||
|
public Guid OrganizationId { get; private init; } = Guid.Empty;
|
||||||
|
public Guid[] Groups { get; private init; } = [];
|
||||||
|
|
||||||
public static OrganizationUserInviteDto Create(string email, OrganizationUserInvite invite)
|
public static OrganizationUserInviteDto Create(string email, OrganizationUserInvite invite, Guid organizationId)
|
||||||
{
|
{
|
||||||
return new OrganizationUserInviteDto
|
return new OrganizationUserInviteDto
|
||||||
{
|
{
|
||||||
@ -21,7 +23,9 @@ public class OrganizationUserInviteDto
|
|||||||
ExternalId = invite.ExternalId,
|
ExternalId = invite.ExternalId,
|
||||||
Type = invite.Type,
|
Type = invite.Type,
|
||||||
Permissions = invite.Permissions,
|
Permissions = invite.Permissions,
|
||||||
AccessSecretsManager = invite.AccessSecretsManager
|
AccessSecretsManager = invite.AccessSecretsManager,
|
||||||
|
OrganizationId = organizationId,
|
||||||
|
Groups = invite.Groups
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -10,7 +10,7 @@ namespace Bit.Core.AdminConsole.OrganizationFeatures.OrganizationUsers.InviteUse
|
|||||||
public class OrganizationUserSingleEmailInvite
|
public class OrganizationUserSingleEmailInvite
|
||||||
{
|
{
|
||||||
public string Email { get; private init; } = string.Empty;
|
public string Email { get; private init; } = string.Empty;
|
||||||
public Guid[] AccessibleCollections { get; private init; } = [];
|
public CollectionAccessSelection[] AccessibleCollections { get; private init; } = [];
|
||||||
public Permissions Permissions { get; private init; } = new();
|
public Permissions Permissions { get; private init; } = new();
|
||||||
public OrganizationUserType Type { get; private init; } = OrganizationUserType.User;
|
public OrganizationUserType Type { get; private init; } = OrganizationUserType.User;
|
||||||
public bool AccessSecretsManager { get; private init; }
|
public bool AccessSecretsManager { get; private init; }
|
||||||
@ -34,7 +34,7 @@ public class OrganizationUserSingleEmailInvite
|
|||||||
return new OrganizationUserSingleEmailInvite
|
return new OrganizationUserSingleEmailInvite
|
||||||
{
|
{
|
||||||
Email = email,
|
Email = email,
|
||||||
AccessibleCollections = accessibleCollections.Select(x => x.Id).ToArray(),
|
AccessibleCollections = accessibleCollections.ToArray(),
|
||||||
Type = type,
|
Type = type,
|
||||||
Permissions = permissions,
|
Permissions = permissions,
|
||||||
AccessSecretsManager = accessSecretsManager
|
AccessSecretsManager = accessSecretsManager
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
using Bit.Core.AdminConsole.Enums;
|
using Bit.Core.AdminConsole.Enums;
|
||||||
|
using Bit.Core.AdminConsole.OrganizationFeatures.OrganizationUsers.InviteUsers.Models;
|
||||||
using Bit.Core.Entities;
|
using Bit.Core.Entities;
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.KeyManagement.UserKey;
|
using Bit.Core.KeyManagement.UserKey;
|
||||||
@ -68,4 +69,6 @@ public interface IOrganizationUserRepository : IRepository<OrganizationUser, Gui
|
|||||||
/// <param name="role">The role to search for</param>
|
/// <param name="role">The role to search for</param>
|
||||||
/// <returns>A list of OrganizationUsersUserDetails with the specified role</returns>
|
/// <returns>A list of OrganizationUsersUserDetails with the specified role</returns>
|
||||||
Task<IEnumerable<OrganizationUserUserDetails>> GetManyDetailsByRoleAsync(Guid organizationId, OrganizationUserType role);
|
Task<IEnumerable<OrganizationUserUserDetails>> GetManyDetailsByRoleAsync(Guid organizationId, OrganizationUserType role);
|
||||||
|
|
||||||
|
Task CreateManyAsync(IEnumerable<CreateOrganizationUser> organizationUserCollection);
|
||||||
}
|
}
|
||||||
|
@ -10,7 +10,20 @@ public class CommandResult(IEnumerable<string> errors)
|
|||||||
public CommandResult() : this(Array.Empty<string>()) { }
|
public CommandResult() : this(Array.Empty<string>()) { }
|
||||||
}
|
}
|
||||||
|
|
||||||
public class CommandResult<T>(T value) : CommandResult
|
public abstract class CommandResult<T> : CommandResult
|
||||||
{
|
{
|
||||||
public T Value { get; set; } = value;
|
|
||||||
|
public T Value { get; set; }
|
||||||
|
}
|
||||||
|
|
||||||
|
public class Success<T> : CommandResult<T>
|
||||||
|
{
|
||||||
|
public Success(T value) => Value = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class Failure<T> : CommandResult<T>
|
||||||
|
{
|
||||||
|
public Failure(string error) => ErrorMessages.Add(error);
|
||||||
|
|
||||||
|
public string ErrorMessage => string.Join(" ", ErrorMessages);
|
||||||
}
|
}
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
using System.Text.Json;
|
using System.Text.Json;
|
||||||
using Bit.Core.AdminConsole.Entities;
|
using Bit.Core.AdminConsole.Entities;
|
||||||
using Bit.Core.AdminConsole.Enums;
|
using Bit.Core.AdminConsole.Enums;
|
||||||
|
using Bit.Core.AdminConsole.OrganizationFeatures.OrganizationUsers.InviteUsers.Models;
|
||||||
using Bit.Core.Entities;
|
using Bit.Core.Entities;
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.KeyManagement.UserKey;
|
using Bit.Core.KeyManagement.UserKey;
|
||||||
@ -580,4 +581,32 @@ public class OrganizationUserRepository : Repository<OrganizationUser, Guid>, IO
|
|||||||
return results.ToList();
|
return results.ToList();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async Task CreateManyAsync(IEnumerable<CreateOrganizationUser> organizationUserCollection)
|
||||||
|
{
|
||||||
|
await using var connection = new SqlConnection(_marsConnectionString);
|
||||||
|
|
||||||
|
await connection.ExecuteAsync(
|
||||||
|
$"[{Schema}].[OrganizationUser_CreateManyWithCollectionsAndGroups]",
|
||||||
|
new
|
||||||
|
{
|
||||||
|
OrganizationUserData = JsonSerializer.Serialize(organizationUserCollection.Select(x => x.User)),
|
||||||
|
CollectionData = JsonSerializer.Serialize(organizationUserCollection
|
||||||
|
.SelectMany(x => x.Collections, (user, collection) => new CollectionUser
|
||||||
|
{
|
||||||
|
CollectionId = collection.Id,
|
||||||
|
OrganizationUserId = user.User.Id,
|
||||||
|
ReadOnly = collection.ReadOnly,
|
||||||
|
HidePasswords = collection.HidePasswords,
|
||||||
|
Manage = collection.Manage
|
||||||
|
})),
|
||||||
|
GroupData = JsonSerializer.Serialize(organizationUserCollection
|
||||||
|
.SelectMany(x => x.Groups, (user, group) => new GroupUser
|
||||||
|
{
|
||||||
|
GroupId = group,
|
||||||
|
OrganizationUserId = user.User.Id
|
||||||
|
}))
|
||||||
|
},
|
||||||
|
commandType: CommandType.StoredProcedure);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
using AutoMapper;
|
using AutoMapper;
|
||||||
using Bit.Core.AdminConsole.Enums;
|
using Bit.Core.AdminConsole.Enums;
|
||||||
|
using Bit.Core.AdminConsole.OrganizationFeatures.OrganizationUsers.InviteUsers.Models;
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
using Bit.Core.KeyManagement.UserKey;
|
using Bit.Core.KeyManagement.UserKey;
|
||||||
using Bit.Core.Models.Data;
|
using Bit.Core.Models.Data;
|
||||||
@ -754,4 +755,28 @@ public class OrganizationUserRepository : Repository<Core.Entities.OrganizationU
|
|||||||
return await query.ToListAsync();
|
return await query.ToListAsync();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async Task CreateManyAsync(IEnumerable<CreateOrganizationUser> organizationUserCollection)
|
||||||
|
{
|
||||||
|
using var scope = ServiceScopeFactory.CreateScope();
|
||||||
|
|
||||||
|
await using var dbContext = GetDatabaseContext(scope);
|
||||||
|
|
||||||
|
dbContext.OrganizationUsers.AddRange(Mapper.Map<List<OrganizationUser>>(organizationUserCollection.Select(x => x.User)));
|
||||||
|
dbContext.CollectionUsers.AddRange(organizationUserCollection.SelectMany(x => x.Collections, (user, collection) => new CollectionUser
|
||||||
|
{
|
||||||
|
CollectionId = collection.Id,
|
||||||
|
HidePasswords = collection.HidePasswords,
|
||||||
|
OrganizationUserId = user.User.Id,
|
||||||
|
Manage = collection.Manage,
|
||||||
|
ReadOnly = collection.ReadOnly
|
||||||
|
}));
|
||||||
|
dbContext.GroupUsers.AddRange(organizationUserCollection.SelectMany(x => x.Groups, (user, group) => new GroupUser
|
||||||
|
{
|
||||||
|
GroupId = group,
|
||||||
|
OrganizationUserId = user.User.Id
|
||||||
|
}));
|
||||||
|
|
||||||
|
await dbContext.SaveChangesAsync();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,97 @@
|
|||||||
|
CREATE PROCEDURE [dbo].[OrganizationUser_CreateManyWithCollectionsAndGroups]
|
||||||
|
@organizationUserData NVARCHAR(MAX),
|
||||||
|
@collectionData NVARCHAR(MAX),
|
||||||
|
@groupData NVARCHAR(MAX)
|
||||||
|
AS
|
||||||
|
BEGIN
|
||||||
|
SET NOCOUNT ON
|
||||||
|
|
||||||
|
INSERT INTO [dbo].[OrganizationUser]
|
||||||
|
(
|
||||||
|
[Id],
|
||||||
|
[OrganizationId],
|
||||||
|
[UserId],
|
||||||
|
[Email],
|
||||||
|
[Key],
|
||||||
|
[Status],
|
||||||
|
[Type],
|
||||||
|
[ExternalId],
|
||||||
|
[CreationDate],
|
||||||
|
[RevisionDate],
|
||||||
|
[Permissions],
|
||||||
|
[ResetPasswordKey],
|
||||||
|
[AccessSecretsManager]
|
||||||
|
)
|
||||||
|
SELECT
|
||||||
|
OUI.[Id],
|
||||||
|
OUI.[OrganizationId],
|
||||||
|
OUI.[UserId],
|
||||||
|
OUI.[Email],
|
||||||
|
OUI.[Key],
|
||||||
|
OUI.[Status],
|
||||||
|
OUI.[Type],
|
||||||
|
OUI.[ExternalId],
|
||||||
|
OUI.[CreationDate],
|
||||||
|
OUI.[RevisionDate],
|
||||||
|
OUI.[Permissions],
|
||||||
|
OUI.[ResetPasswordKey],
|
||||||
|
OUI.[AccessSecretsManager]
|
||||||
|
FROM
|
||||||
|
OPENJSON(@organizationUserData)
|
||||||
|
WITH (
|
||||||
|
[Id] UNIQUEIDENTIFIER '$.Id',
|
||||||
|
[OrganizationId] UNIQUEIDENTIFIER '$.OrganizationId',
|
||||||
|
[UserId] UNIQUEIDENTIFIER '$.UserId',
|
||||||
|
[Email] NVARCHAR(256) '$.Email',
|
||||||
|
[Key] VARCHAR(MAX) '$.Key',
|
||||||
|
[Status] SMALLINT '$.Status',
|
||||||
|
[Type] TINYINT '$.Type',
|
||||||
|
[ExternalId] NVARCHAR(300) '$.ExternalId',
|
||||||
|
[CreationDate] DATETIME2(7) '$.CreationDate',
|
||||||
|
[RevisionDate] DATETIME2(7) '$.RevisionDate',
|
||||||
|
[Permissions] NVARCHAR (MAX) '$.Permissions',
|
||||||
|
[ResetPasswordKey] VARCHAR (MAX) '$.ResetPasswordKey',
|
||||||
|
[AccessSecretsManager] BIT '$.AccessSecretsManager'
|
||||||
|
) OUI
|
||||||
|
|
||||||
|
INSERT INTO [dbo].[GroupUser]
|
||||||
|
(
|
||||||
|
[OrganizationUserId],
|
||||||
|
[GroupId]
|
||||||
|
)
|
||||||
|
SELECT
|
||||||
|
OUG.OrganizationUserId,
|
||||||
|
OUG.GroupId
|
||||||
|
FROM
|
||||||
|
OPENJSON(@groupData)
|
||||||
|
WITH(
|
||||||
|
[OrganizationUserId] UNIQUEIDENTIFIER '$.OrganizationUserId',
|
||||||
|
[GroupId] UNIQUEIDENTIFIER '$.GroupId'
|
||||||
|
) OUG
|
||||||
|
|
||||||
|
INSERT INTO [dbo].[CollectionUser]
|
||||||
|
(
|
||||||
|
[CollectionId],
|
||||||
|
[OrganizationUserId],
|
||||||
|
[ReadOnly],
|
||||||
|
[HidePasswords],
|
||||||
|
[Manage]
|
||||||
|
)
|
||||||
|
SELECT
|
||||||
|
OUC.[CollectionId],
|
||||||
|
OUC.[OrganizationUserId],
|
||||||
|
OUC.[ReadOnly],
|
||||||
|
OUC.[HidePasswords],
|
||||||
|
OUC.[Manage]
|
||||||
|
FROM
|
||||||
|
OPENJSON(@collectionData)
|
||||||
|
WITH(
|
||||||
|
[CollectionId] UNIQUEIDENTIFIER '$.CollectionId',
|
||||||
|
[OrganizationUserId] UNIQUEIDENTIFIER '$.OrganizationUserId',
|
||||||
|
[ReadOnly] BIT '$.ReadOnly',
|
||||||
|
[HidePasswords] BIT '$.HidePasswords',
|
||||||
|
[Manage] BIT '$.Manage'
|
||||||
|
) OUC
|
||||||
|
END
|
||||||
|
go
|
||||||
|
|
@ -49,6 +49,6 @@ public class InviteOrganizationUserRequestTests
|
|||||||
|
|
||||||
Assert.NotNull(invite);
|
Assert.NotNull(invite);
|
||||||
Assert.Equal(validEmail, invite.Email);
|
Assert.Equal(validEmail, invite.Email);
|
||||||
Assert.Contains(validCollectionConfiguration.Id, invite.AccessibleCollections);
|
Assert.Contains(validCollectionConfiguration, invite.AccessibleCollections);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -50,6 +50,6 @@ public class InviteOrganizationUsersRequestTests
|
|||||||
|
|
||||||
Assert.NotNull(invite);
|
Assert.NotNull(invite);
|
||||||
Assert.Contains(validEmail, invite.Emails);
|
Assert.Contains(validEmail, invite.Emails);
|
||||||
Assert.Contains(validCollectionConfiguration.Id, invite.AccessibleCollections);
|
Assert.Contains(validCollectionConfiguration, invite.AccessibleCollections);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -85,7 +85,7 @@ public class SecretsManagerInviteUserValidationTests
|
|||||||
|
|
||||||
var request = new InviteUserOrganizationValidationRequest
|
var request = new InviteUserOrganizationValidationRequest
|
||||||
{
|
{
|
||||||
Invites = [OrganizationUserInviteDto.Create("email@test.com", OrganizationUserInvite.Create(["email@test.com"], [], OrganizationUserType.User, new Permissions(), string.Empty, true))],
|
Invites = [OrganizationUserInviteDto.Create("email@test.com", OrganizationUserInvite.Create(["email@test.com"], [], OrganizationUserType.User, new Permissions(), string.Empty, true), organization.Id)],
|
||||||
Organization = organizationDto,
|
Organization = organizationDto,
|
||||||
PerformedBy = Guid.Empty,
|
PerformedBy = Guid.Empty,
|
||||||
PerformedAt = default,
|
PerformedAt = default,
|
||||||
@ -116,7 +116,7 @@ public class SecretsManagerInviteUserValidationTests
|
|||||||
|
|
||||||
var request = new InviteUserOrganizationValidationRequest
|
var request = new InviteUserOrganizationValidationRequest
|
||||||
{
|
{
|
||||||
Invites = [OrganizationUserInviteDto.Create("email@test.com", OrganizationUserInvite.Create(["email@test.com"], [], OrganizationUserType.User, new Permissions(), string.Empty, true))],
|
Invites = [OrganizationUserInviteDto.Create("email@test.com", OrganizationUserInvite.Create(["email@test.com"], [], OrganizationUserType.User, new Permissions(), string.Empty, true), organization.Id)],
|
||||||
Organization = organizationDto,
|
Organization = organizationDto,
|
||||||
PerformedBy = Guid.Empty,
|
PerformedBy = Guid.Empty,
|
||||||
PerformedAt = default,
|
PerformedAt = default,
|
||||||
|
@ -1,7 +1,11 @@
|
|||||||
using Bit.Core.AdminConsole.Entities;
|
using Bit.Core.AdminConsole.Entities;
|
||||||
|
using Bit.Core.AdminConsole.OrganizationFeatures.OrganizationUsers.InviteUsers.Models;
|
||||||
|
using Bit.Core.AdminConsole.Repositories;
|
||||||
using Bit.Core.Entities;
|
using Bit.Core.Entities;
|
||||||
using Bit.Core.Enums;
|
using Bit.Core.Enums;
|
||||||
|
using Bit.Core.Models.Data;
|
||||||
using Bit.Core.Repositories;
|
using Bit.Core.Repositories;
|
||||||
|
using Bit.Core.Utilities;
|
||||||
using Xunit;
|
using Xunit;
|
||||||
|
|
||||||
namespace Bit.Infrastructure.IntegrationTest.Repositories;
|
namespace Bit.Infrastructure.IntegrationTest.Repositories;
|
||||||
@ -354,4 +358,73 @@ public class OrganizationUserRepositoryTests
|
|||||||
Assert.Single(responseModel);
|
Assert.Single(responseModel);
|
||||||
Assert.Equal(orgUser1.Id, responseModel.Single().Id);
|
Assert.Equal(orgUser1.Id, responseModel.Single().Id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[DatabaseTheory, DatabaseData]
|
||||||
|
public async Task CreateManyAsync_WithCollectionAndGroup_SaveSuccessfully(
|
||||||
|
IOrganizationUserRepository organizationUserRepository,
|
||||||
|
IOrganizationRepository organizationRepository,
|
||||||
|
ICollectionRepository collectionRepository,
|
||||||
|
IGroupRepository groupRepository)
|
||||||
|
{
|
||||||
|
var requestTime = DateTime.UtcNow;
|
||||||
|
|
||||||
|
var organization = await organizationRepository.CreateAsync(new Organization
|
||||||
|
{
|
||||||
|
Name = "Test Org",
|
||||||
|
BillingEmail = "billing@test.com", // TODO: EF does not enfore this being NOT NULL
|
||||||
|
Plan = "Test", // TODO: EF does not enforce this being NOT NULl,
|
||||||
|
CreationDate = requestTime
|
||||||
|
});
|
||||||
|
|
||||||
|
var collection = await collectionRepository.CreateAsync(new Collection
|
||||||
|
{
|
||||||
|
Id = CoreHelpers.GenerateComb(),
|
||||||
|
OrganizationId = organization.Id,
|
||||||
|
Name = "Test Collection",
|
||||||
|
ExternalId = "external-collection-1",
|
||||||
|
CreationDate = requestTime,
|
||||||
|
RevisionDate = requestTime
|
||||||
|
});
|
||||||
|
|
||||||
|
var group = await groupRepository.CreateAsync(new Group
|
||||||
|
{
|
||||||
|
Id = CoreHelpers.GenerateComb(),
|
||||||
|
OrganizationId = organization.Id,
|
||||||
|
Name = "Test Group",
|
||||||
|
ExternalId = "external-group-1"
|
||||||
|
});
|
||||||
|
|
||||||
|
var orgUserCollection = new List<CreateOrganizationUser>
|
||||||
|
{
|
||||||
|
new CreateOrganizationUser
|
||||||
|
{
|
||||||
|
User = new OrganizationUser
|
||||||
|
{
|
||||||
|
Id = CoreHelpers.GenerateComb(),
|
||||||
|
OrganizationId = organization.Id,
|
||||||
|
Email = "test-user@test.com",
|
||||||
|
Status = OrganizationUserStatusType.Invited,
|
||||||
|
Type = OrganizationUserType.Owner,
|
||||||
|
ExternalId = "externalid-1",
|
||||||
|
Permissions = CoreHelpers.ClassToJsonData(new Permissions()),
|
||||||
|
AccessSecretsManager = false
|
||||||
|
},
|
||||||
|
Collections =
|
||||||
|
[
|
||||||
|
new CollectionAccessSelection
|
||||||
|
{
|
||||||
|
Id = collection.Id,
|
||||||
|
ReadOnly = true,
|
||||||
|
HidePasswords = false,
|
||||||
|
Manage = false
|
||||||
|
}
|
||||||
|
],
|
||||||
|
Groups = [group.Id]
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
await organizationUserRepository.CreateManyAsync(orgUserCollection);
|
||||||
|
|
||||||
|
var orgUser = await organizationUserRepository.GetDetailsByIdAsync(orgUserCollection.First().User.Id);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,97 @@
|
|||||||
|
CREATE OR ALTER PROCEDURE [dbo].[OrganizationUser_CreateManyWithCollectionsAndGroups]
|
||||||
|
@organizationUserData NVARCHAR(MAX),
|
||||||
|
@collectionData NVARCHAR(MAX),
|
||||||
|
@groupData NVARCHAR(MAX)
|
||||||
|
AS
|
||||||
|
BEGIN
|
||||||
|
SET NOCOUNT ON
|
||||||
|
|
||||||
|
INSERT INTO [dbo].[OrganizationUser]
|
||||||
|
(
|
||||||
|
[Id],
|
||||||
|
[OrganizationId],
|
||||||
|
[UserId],
|
||||||
|
[Email],
|
||||||
|
[Key],
|
||||||
|
[Status],
|
||||||
|
[Type],
|
||||||
|
[ExternalId],
|
||||||
|
[CreationDate],
|
||||||
|
[RevisionDate],
|
||||||
|
[Permissions],
|
||||||
|
[ResetPasswordKey],
|
||||||
|
[AccessSecretsManager]
|
||||||
|
)
|
||||||
|
SELECT
|
||||||
|
OUI.[Id],
|
||||||
|
OUI.[OrganizationId],
|
||||||
|
OUI.[UserId],
|
||||||
|
OUI.[Email],
|
||||||
|
OUI.[Key],
|
||||||
|
OUI.[Status],
|
||||||
|
OUI.[Type],
|
||||||
|
OUI.[ExternalId],
|
||||||
|
OUI.[CreationDate],
|
||||||
|
OUI.[RevisionDate],
|
||||||
|
OUI.[Permissions],
|
||||||
|
OUI.[ResetPasswordKey],
|
||||||
|
OUI.[AccessSecretsManager]
|
||||||
|
FROM
|
||||||
|
OPENJSON(@organizationUserData)
|
||||||
|
WITH (
|
||||||
|
[Id] UNIQUEIDENTIFIER '$.Id',
|
||||||
|
[OrganizationId] UNIQUEIDENTIFIER '$.OrganizationId',
|
||||||
|
[UserId] UNIQUEIDENTIFIER '$.UserId',
|
||||||
|
[Email] NVARCHAR(256) '$.Email',
|
||||||
|
[Key] VARCHAR(MAX) '$.Key',
|
||||||
|
[Status] SMALLINT '$.Status',
|
||||||
|
[Type] TINYINT '$.Type',
|
||||||
|
[ExternalId] NVARCHAR(300) '$.ExternalId',
|
||||||
|
[CreationDate] DATETIME2(7) '$.CreationDate',
|
||||||
|
[RevisionDate] DATETIME2(7) '$.RevisionDate',
|
||||||
|
[Permissions] NVARCHAR (MAX) '$.Permissions',
|
||||||
|
[ResetPasswordKey] VARCHAR (MAX) '$.ResetPasswordKey',
|
||||||
|
[AccessSecretsManager] BIT '$.AccessSecretsManager'
|
||||||
|
) OUI
|
||||||
|
|
||||||
|
INSERT INTO [dbo].[GroupUser]
|
||||||
|
(
|
||||||
|
[OrganizationUserId],
|
||||||
|
[GroupId]
|
||||||
|
)
|
||||||
|
SELECT
|
||||||
|
OUG.OrganizationUserId,
|
||||||
|
OUG.GroupId
|
||||||
|
FROM
|
||||||
|
OPENJSON(@groupData)
|
||||||
|
WITH(
|
||||||
|
[OrganizationUserId] UNIQUEIDENTIFIER '$.OrganizationUserId',
|
||||||
|
[GroupId] UNIQUEIDENTIFIER '$.GroupId'
|
||||||
|
) OUG
|
||||||
|
|
||||||
|
INSERT INTO [dbo].[CollectionUser]
|
||||||
|
(
|
||||||
|
[CollectionId],
|
||||||
|
[OrganizationUserId],
|
||||||
|
[ReadOnly],
|
||||||
|
[HidePasswords],
|
||||||
|
[Manage]
|
||||||
|
)
|
||||||
|
SELECT
|
||||||
|
OUC.[CollectionId],
|
||||||
|
OUC.[OrganizationUserId],
|
||||||
|
OUC.[ReadOnly],
|
||||||
|
OUC.[HidePasswords],
|
||||||
|
OUC.[Manage]
|
||||||
|
FROM
|
||||||
|
OPENJSON(@collectionData)
|
||||||
|
WITH(
|
||||||
|
[CollectionId] UNIQUEIDENTIFIER '$.CollectionId',
|
||||||
|
[OrganizationUserId] UNIQUEIDENTIFIER '$.OrganizationUserId',
|
||||||
|
[ReadOnly] BIT '$.ReadOnly',
|
||||||
|
[HidePasswords] BIT '$.HidePasswords',
|
||||||
|
[Manage] BIT '$.Manage'
|
||||||
|
) OUC
|
||||||
|
END
|
||||||
|
go
|
||||||
|
|
Loading…
x
Reference in New Issue
Block a user