mirror of https://github.com/Squidex/squidex.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
415 lines
14 KiB
415 lines
14 KiB
// ==========================================================================
|
|
// Squidex Headless CMS
|
|
// ==========================================================================
|
|
// Copyright (c) Squidex UG (haftungsbeschraenkt)
|
|
// All rights reserved. Licensed under the MIT license.
|
|
// ==========================================================================
|
|
|
|
using System.Security.Claims;
|
|
using Microsoft.Extensions.DependencyInjection;
|
|
using Squidex.Domain.Apps.Core.Contents;
|
|
using Squidex.Domain.Apps.Core.Schemas;
|
|
using Squidex.Domain.Apps.Core.TestHelpers;
|
|
using Squidex.Domain.Apps.Entities.Contents.Commands;
|
|
using Squidex.Domain.Apps.Entities.Contents.Repositories;
|
|
using Squidex.Domain.Apps.Entities.TestHelpers;
|
|
using Squidex.Infrastructure;
|
|
using Squidex.Infrastructure.Validation;
|
|
using Squidex.Shared;
|
|
|
|
#pragma warning disable CA2012 // Use ValueTasks correctly
|
|
|
|
namespace Squidex.Domain.Apps.Entities.Contents.DomainObject.Guards;
|
|
|
|
public class GuardContentTests : GivenContext, IClassFixture<TranslationsFixture>
|
|
{
|
|
private readonly IContentWorkflow contentWorkflow = A.Fake<IContentWorkflow>();
|
|
private readonly IContentRepository contentRepository = A.Fake<IContentRepository>();
|
|
private readonly Schema normalSchema;
|
|
private readonly Schema normalUnpublishedSchema;
|
|
private readonly Schema singletonSchema;
|
|
private readonly Schema singletonUnpublishedSchema;
|
|
private readonly Schema componentSchema;
|
|
|
|
public GuardContentTests()
|
|
{
|
|
normalUnpublishedSchema =
|
|
Schema.Unpublish();
|
|
|
|
normalSchema =
|
|
normalUnpublishedSchema.Publish();
|
|
|
|
singletonUnpublishedSchema =
|
|
normalUnpublishedSchema with { Type = SchemaType.Singleton };
|
|
|
|
singletonSchema =
|
|
normalSchema with { Type = SchemaType.Singleton };
|
|
|
|
componentSchema =
|
|
normalSchema with { Type = SchemaType.Component };
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_throw_exception_if_creating_content_for_unpublished_schema()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), normalUnpublishedSchema);
|
|
|
|
Assert.Throws<DomainException>(() => operation.MustNotCreateForUnpublishedSchema());
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_not_throw_exception_if_creating_content_for_unpublished_singleton()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft, singletonSchema.Id), singletonUnpublishedSchema);
|
|
|
|
operation.MustNotCreateSingleton();
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_not_throw_exception_if_creating_content_for_published_schema()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft, singletonSchema.Id), normalSchema);
|
|
|
|
operation.MustNotCreateSingleton();
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_not_throw_exception_if_creating_content_for_published_singleton_schema()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft, singletonSchema.Id), singletonSchema);
|
|
|
|
operation.MustNotCreateSingleton();
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_throw_exception_if_creating_singleton_content()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), singletonSchema);
|
|
|
|
Assert.Throws<DomainException>(() => operation.MustNotCreateSingleton());
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_throw_exception_if_creating_component_content()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), componentSchema);
|
|
|
|
Assert.Throws<DomainException>(() => operation.MustNotCreateComponent());
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_not_throw_exception_if_creating_singleton_content_with_schema_id()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft, singletonSchema.Id), singletonSchema);
|
|
|
|
operation.MustNotCreateSingleton();
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_not_throw_exception_if_creating_non_singleton_content()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft, singletonSchema.Id), normalSchema);
|
|
|
|
operation.MustNotCreateSingleton();
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_throw_exception_if_changing_singleton_content()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), singletonSchema);
|
|
|
|
Assert.Throws<DomainException>(() => operation.MustNotChangeSingleton(Status.Archived));
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_not_throw_exception_if_changing_singleton_to_published()
|
|
{
|
|
var operation = Operation(CreateDraftContent(Status.Published, singletonSchema.Id), singletonSchema);
|
|
|
|
operation.MustNotChangeSingleton(Status.Published);
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_not_throw_exception_if_changing_non_singleton_content()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft, singletonSchema.Id), normalSchema);
|
|
|
|
operation.MustNotChangeSingleton(Status.Archived);
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_throw_exception_if_deleting_singleton_content()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), singletonSchema);
|
|
|
|
Assert.Throws<DomainException>(() => operation.MustNotDeleteSingleton());
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_not_throw_exception_if_deleting_non_singleton_content()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft, singletonSchema.Id), normalSchema);
|
|
|
|
operation.MustNotDeleteSingleton();
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_throw_exception_if_draft_already_created()
|
|
{
|
|
var operation = Operation(CreateDraftContent(Status.Draft), normalSchema);
|
|
|
|
Assert.Throws<DomainException>(() => operation.MustCreateDraft());
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_throw_exception_if_draft_cannot_be_created()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), normalSchema);
|
|
|
|
Assert.Throws<DomainException>(() => operation.MustCreateDraft());
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_not_throw_exception_if_draft_can_be_created()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Published), normalSchema);
|
|
|
|
operation.MustCreateDraft();
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_throw_exception_if_draft_cannot_be_deleted()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Published), normalSchema);
|
|
|
|
Assert.Throws<DomainException>(() => operation.MustDeleteDraft());
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_not_throw_exception_if_draft_can_be_deleted()
|
|
{
|
|
var operation = Operation(CreateDraftContent(Status.Draft), normalSchema);
|
|
|
|
operation.MustDeleteDraft();
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_throw_exception_if_data_is_not_defined()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), normalSchema);
|
|
|
|
Assert.Throws<ValidationException>(() => operation.MustHaveData(null));
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_not_throw_exception_if_data_is_defined()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), normalSchema);
|
|
|
|
operation.MustHaveData([]);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_provide_initial_status()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), normalSchema);
|
|
|
|
A.CallTo(() => contentWorkflow.GetInitialStatusAsync(operation.Schema))
|
|
.Returns(Status.Archived);
|
|
|
|
Assert.Equal(Status.Archived, await operation.GetInitialStatusAsync());
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_throw_exception_if_workflow_permits_update()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), normalSchema);
|
|
|
|
A.CallTo(() => contentWorkflow.CanUpdateAsync(operation.Snapshot.ToContent(), operation.Snapshot.EditingStatus, operation.User))
|
|
.Returns(false);
|
|
|
|
await Assert.ThrowsAsync<DomainException>(() => operation.CheckUpdateAsync());
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_not_throw_exception_if_workflow_allows_update()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), normalSchema);
|
|
|
|
A.CallTo(() => contentWorkflow.CanUpdateAsync(operation.Snapshot.ToContent(), operation.Snapshot.EditingStatus, operation.User))
|
|
.Returns(true);
|
|
|
|
await operation.CheckUpdateAsync();
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_throw_exception_if_workflow_status_not_valid()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), normalSchema);
|
|
|
|
A.CallTo(() => contentWorkflow.GetInfoAsync(operation.Snapshot.ToContent(), Status.Archived))
|
|
.Returns(ValueTask.FromResult<StatusInfo?>(null));
|
|
|
|
await Assert.ThrowsAsync<ValidationException>(() => operation.CheckStatusAsync(Status.Archived));
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_not_throw_exception_if_workflow_status_is_valid()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), normalSchema);
|
|
|
|
A.CallTo(() => contentWorkflow.GetInfoAsync(operation.Snapshot.ToContent(), Status.Archived))
|
|
.Returns(new StatusInfo(Status.Archived, StatusColors.Archived));
|
|
|
|
await operation.CheckStatusAsync(Status.Archived);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_not_throw_exception_if_workflow_status_is_checked_for_singleton()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), singletonSchema);
|
|
|
|
await operation.CheckStatusAsync(Status.Archived);
|
|
|
|
A.CallTo(() => contentWorkflow.GetInfoAsync(operation.Snapshot.ToContent(), Status.Archived))
|
|
.MustNotHaveHappened();
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_throw_exception_if_workflow_transition_not_valid()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), normalSchema);
|
|
|
|
A.CallTo(() => contentWorkflow.CanMoveToAsync(operation.Snapshot.ToContent(), operation.Snapshot.EditingStatus, Status.Archived, operation.User))
|
|
.Returns(false);
|
|
|
|
await Assert.ThrowsAsync<ValidationException>(() => operation.CheckTransitionAsync(Status.Archived));
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_not_throw_exception_if_workflow_transition_is_valid()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), normalSchema);
|
|
|
|
A.CallTo(() => contentWorkflow.CanMoveToAsync(operation.Snapshot.ToContent(), operation.Snapshot.EditingStatus, Status.Archived, operation.User))
|
|
.Returns(true);
|
|
|
|
await operation.CheckTransitionAsync(Status.Archived);
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_not_throw_exception_if_workflow_transition_is_checked_for_singleton()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), singletonSchema);
|
|
|
|
await operation.CheckTransitionAsync(Status.Archived);
|
|
|
|
A.CallTo(() => contentWorkflow.CanMoveToAsync(operation.Snapshot.ToContent(), operation.Snapshot.EditingStatus, A<Status>._, A<ClaimsPrincipal>._))
|
|
.MustNotHaveHappened();
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_not_throw_exception_if_content_is_from_another_user_but_user_has_permission()
|
|
{
|
|
var userPermissions = PermissionIds.ForApp(PermissionIds.AppContentsDelete, AppId.Name, SchemaId.Name).Id;
|
|
var userPrincipal = Mocks.FrontendUser(permissions: [userPermissions]);
|
|
|
|
var operation = Operation(CreateContent(Status.Draft) with { CreatedBy = RefToken.User("invalid") }, normalSchema, userPrincipal);
|
|
|
|
operation.MustHavePermission(PermissionIds.AppContentsDelete);
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_not_throw_exception_if_content_is_from_current_user()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), normalSchema);
|
|
|
|
operation.MustHavePermission(PermissionIds.AppContentsDelete);
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_not_throw_exception_if_user_is_null()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft) with { CreatedBy = RefToken.User("invalid") }, normalSchema, null);
|
|
|
|
operation.MustHavePermission(PermissionIds.AppContentsDelete);
|
|
}
|
|
|
|
[Fact]
|
|
public void Should_throw_exception_if_content_is_from_another_user_and_user_has_no_permission()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft) with { CreatedBy = RefToken.User("invalid") }, normalSchema);
|
|
|
|
Assert.Throws<DomainForbiddenException>(() => operation.MustHavePermission(PermissionIds.AppContentsDelete));
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_throw_exception_if_referenced()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), normalSchema);
|
|
|
|
A.CallTo(() => contentRepository.HasReferrersAsync(App, operation.CommandId, SearchScope.All, CancellationToken))
|
|
.Returns(true);
|
|
|
|
await Assert.ThrowsAsync<DomainException>(() => operation.CheckReferrersAsync(CancellationToken));
|
|
}
|
|
|
|
[Fact]
|
|
public async Task Should_not_throw_exception_if_not_referenced()
|
|
{
|
|
var operation = Operation(CreateContent(Status.Draft), normalSchema);
|
|
|
|
A.CallTo(() => contentRepository.HasReferrersAsync(App, operation.CommandId, SearchScope.All, CancellationToken))
|
|
.Returns(true);
|
|
|
|
await Assert.ThrowsAsync<DomainException>(() => operation.CheckReferrersAsync(CancellationToken));
|
|
}
|
|
|
|
private ContentOperation Operation(WriteContent content, Schema operationSchema)
|
|
{
|
|
return Operation(content, operationSchema, Mocks.FrontendUser());
|
|
}
|
|
|
|
private ContentOperation Operation(WriteContent content, Schema operationSchema, ClaimsPrincipal? currentUser)
|
|
{
|
|
var serviceProvider =
|
|
new ServiceCollection()
|
|
.AddSingleton(contentRepository)
|
|
.AddSingleton(contentWorkflow)
|
|
.BuildServiceProvider();
|
|
|
|
return new ContentOperation(serviceProvider, () => content)
|
|
{
|
|
App = App,
|
|
Command = new CreateContent { User = currentUser, Actor = User },
|
|
CommandId = content.Id,
|
|
Schema = operationSchema
|
|
};
|
|
}
|
|
|
|
private WriteContent CreateDraftContent(Status status, DomainId? id = null)
|
|
{
|
|
return CreateContentCore(Status.Published, status, id);
|
|
}
|
|
|
|
private WriteContent CreateContent(Status status, DomainId? id = null)
|
|
{
|
|
return CreateContentCore(status, null, id);
|
|
}
|
|
|
|
private WriteContent CreateContentCore(Status status, Status? newStatus, DomainId? id = null)
|
|
{
|
|
return new WriteContent
|
|
{
|
|
Id = id ?? DomainId.NewGuid(),
|
|
AppId = AppId,
|
|
Created = default,
|
|
CreatedBy = User,
|
|
CurrentVersion = new ContentVersion(status, []),
|
|
LastModified = default,
|
|
LastModifiedBy = User,
|
|
NewVersion = newStatus != null ? new ContentVersion(newStatus.Value, []) : null,
|
|
SchemaId = SchemaId
|
|
};
|
|
}
|
|
}
|
|
|