mirror of
https://github.com/bitwarden/server
synced 2025-12-06 00:03:34 +00:00
* Adding SecretVersion table to server * making the names singular not plural for new table * removing migration * fixing migration * Adding indexes for serviceacct and orguserId * indexes for sqllite * fixing migrations * adding indexes to secretVeriosn.sql * tests * removing tests * adding GO * api repository and controller additions for SecretVersion table, as well as tests * test fix sqllite * improvements * removing comments * making files nullable safe * Justin Baurs suggested changes * claude suggestions * Claude fixes * test fixes
308 lines
14 KiB
C#
308 lines
14 KiB
C#
using Bit.Api.SecretsManager.Controllers;
|
|
using Bit.Api.SecretsManager.Models.Request;
|
|
using Bit.Core.Auth.Identity;
|
|
using Bit.Core.Context;
|
|
using Bit.Core.Entities;
|
|
using Bit.Core.Exceptions;
|
|
using Bit.Core.Repositories;
|
|
using Bit.Core.SecretsManager.Entities;
|
|
using Bit.Core.SecretsManager.Repositories;
|
|
using Bit.Core.Services;
|
|
using Bit.Core.Test.SecretsManager.AutoFixture.SecretsFixture;
|
|
using Bit.Test.Common.AutoFixture;
|
|
using Bit.Test.Common.AutoFixture.Attributes;
|
|
using NSubstitute;
|
|
using Xunit;
|
|
|
|
namespace Bit.Api.Test.SecretsManager.Controllers;
|
|
|
|
[ControllerCustomize(typeof(SecretVersionsController))]
|
|
[SutProviderCustomize]
|
|
[SecretCustomize]
|
|
public class SecretVersionsControllerTests
|
|
{
|
|
[Theory]
|
|
[BitAutoData]
|
|
public async Task GetVersionsBySecretId_SecretNotFound_Throws(
|
|
SutProvider<SecretVersionsController> sutProvider,
|
|
Guid secretId)
|
|
{
|
|
sutProvider.GetDependency<ISecretRepository>().GetByIdAsync(secretId).Returns((Secret?)null);
|
|
|
|
await Assert.ThrowsAsync<NotFoundException>(() =>
|
|
sutProvider.Sut.GetVersionsBySecretIdAsync(secretId));
|
|
}
|
|
|
|
[Theory]
|
|
[BitAutoData]
|
|
public async Task GetVersionsBySecretId_NoAccess_Throws(
|
|
SutProvider<SecretVersionsController> sutProvider,
|
|
Secret secret)
|
|
{
|
|
sutProvider.GetDependency<ISecretRepository>().GetByIdAsync(secret.Id).Returns(secret);
|
|
sutProvider.GetDependency<ICurrentContext>().AccessSecretsManager(secret.OrganizationId).Returns(false);
|
|
|
|
await Assert.ThrowsAsync<NotFoundException>(() =>
|
|
sutProvider.Sut.GetVersionsBySecretIdAsync(secret.Id));
|
|
}
|
|
|
|
[Theory]
|
|
[BitAutoData]
|
|
public async Task GetVersionsBySecretId_NoReadAccess_Throws(
|
|
SutProvider<SecretVersionsController> sutProvider,
|
|
Secret secret,
|
|
Guid userId)
|
|
{
|
|
sutProvider.GetDependency<ISecretRepository>().GetByIdAsync(secret.Id).Returns(secret);
|
|
sutProvider.GetDependency<ICurrentContext>().AccessSecretsManager(secret.OrganizationId).Returns(true);
|
|
sutProvider.GetDependency<IUserService>().GetProperUserId(default).ReturnsForAnyArgs(userId);
|
|
sutProvider.GetDependency<ICurrentContext>().OrganizationAdmin(secret.OrganizationId).Returns(false);
|
|
sutProvider.GetDependency<ISecretRepository>().AccessToSecretAsync(secret.Id, userId, default)
|
|
.ReturnsForAnyArgs((false, false));
|
|
|
|
await Assert.ThrowsAsync<NotFoundException>(() =>
|
|
sutProvider.Sut.GetVersionsBySecretIdAsync(secret.Id));
|
|
}
|
|
|
|
[Theory]
|
|
[BitAutoData]
|
|
public async Task GetVersionsBySecretId_Success(
|
|
SutProvider<SecretVersionsController> sutProvider,
|
|
Secret secret,
|
|
List<SecretVersion> versions,
|
|
Guid userId)
|
|
{
|
|
sutProvider.GetDependency<ISecretRepository>().GetByIdAsync(secret.Id).Returns(secret);
|
|
sutProvider.GetDependency<ICurrentContext>().AccessSecretsManager(secret.OrganizationId).Returns(true);
|
|
sutProvider.GetDependency<IUserService>().GetProperUserId(default).ReturnsForAnyArgs(userId);
|
|
sutProvider.GetDependency<ICurrentContext>().OrganizationAdmin(secret.OrganizationId).Returns(false);
|
|
sutProvider.GetDependency<ISecretRepository>().AccessToSecretAsync(secret.Id, userId, default)
|
|
.ReturnsForAnyArgs((true, false));
|
|
|
|
foreach (var version in versions)
|
|
{
|
|
version.SecretId = secret.Id;
|
|
}
|
|
sutProvider.GetDependency<ISecretVersionRepository>().GetManyBySecretIdAsync(secret.Id).Returns(versions);
|
|
|
|
var result = await sutProvider.Sut.GetVersionsBySecretIdAsync(secret.Id);
|
|
|
|
Assert.Equal(versions.Count, result.Data.Count());
|
|
await sutProvider.GetDependency<ISecretVersionRepository>().Received(1)
|
|
.GetManyBySecretIdAsync(Arg.Is(secret.Id));
|
|
}
|
|
|
|
[Theory]
|
|
[BitAutoData]
|
|
public async Task GetById_VersionNotFound_Throws(
|
|
SutProvider<SecretVersionsController> sutProvider,
|
|
Guid versionId)
|
|
{
|
|
sutProvider.GetDependency<ISecretVersionRepository>().GetByIdAsync(versionId).Returns((SecretVersion?)null);
|
|
|
|
await Assert.ThrowsAsync<NotFoundException>(() =>
|
|
sutProvider.Sut.GetByIdAsync(versionId));
|
|
}
|
|
|
|
[Theory]
|
|
[BitAutoData]
|
|
public async Task GetById_Success(
|
|
SutProvider<SecretVersionsController> sutProvider,
|
|
SecretVersion version,
|
|
Secret secret,
|
|
Guid userId)
|
|
{
|
|
version.SecretId = secret.Id;
|
|
sutProvider.GetDependency<ISecretVersionRepository>().GetByIdAsync(version.Id).Returns(version);
|
|
sutProvider.GetDependency<ISecretRepository>().GetByIdAsync(secret.Id).Returns(secret);
|
|
sutProvider.GetDependency<ICurrentContext>().AccessSecretsManager(secret.OrganizationId).Returns(true);
|
|
sutProvider.GetDependency<IUserService>().GetProperUserId(default).ReturnsForAnyArgs(userId);
|
|
sutProvider.GetDependency<ICurrentContext>().OrganizationAdmin(secret.OrganizationId).Returns(false);
|
|
sutProvider.GetDependency<ISecretRepository>().AccessToSecretAsync(secret.Id, userId, default)
|
|
.ReturnsForAnyArgs((true, false));
|
|
|
|
var result = await sutProvider.Sut.GetByIdAsync(version.Id);
|
|
|
|
Assert.Equal(version.Id, result.Id);
|
|
Assert.Equal(version.SecretId, result.SecretId);
|
|
}
|
|
|
|
[Theory]
|
|
[BitAutoData]
|
|
public async Task RestoreVersion_NoWriteAccess_Throws(
|
|
SutProvider<SecretVersionsController> sutProvider,
|
|
Secret secret,
|
|
SecretVersion version,
|
|
RestoreSecretVersionRequestModel request,
|
|
Guid userId)
|
|
{
|
|
version.SecretId = secret.Id;
|
|
request.VersionId = version.Id;
|
|
|
|
sutProvider.GetDependency<ISecretRepository>().GetByIdAsync(secret.Id).Returns(secret);
|
|
sutProvider.GetDependency<ICurrentContext>().AccessSecretsManager(secret.OrganizationId).Returns(true);
|
|
sutProvider.GetDependency<IUserService>().GetProperUserId(default).ReturnsForAnyArgs(userId);
|
|
sutProvider.GetDependency<ICurrentContext>().OrganizationAdmin(secret.OrganizationId).Returns(false);
|
|
sutProvider.GetDependency<ISecretRepository>().AccessToSecretAsync(secret.Id, userId, default)
|
|
.ReturnsForAnyArgs((true, false));
|
|
|
|
await Assert.ThrowsAsync<NotFoundException>(() =>
|
|
sutProvider.Sut.RestoreVersionAsync(secret.Id, request));
|
|
}
|
|
|
|
[Theory]
|
|
[BitAutoData]
|
|
public async Task RestoreVersion_VersionNotFound_Throws(
|
|
SutProvider<SecretVersionsController> sutProvider,
|
|
Secret secret,
|
|
RestoreSecretVersionRequestModel request,
|
|
Guid userId)
|
|
{
|
|
sutProvider.GetDependency<ISecretRepository>().GetByIdAsync(secret.Id).Returns(secret);
|
|
sutProvider.GetDependency<ICurrentContext>().AccessSecretsManager(secret.OrganizationId).Returns(true);
|
|
sutProvider.GetDependency<IUserService>().GetProperUserId(default).ReturnsForAnyArgs(userId);
|
|
sutProvider.GetDependency<ICurrentContext>().OrganizationAdmin(secret.OrganizationId).Returns(true);
|
|
sutProvider.GetDependency<ISecretRepository>().AccessToSecretAsync(secret.Id, userId, default)
|
|
.ReturnsForAnyArgs((true, true));
|
|
sutProvider.GetDependency<ISecretVersionRepository>().GetByIdAsync(request.VersionId).Returns((SecretVersion?)null);
|
|
|
|
await Assert.ThrowsAsync<NotFoundException>(() =>
|
|
sutProvider.Sut.RestoreVersionAsync(secret.Id, request));
|
|
}
|
|
|
|
[Theory]
|
|
[BitAutoData]
|
|
public async Task RestoreVersion_VersionBelongsToDifferentSecret_Throws(
|
|
SutProvider<SecretVersionsController> sutProvider,
|
|
Secret secret,
|
|
SecretVersion version,
|
|
RestoreSecretVersionRequestModel request,
|
|
Guid userId)
|
|
{
|
|
version.SecretId = Guid.NewGuid(); // Different secret
|
|
request.VersionId = version.Id;
|
|
|
|
sutProvider.GetDependency<ISecretRepository>().GetByIdAsync(secret.Id).Returns(secret);
|
|
sutProvider.GetDependency<ICurrentContext>().AccessSecretsManager(secret.OrganizationId).Returns(true);
|
|
sutProvider.GetDependency<IUserService>().GetProperUserId(default).ReturnsForAnyArgs(userId);
|
|
sutProvider.GetDependency<ICurrentContext>().OrganizationAdmin(secret.OrganizationId).Returns(true);
|
|
sutProvider.GetDependency<ISecretRepository>().AccessToSecretAsync(secret.Id, userId, default)
|
|
.ReturnsForAnyArgs((true, true));
|
|
sutProvider.GetDependency<ISecretVersionRepository>().GetByIdAsync(request.VersionId).Returns(version);
|
|
|
|
await Assert.ThrowsAsync<NotFoundException>(() =>
|
|
sutProvider.Sut.RestoreVersionAsync(secret.Id, request));
|
|
}
|
|
|
|
[Theory]
|
|
[BitAutoData]
|
|
public async Task RestoreVersion_Success(
|
|
SutProvider<SecretVersionsController> sutProvider,
|
|
Secret secret,
|
|
SecretVersion version,
|
|
RestoreSecretVersionRequestModel request,
|
|
Guid userId,
|
|
OrganizationUser organizationUser)
|
|
{
|
|
version.SecretId = secret.Id;
|
|
request.VersionId = version.Id;
|
|
var versionValue = version.Value;
|
|
organizationUser.OrganizationId = secret.OrganizationId;
|
|
organizationUser.UserId = userId;
|
|
|
|
sutProvider.GetDependency<ISecretRepository>().GetByIdAsync(secret.Id).Returns(secret);
|
|
sutProvider.GetDependency<ICurrentContext>().AccessSecretsManager(secret.OrganizationId).Returns(true);
|
|
sutProvider.GetDependency<IUserService>().GetProperUserId(default).ReturnsForAnyArgs(userId);
|
|
sutProvider.GetDependency<ICurrentContext>().OrganizationAdmin(secret.OrganizationId).Returns(true);
|
|
sutProvider.GetDependency<ISecretRepository>().AccessToSecretAsync(secret.Id, userId, default)
|
|
.ReturnsForAnyArgs((true, true));
|
|
sutProvider.GetDependency<ISecretVersionRepository>().GetByIdAsync(request.VersionId).Returns(version);
|
|
sutProvider.GetDependency<IOrganizationUserRepository>()
|
|
.GetByOrganizationAsync(secret.OrganizationId, userId).Returns(organizationUser);
|
|
sutProvider.GetDependency<ISecretRepository>().UpdateAsync(Arg.Any<Secret>()).Returns(x => x.Arg<Secret>());
|
|
|
|
var result = await sutProvider.Sut.RestoreVersionAsync(secret.Id, request);
|
|
|
|
await sutProvider.GetDependency<ISecretRepository>().Received(1)
|
|
.UpdateAsync(Arg.Is<Secret>(s => s.Value == versionValue));
|
|
}
|
|
|
|
[Theory]
|
|
[BitAutoData]
|
|
public async Task BulkDelete_EmptyIds_Throws(
|
|
SutProvider<SecretVersionsController> sutProvider)
|
|
{
|
|
await Assert.ThrowsAsync<BadRequestException>(() =>
|
|
sutProvider.Sut.BulkDeleteAsync(new List<Guid>()));
|
|
}
|
|
|
|
[Theory]
|
|
[BitAutoData]
|
|
public async Task BulkDelete_VersionNotFound_Throws(
|
|
SutProvider<SecretVersionsController> sutProvider,
|
|
List<Guid> ids)
|
|
{
|
|
sutProvider.GetDependency<ISecretVersionRepository>().GetByIdAsync(ids[0]).Returns((SecretVersion?)null);
|
|
|
|
await Assert.ThrowsAsync<NotFoundException>(() =>
|
|
sutProvider.Sut.BulkDeleteAsync(ids));
|
|
}
|
|
|
|
[Theory]
|
|
[BitAutoData]
|
|
public async Task BulkDelete_NoWriteAccess_Throws(
|
|
SutProvider<SecretVersionsController> sutProvider,
|
|
List<SecretVersion> versions,
|
|
Secret secret,
|
|
Guid userId)
|
|
{
|
|
var ids = versions.Select(v => v.Id).ToList();
|
|
foreach (var version in versions)
|
|
{
|
|
version.SecretId = secret.Id;
|
|
sutProvider.GetDependency<ISecretVersionRepository>().GetByIdAsync(version.Id).Returns(version);
|
|
}
|
|
|
|
sutProvider.GetDependency<ISecretRepository>().GetManyByIds(Arg.Any<IEnumerable<Guid>>())
|
|
.Returns(new List<Secret> { secret });
|
|
sutProvider.GetDependency<ICurrentContext>().AccessSecretsManager(secret.OrganizationId).Returns(true);
|
|
sutProvider.GetDependency<IUserService>().GetProperUserId(default).ReturnsForAnyArgs(userId);
|
|
sutProvider.GetDependency<ICurrentContext>().OrganizationAdmin(secret.OrganizationId).Returns(false);
|
|
sutProvider.GetDependency<ISecretRepository>().AccessToSecretAsync(secret.Id, userId, default)
|
|
.ReturnsForAnyArgs((true, false));
|
|
|
|
await Assert.ThrowsAsync<NotFoundException>(() =>
|
|
sutProvider.Sut.BulkDeleteAsync(ids));
|
|
}
|
|
|
|
[Theory]
|
|
[BitAutoData]
|
|
public async Task BulkDelete_Success(
|
|
SutProvider<SecretVersionsController> sutProvider,
|
|
List<SecretVersion> versions,
|
|
Secret secret,
|
|
Guid userId)
|
|
{
|
|
var ids = versions.Select(v => v.Id).ToList();
|
|
foreach (var version in versions)
|
|
{
|
|
version.SecretId = secret.Id;
|
|
}
|
|
|
|
sutProvider.GetDependency<ISecretVersionRepository>().GetManyByIdsAsync(ids).Returns(versions);
|
|
sutProvider.GetDependency<ISecretRepository>().GetManyByIds(Arg.Any<IEnumerable<Guid>>())
|
|
.Returns(new List<Secret> { secret });
|
|
sutProvider.GetDependency<ICurrentContext>().AccessSecretsManager(secret.OrganizationId).Returns(true);
|
|
sutProvider.GetDependency<ICurrentContext>().IdentityClientType.Returns(IdentityClientType.ServiceAccount);
|
|
sutProvider.GetDependency<IUserService>().GetProperUserId(default).ReturnsForAnyArgs(userId);
|
|
sutProvider.GetDependency<ICurrentContext>().OrganizationAdmin(secret.OrganizationId).Returns(true);
|
|
sutProvider.GetDependency<ISecretRepository>().AccessToSecretAsync(secret.Id, userId, default)
|
|
.ReturnsForAnyArgs((true, true));
|
|
|
|
await sutProvider.Sut.BulkDeleteAsync(ids);
|
|
|
|
await sutProvider.GetDependency<ISecretVersionRepository>().Received(1)
|
|
.DeleteManyByIdAsync(Arg.Is<IEnumerable<Guid>>(x => x.SequenceEqual(ids)));
|
|
}
|
|
}
|