Headless CMS and Content Managment Hub
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.
 
 
 
 
 

470 lines
15 KiB

// ==========================================================================
// Squidex Headless CMS
// ==========================================================================
// Copyright (c) Squidex UG (haftungsbeschraenkt)
// All rights reserved. Licensed under the MIT license.
// ==========================================================================
using System;
using System.Collections.Generic;
using System.Threading.Tasks;
using FakeItEasy;
using Microsoft.Extensions.Caching.Memory;
using Microsoft.Extensions.Options;
using Orleans;
using Squidex.Domain.Apps.Core.Apps;
using Squidex.Domain.Apps.Entities.Apps.Commands;
using Squidex.Infrastructure;
using Squidex.Infrastructure.Caching;
using Squidex.Infrastructure.Commands;
using Squidex.Infrastructure.Orleans;
using Squidex.Infrastructure.Security;
using Squidex.Infrastructure.Validation;
using Xunit;
#pragma warning disable SA1133 // Do not combine attributes
namespace Squidex.Domain.Apps.Entities.Apps.Indexes
{
public sealed class AppsIndexTests
{
private readonly IGrainFactory grainFactory = A.Fake<IGrainFactory>();
private readonly IAppsByNameIndexGrain indexByName = A.Fake<IAppsByNameIndexGrain>();
private readonly IAppsByUserIndexGrain indexByUser = A.Fake<IAppsByUserIndexGrain>();
private readonly ICommandBus commandBus = A.Fake<ICommandBus>();
private readonly NamedId<Guid> appId = NamedId.Of(Guid.NewGuid(), "my-app");
private readonly string userId = "user-1";
private readonly AppsIndex sut;
public AppsIndexTests()
{
A.CallTo(() => grainFactory.GetGrain<IAppsByNameIndexGrain>(SingleGrain.Id, null))
.Returns(indexByName);
A.CallTo(() => grainFactory.GetGrain<IAppsByUserIndexGrain>(userId, null))
.Returns(indexByUser);
var cache = new ReplicatedCache(new MemoryCache(Options.Create(new MemoryCacheOptions())), new SimplePubSub());
sut = new AppsIndex(grainFactory, cache);
}
[Fact]
public async Task Should_resolve_all_apps_from_user_permissions()
{
var expected = SetupApp();
A.CallTo(() => indexByName.GetIdsAsync(A<string[]>.That.IsSameSequenceAs(new[] { appId.Name })))
.Returns(new List<Guid> { appId.Id });
var actual = await sut.GetAppsForUserAsync(userId, new PermissionSet($"squidex.apps.{appId.Name}"));
Assert.Same(expected, actual[0]);
}
[Fact]
public async Task Should_resolve_all_apps_from_user()
{
var expected = SetupApp();
A.CallTo(() => indexByUser.GetIdsAsync())
.Returns(new List<Guid> { appId.Id });
var actual = await sut.GetAppsForUserAsync(userId, PermissionSet.Empty);
Assert.Same(expected, actual[0]);
}
[Fact]
public async Task Should_resolve_combined_apps()
{
var expected = SetupApp();
A.CallTo(() => indexByName.GetIdsAsync(A<string[]>.That.IsSameSequenceAs(new[] { appId.Name })))
.Returns(new List<Guid> { appId.Id });
A.CallTo(() => indexByUser.GetIdsAsync())
.Returns(new List<Guid> { appId.Id });
var actual = await sut.GetAppsForUserAsync(userId, new PermissionSet($"squidex.apps.{appId.Name}"));
Assert.Single(actual);
Assert.Same(expected, actual[0]);
}
[Fact]
public async Task Should_resolve_all_apps()
{
var expected = SetupApp();
A.CallTo(() => indexByName.GetIdsAsync())
.Returns(new List<Guid> { appId.Id });
var actual = await sut.GetAppsAsync();
Assert.Same(expected, actual[0]);
}
[Fact]
public async Task Should_resolve_app_by_name()
{
var expected = SetupApp();
A.CallTo(() => indexByName.GetIdAsync(appId.Name))
.Returns(appId.Id);
var actual1 = await sut.GetAppByNameAsync(appId.Name, false);
var actual2 = await sut.GetAppByNameAsync(appId.Name, false);
Assert.Same(expected, actual1);
Assert.Same(expected, actual2);
A.CallTo(() => grainFactory.GetGrain<IAppGrain>(appId.Id, null))
.MustHaveHappenedTwiceExactly();
A.CallTo(() => indexByName.GetIdAsync(A<string>._))
.MustHaveHappenedTwiceExactly();
}
[Fact]
public async Task Should_resolve_app_by_name_and_id_if_cached_before()
{
var expected = SetupApp();
A.CallTo(() => indexByName.GetIdAsync(appId.Name))
.Returns(appId.Id);
var actual1 = await sut.GetAppByNameAsync(appId.Name, true);
var actual2 = await sut.GetAppByNameAsync(appId.Name, true);
var actual3 = await sut.GetAppAsync(appId.Id, true);
Assert.Same(expected, actual1);
Assert.Same(expected, actual2);
Assert.Same(expected, actual3);
A.CallTo(() => grainFactory.GetGrain<IAppGrain>(appId.Id, null))
.MustHaveHappenedOnceExactly();
A.CallTo(() => indexByName.GetIdAsync(A<string>._))
.MustHaveHappenedOnceExactly();
}
[Fact]
public async Task Should_resolve_app_by_id()
{
var expected = SetupApp();
var actual1 = await sut.GetAppAsync(appId.Id, false);
var actual2 = await sut.GetAppAsync(appId.Id, false);
Assert.Same(expected, actual1);
Assert.Same(expected, actual2);
A.CallTo(() => grainFactory.GetGrain<IAppGrain>(appId.Id, null))
.MustHaveHappenedTwiceExactly();
A.CallTo(() => indexByName.GetIdAsync(A<string>._))
.MustNotHaveHappened();
}
[Fact]
public async Task Should_resolve_app_by_id_and_name_if_cached_before()
{
var expected = SetupApp();
var actual1 = await sut.GetAppAsync(appId.Id, true);
var actual2 = await sut.GetAppAsync(appId.Id, true);
var actual3 = await sut.GetAppByNameAsync(appId.Name, true);
Assert.Same(expected, actual1);
Assert.Same(expected, actual2);
Assert.Same(expected, actual3);
A.CallTo(() => grainFactory.GetGrain<IAppGrain>(appId.Id, null))
.MustHaveHappenedOnceExactly();
A.CallTo(() => indexByName.GetIdAsync(A<string>._))
.MustNotHaveHappened();
}
[Fact]
public async Task Should_return_null_if_app_not_created()
{
SetupApp(EtagVersion.NotFound);
var actual = await sut.GetAppAsync(appId.Id, false);
Assert.Null(actual);
}
[Fact]
public async Task Should_add_app_to_indexes_on_create()
{
var token = RandomHash.Simple();
A.CallTo(() => indexByName.ReserveAsync(appId.Id, appId.Name))
.Returns(token);
var context =
new CommandContext(Create(appId.Name), commandBus)
.Complete();
await sut.HandleAsync(context);
A.CallTo(() => indexByName.AddAsync(token))
.MustHaveHappened();
A.CallTo(() => indexByName.RemoveReservationAsync(A<string>._))
.MustNotHaveHappened();
A.CallTo(() => indexByUser.AddAsync(appId.Id))
.MustHaveHappened();
}
[Fact]
public async Task Should_also_app_to_user_index_if_app_created_by_client()
{
var token = RandomHash.Simple();
A.CallTo(() => indexByName.ReserveAsync(appId.Id, appId.Name))
.Returns(token);
var context =
new CommandContext(CreateFromClient(appId.Name), commandBus)
.Complete();
await sut.HandleAsync(context);
A.CallTo(() => indexByName.AddAsync(token))
.MustHaveHappened();
A.CallTo(() => indexByName.RemoveReservationAsync(A<string>._))
.MustNotHaveHappened();
A.CallTo(() => indexByUser.AddAsync(appId.Id))
.MustHaveHappened();
}
[Fact]
public async Task Should_clear_reservation_when_app_creation_failed()
{
var token = RandomHash.Simple();
A.CallTo(() => indexByName.ReserveAsync(appId.Id, appId.Name))
.Returns(token);
var context =
new CommandContext(CreateFromClient(appId.Name), commandBus);
await sut.HandleAsync(context);
A.CallTo(() => indexByName.AddAsync(token))
.MustNotHaveHappened();
A.CallTo(() => indexByName.RemoveReservationAsync(token))
.MustHaveHappened();
A.CallTo(() => indexByUser.AddAsync(appId.Id))
.MustNotHaveHappened();
}
[Fact]
public async Task Should_not_add_to_indexes_on_create_if_name_taken()
{
A.CallTo(() => indexByName.ReserveAsync(appId.Id, appId.Name))
.Returns(Task.FromResult<string?>(null));
var context =
new CommandContext(Create(appId.Name), commandBus)
.Complete();
await Assert.ThrowsAsync<ValidationException>(() => sut.HandleAsync(context));
A.CallTo(() => indexByName.AddAsync(A<string>._))
.MustNotHaveHappened();
A.CallTo(() => indexByName.RemoveReservationAsync(A<string>._))
.MustNotHaveHappened();
A.CallTo(() => indexByUser.AddAsync(appId.Id))
.MustNotHaveHappened();
}
[Fact]
public async Task Should_not_add_to_indexes_on_create_if_name_invalid()
{
var context =
new CommandContext(Create("INVALID"), commandBus)
.Complete();
await sut.HandleAsync(context);
A.CallTo(() => indexByName.ReserveAsync(appId.Id, A<string>._))
.MustNotHaveHappened();
A.CallTo(() => indexByName.RemoveReservationAsync(A<string>._))
.MustNotHaveHappened();
A.CallTo(() => indexByUser.AddAsync(appId.Id))
.MustNotHaveHappened();
}
[Fact]
public async Task Should_add_app_to_index_on_contributor_assignment()
{
SetupApp();
var command = new AssignContributor { AppId = appId.Id, ContributorId = userId };
var context =
new CommandContext(command, commandBus)
.Complete();
await sut.HandleAsync(context);
A.CallTo(() => indexByUser.AddAsync(appId.Id))
.MustHaveHappened();
}
[Fact]
public async Task Should_remove_from_user_index_on_remove_of_contributor()
{
SetupApp();
var command = new RemoveContributor { AppId = appId.Id, ContributorId = userId };
var context =
new CommandContext(command, commandBus)
.Complete();
await sut.HandleAsync(context);
A.CallTo(() => indexByUser.RemoveAsync(appId.Id))
.MustHaveHappened();
}
[Fact]
public async Task Should_remove_app_from_indexes_on_archive()
{
SetupApp();
var command = new ArchiveApp { AppId = appId.Id };
var context =
new CommandContext(command, commandBus)
.Complete();
await sut.HandleAsync(context);
A.CallTo(() => indexByName.RemoveAsync(appId.Id))
.MustHaveHappened();
A.CallTo(() => indexByUser.RemoveAsync(appId.Id))
.MustHaveHappened();
}
[Fact]
public async Task Should_forward_call_when_rebuilding_for_contributors1()
{
var apps = new HashSet<Guid>();
await sut.RebuildByContributorsAsync(userId, apps);
A.CallTo(() => indexByUser.RebuildAsync(apps))
.MustHaveHappened();
}
[Fact]
public async Task Should_forward_call_when_rebuilding_for_contributors2()
{
var users = new HashSet<string> { userId };
await sut.RebuildByContributorsAsync(appId.Id, users);
A.CallTo(() => indexByUser.AddAsync(appId.Id))
.MustHaveHappened();
}
[Fact]
public async Task Should_forward_call_when_rebuilding()
{
var apps = new Dictionary<string, Guid>();
await sut.RebuildAsync(apps);
A.CallTo(() => indexByName.RebuildAsync(apps))
.MustHaveHappened();
}
[Fact]
public async Task Should_forward_reserveration()
{
await sut.AddAsync("token");
A.CallTo(() => indexByName.AddAsync("token"))
.MustHaveHappened();
}
[Fact]
public async Task Should_forward_remove_reservation()
{
await sut.RemoveReservationAsync("token");
A.CallTo(() => indexByName.RemoveReservationAsync("token"))
.MustHaveHappened();
}
[Fact]
public async Task Should_forward_request_for_ids()
{
await sut.GetIdsAsync();
A.CallTo(() => indexByName.GetIdsAsync())
.MustHaveHappened();
}
private IAppEntity SetupApp(long version = 0)
{
var appEntity = A.Fake<IAppEntity>();
A.CallTo(() => appEntity.Id)
.Returns(appId.Id);
A.CallTo(() => appEntity.Name)
.Returns(appId.Name);
A.CallTo(() => appEntity.Version)
.Returns(version);
A.CallTo(() => appEntity.Contributors)
.Returns(AppContributors.Empty.Assign(userId, Role.Owner));
var appGrain = A.Fake<IAppGrain>();
A.CallTo(() => appGrain.GetStateAsync())
.Returns(J.Of(appEntity));
A.CallTo(() => grainFactory.GetGrain<IAppGrain>(appId.Id, null))
.Returns(appGrain);
return appEntity;
}
private CreateApp Create(string name)
{
return new CreateApp { AppId = appId.Id, Name = name, Actor = ActorSubject() };
}
private CreateApp CreateFromClient(string name)
{
return new CreateApp { AppId = appId.Id, Name = name, Actor = ActorClient() };
}
private RefToken ActorSubject()
{
return new RefToken(RefTokenType.Subject, userId);
}
private RefToken ActorClient()
{
return new RefToken(RefTokenType.Client, userId);
}
}
}