2019-02-10 04:01:45 +01:00
|
|
|
|
using System;
|
2021-06-01 17:13:08 +02:00
|
|
|
|
using System.Collections.Generic;
|
2019-02-10 04:01:45 +01:00
|
|
|
|
using System.Threading.Tasks;
|
2021-06-01 17:13:08 +02:00
|
|
|
|
using Bit.Core.Enums;
|
|
|
|
|
using Bit.Core.Exceptions;
|
|
|
|
|
using Bit.Core.Models.Data;
|
|
|
|
|
using Bit.Core.Models.Table;
|
2019-02-10 04:01:45 +01:00
|
|
|
|
using Bit.Core.Repositories;
|
|
|
|
|
using Bit.Core.Services;
|
Postgres & MySql Support For Self-Hosted Installations (#1386)
* EF Database Support Init (#1221)
* scaffolding for ef support
* deleted old postgres repos
* added tables to oncreate
* updated all the things to .NET 5
* Addition to #1221: Migrated DockerFiles from dotnet/3.1 to 5.0 (#1223)
* Migrated DockerFiles from dotnet/3.1 to 5.0
* Migrated SSO/Dockerfile from dotnet 3.1 to 5.0
Co-authored-by: Daniel James Smith <djsmith85@users.noreply.github.com>
* EFDatabaseSupport: Updated links and description in README.md and SETUP.md (#1232)
* Updated requirements in README.md
* Updated link to documentation of app-secrets
* upgraded dotnet version to 5.0
* Ef database support implementation examples (#1265)
* mostly finished testing the user repo
* finished testing user repo
* finished org, user, ssoconfig, and ssouser ef implementations
* removed unused prop
* fixed a sql file
* fixed a spacing issue
* fixed a spacing issue
* removed extra database creation
* refactoring
* MsSql => SqlServer
* refactoring
* code review fixes
* build fix
* code review
* continued attempts to fix the the build
* skipped another test
* finished all create test
* initial pass at several repos
* continued building out repos
* initial pass at several repos
* initial pass at device repo
* initial pass at collection repo
* initial run of all Entity Framework implementations
* signup, signin, create/edit ciphers works
* sync working
* all web vault pages seem to load with 100% 200s
* bulkcopy, folders, and favorites
* group and collection management
* sso, groups, emergency access, send
* get basic creates matching on all repos
* got everything building again post merge
* removed some IDE config files
* cleanup
* no more notimplemented methods in the cipher repo
* no more not implementeds everywhere
* cleaned up schema/navigation properties and fixed tests
* removed a sql comment that was written in c# style
* fixed build issues from merge
* removed unsupported db providers
* formatting
* code review refactors
* naming cleanup for queries
* added provider methods
* cipher repo cleanup
* implemented several missing procedures from the EF implementation surround account revision dates, keys, and storage
* fixed the build
* added a null check
* consolidated some cipher repo methods
* formatting fix
* cleaned up indentation of queries
* removed .idea file
* generated postgres migrations
* added mysql migrations
* formatting
* Bug Fixes & Formatting
* Formatting
* fixed a bug with bulk import when using MySql
* code review fixes
* fixed the build
* implemented new methods
* formatting
* fixed the build
* cleaned up select statements in ef queries
* formatting
* formatting
* formatting
Co-authored-by: Daniel James Smith <djsmith85@users.noreply.github.com>
2021-07-08 18:35:48 +02:00
|
|
|
|
using Bit.Core.Test.AutoFixture.CollectionFixtures;
|
2021-10-29 21:05:45 +02:00
|
|
|
|
using Bit.Test.Common.AutoFixture;
|
|
|
|
|
using Bit.Test.Common.AutoFixture.Attributes;
|
2019-02-10 04:01:45 +01:00
|
|
|
|
using NSubstitute;
|
2021-06-01 17:13:08 +02:00
|
|
|
|
using Xunit;
|
2019-02-10 04:01:45 +01:00
|
|
|
|
|
|
|
|
|
namespace Bit.Core.Test.Services
|
|
|
|
|
{
|
|
|
|
|
public class CollectionServiceTest
|
|
|
|
|
{
|
2021-06-01 17:13:08 +02:00
|
|
|
|
[Theory, CollectionAutoData]
|
|
|
|
|
public async Task SaveAsync_DefaultId_CreatesCollectionInTheRepository(Collection collection, Organization organization, SutProvider<CollectionService> sutProvider)
|
2019-02-10 04:01:45 +01:00
|
|
|
|
{
|
2021-06-01 17:13:08 +02:00
|
|
|
|
collection.Id = default;
|
|
|
|
|
sutProvider.GetDependency<IOrganizationRepository>().GetByIdAsync(organization.Id).Returns(organization);
|
|
|
|
|
var utcNow = DateTime.UtcNow;
|
|
|
|
|
|
|
|
|
|
await sutProvider.Sut.SaveAsync(collection);
|
|
|
|
|
|
|
|
|
|
await sutProvider.GetDependency<ICollectionRepository>().Received().CreateAsync(collection);
|
|
|
|
|
await sutProvider.GetDependency<IEventService>().Received()
|
|
|
|
|
.LogCollectionEventAsync(collection, EventType.Collection_Created);
|
|
|
|
|
Assert.True(collection.CreationDate - utcNow < TimeSpan.FromSeconds(1));
|
|
|
|
|
Assert.True(collection.RevisionDate - utcNow < TimeSpan.FromSeconds(1));
|
2019-02-10 04:01:45 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-06-01 17:13:08 +02:00
|
|
|
|
[Theory, CollectionAutoData]
|
|
|
|
|
public async Task SaveAsync_DefaultIdWithGroups_CreateCollectionWithGroupsInRepository(Collection collection,
|
|
|
|
|
IEnumerable<SelectionReadOnly> groups, Organization organization, OrganizationUser organizationUser,
|
|
|
|
|
SutProvider<CollectionService> sutProvider)
|
2019-02-10 04:01:45 +01:00
|
|
|
|
{
|
2021-06-01 17:13:08 +02:00
|
|
|
|
collection.Id = default;
|
|
|
|
|
organization.UseGroups = true;
|
|
|
|
|
sutProvider.GetDependency<IOrganizationRepository>().GetByIdAsync(organization.Id).Returns(organization);
|
|
|
|
|
var utcNow = DateTime.UtcNow;
|
|
|
|
|
|
|
|
|
|
await sutProvider.Sut.SaveAsync(collection, groups);
|
|
|
|
|
|
|
|
|
|
await sutProvider.GetDependency<ICollectionRepository>().Received().CreateAsync(collection, groups);
|
|
|
|
|
await sutProvider.GetDependency<IEventService>().Received()
|
|
|
|
|
.LogCollectionEventAsync(collection, EventType.Collection_Created);
|
|
|
|
|
Assert.True(collection.CreationDate - utcNow < TimeSpan.FromSeconds(1));
|
|
|
|
|
Assert.True(collection.RevisionDate - utcNow < TimeSpan.FromSeconds(1));
|
|
|
|
|
}
|
2019-02-10 04:01:45 +01:00
|
|
|
|
|
2021-06-01 17:13:08 +02:00
|
|
|
|
[Theory, CollectionAutoData]
|
|
|
|
|
public async Task SaveAsync_NonDefaultId_ReplacesCollectionInRepository(Collection collection, Organization organization, SutProvider<CollectionService> sutProvider)
|
|
|
|
|
{
|
|
|
|
|
var creationDate = collection.CreationDate;
|
|
|
|
|
sutProvider.GetDependency<IOrganizationRepository>().GetByIdAsync(organization.Id).Returns(organization);
|
|
|
|
|
var utcNow = DateTime.UtcNow;
|
2019-02-10 04:01:45 +01:00
|
|
|
|
|
2021-06-01 17:13:08 +02:00
|
|
|
|
await sutProvider.Sut.SaveAsync(collection);
|
2019-02-10 04:01:45 +01:00
|
|
|
|
|
2021-06-01 17:13:08 +02:00
|
|
|
|
await sutProvider.GetDependency<ICollectionRepository>().Received().ReplaceAsync(collection);
|
|
|
|
|
await sutProvider.GetDependency<IEventService>().Received()
|
|
|
|
|
.LogCollectionEventAsync(collection, EventType.Collection_Updated);
|
|
|
|
|
Assert.Equal(collection.CreationDate, creationDate);
|
|
|
|
|
Assert.True(collection.RevisionDate - utcNow < TimeSpan.FromSeconds(1));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Theory, CollectionAutoData]
|
|
|
|
|
public async Task SaveAsync_OrganizationNotUseGroup_CreateCollectionWithoutGroupsInRepository(Collection collection, IEnumerable<SelectionReadOnly> groups,
|
|
|
|
|
Organization organization, OrganizationUser organizationUser, SutProvider<CollectionService> sutProvider)
|
|
|
|
|
{
|
|
|
|
|
collection.Id = default;
|
|
|
|
|
sutProvider.GetDependency<IOrganizationRepository>().GetByIdAsync(organization.Id).Returns(organization);
|
|
|
|
|
var utcNow = DateTime.UtcNow;
|
2019-02-10 04:01:45 +01:00
|
|
|
|
|
2021-06-01 17:13:08 +02:00
|
|
|
|
await sutProvider.Sut.SaveAsync(collection, groups);
|
|
|
|
|
|
|
|
|
|
await sutProvider.GetDependency<ICollectionRepository>().Received().CreateAsync(collection);
|
|
|
|
|
await sutProvider.GetDependency<IEventService>().Received()
|
|
|
|
|
.LogCollectionEventAsync(collection, EventType.Collection_Created);
|
|
|
|
|
Assert.True(collection.CreationDate - utcNow < TimeSpan.FromSeconds(1));
|
|
|
|
|
Assert.True(collection.RevisionDate - utcNow < TimeSpan.FromSeconds(1));
|
2019-02-10 04:01:45 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-06-01 17:13:08 +02:00
|
|
|
|
[Theory, CollectionAutoData]
|
|
|
|
|
public async Task SaveAsync_DefaultIdWithUserId_UpdateUserInCollectionRepository(Collection collection,
|
|
|
|
|
Organization organization, OrganizationUser organizationUser, SutProvider<CollectionService> sutProvider)
|
2019-02-10 04:01:45 +01:00
|
|
|
|
{
|
2021-06-01 17:13:08 +02:00
|
|
|
|
collection.Id = default;
|
|
|
|
|
organizationUser.Status = OrganizationUserStatusType.Confirmed;
|
|
|
|
|
sutProvider.GetDependency<IOrganizationRepository>().GetByIdAsync(organization.Id).Returns(organization);
|
|
|
|
|
sutProvider.GetDependency<IOrganizationUserRepository>().GetByOrganizationAsync(organization.Id, organizationUser.Id)
|
|
|
|
|
.Returns(organizationUser);
|
|
|
|
|
var utcNow = DateTime.UtcNow;
|
|
|
|
|
|
|
|
|
|
await sutProvider.Sut.SaveAsync(collection, null, organizationUser.Id);
|
|
|
|
|
|
|
|
|
|
await sutProvider.GetDependency<ICollectionRepository>().Received().CreateAsync(collection);
|
|
|
|
|
await sutProvider.GetDependency<IOrganizationUserRepository>().Received()
|
|
|
|
|
.GetByOrganizationAsync(organization.Id, organizationUser.Id);
|
|
|
|
|
await sutProvider.GetDependency<ICollectionRepository>().Received().UpdateUsersAsync(collection.Id, Arg.Any<List<SelectionReadOnly>>());
|
|
|
|
|
await sutProvider.GetDependency<IEventService>().Received()
|
|
|
|
|
.LogCollectionEventAsync(collection, EventType.Collection_Created);
|
|
|
|
|
Assert.True(collection.CreationDate - utcNow < TimeSpan.FromSeconds(1));
|
|
|
|
|
Assert.True(collection.RevisionDate - utcNow < TimeSpan.FromSeconds(1));
|
2019-02-10 04:01:45 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-06-01 17:13:08 +02:00
|
|
|
|
[Theory, CustomAutoData(typeof(SutProviderCustomization))]
|
|
|
|
|
public async Task SaveAsync_NonExistingOrganizationId_ThrowsBadRequest(Collection collection, SutProvider<CollectionService> sutProvider)
|
2019-02-10 04:01:45 +01:00
|
|
|
|
{
|
2021-06-01 17:13:08 +02:00
|
|
|
|
var ex = await Assert.ThrowsAsync<BadRequestException>(() => sutProvider.Sut.SaveAsync(collection));
|
|
|
|
|
Assert.Contains("Organization not found", ex.Message);
|
|
|
|
|
await sutProvider.GetDependency<ICollectionRepository>().DidNotReceiveWithAnyArgs().CreateAsync(default);
|
|
|
|
|
await sutProvider.GetDependency<ICollectionRepository>().DidNotReceiveWithAnyArgs().CreateAsync(default, default);
|
|
|
|
|
await sutProvider.GetDependency<ICollectionRepository>().DidNotReceiveWithAnyArgs().ReplaceAsync(default);
|
|
|
|
|
await sutProvider.GetDependency<IEventService>().DidNotReceiveWithAnyArgs().LogCollectionEventAsync(default, default);
|
2019-02-10 04:01:45 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-06-01 17:13:08 +02:00
|
|
|
|
[Theory, CollectionAutoData]
|
|
|
|
|
public async Task SaveAsync_ExceedsOrganizationMaxCollections_ThrowsBadRequest(Collection collection, Collection collection1, Collection collection2, Organization organization, SutProvider<CollectionService> sutProvider)
|
2019-02-10 04:01:45 +01:00
|
|
|
|
{
|
2021-06-01 17:13:08 +02:00
|
|
|
|
collection.Id = default;
|
|
|
|
|
sutProvider.GetDependency<IOrganizationRepository>().GetByIdAsync(organization.Id).Returns(organization);
|
|
|
|
|
sutProvider.GetDependency<ICollectionRepository>().GetCountByOrganizationIdAsync(organization.Id)
|
|
|
|
|
.Returns(organization.MaxCollections.Value);
|
|
|
|
|
|
|
|
|
|
var ex = await Assert.ThrowsAsync<BadRequestException>(() => sutProvider.Sut.SaveAsync(collection));
|
|
|
|
|
Assert.Equal($@"You have reached the maximum number of collections ({organization.MaxCollections.Value}) for this organization.", ex.Message);
|
|
|
|
|
await sutProvider.GetDependency<ICollectionRepository>().DidNotReceiveWithAnyArgs().CreateAsync(default);
|
|
|
|
|
await sutProvider.GetDependency<ICollectionRepository>().DidNotReceiveWithAnyArgs().CreateAsync(default, default);
|
|
|
|
|
await sutProvider.GetDependency<ICollectionRepository>().DidNotReceiveWithAnyArgs().ReplaceAsync(default);
|
|
|
|
|
await sutProvider.GetDependency<IEventService>().DidNotReceiveWithAnyArgs().LogCollectionEventAsync(default, default);
|
2019-02-10 04:01:45 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-06-01 17:13:08 +02:00
|
|
|
|
[Theory, CollectionAutoData]
|
|
|
|
|
public async Task DeleteUserAsync_DeletesValidUserWhoBelongsToCollection(Collection collection,
|
|
|
|
|
Organization organization, OrganizationUser organizationUser, SutProvider<CollectionService> sutProvider)
|
2019-02-10 04:01:45 +01:00
|
|
|
|
{
|
2021-06-01 17:13:08 +02:00
|
|
|
|
collection.OrganizationId = organization.Id;
|
|
|
|
|
organizationUser.OrganizationId = organization.Id;
|
|
|
|
|
sutProvider.GetDependency<IOrganizationRepository>().GetByIdAsync(organization.Id).Returns(organization);
|
|
|
|
|
sutProvider.GetDependency<IOrganizationUserRepository>().GetByIdAsync(organizationUser.Id)
|
|
|
|
|
.Returns(organizationUser);
|
|
|
|
|
|
|
|
|
|
await sutProvider.Sut.DeleteUserAsync(collection, organizationUser.Id);
|
|
|
|
|
|
|
|
|
|
await sutProvider.GetDependency<ICollectionRepository>().Received()
|
|
|
|
|
.DeleteUserAsync(collection.Id, organizationUser.Id);
|
|
|
|
|
await sutProvider.GetDependency<IEventService>().Received().LogOrganizationUserEventAsync(organizationUser, EventType.OrganizationUser_Updated);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[Theory, CollectionAutoData]
|
|
|
|
|
public async Task DeleteUserAsync_InvalidUser_ThrowsNotFound(Collection collection, Organization organization,
|
|
|
|
|
OrganizationUser organizationUser, SutProvider<CollectionService> sutProvider)
|
|
|
|
|
{
|
|
|
|
|
collection.OrganizationId = organization.Id;
|
|
|
|
|
sutProvider.GetDependency<IOrganizationRepository>().GetByIdAsync(organization.Id).Returns(organization);
|
|
|
|
|
sutProvider.GetDependency<IOrganizationUserRepository>().GetByIdAsync(organizationUser.Id)
|
|
|
|
|
.Returns(organizationUser);
|
|
|
|
|
|
|
|
|
|
// user not in organization
|
2019-02-10 04:05:24 +01:00
|
|
|
|
await Assert.ThrowsAsync<NotFoundException>(() =>
|
2021-06-01 17:13:08 +02:00
|
|
|
|
sutProvider.Sut.DeleteUserAsync(collection, organizationUser.Id));
|
|
|
|
|
// invalid user
|
|
|
|
|
await Assert.ThrowsAsync<NotFoundException>(() => sutProvider.Sut.DeleteUserAsync(collection, Guid.NewGuid()));
|
|
|
|
|
await sutProvider.GetDependency<ICollectionRepository>().DidNotReceiveWithAnyArgs().DeleteUserAsync(default, default);
|
|
|
|
|
await sutProvider.GetDependency<IEventService>().DidNotReceiveWithAnyArgs()
|
|
|
|
|
.LogOrganizationUserEventAsync(default, default);
|
2019-02-10 04:01:45 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|