// ========================================================================== // Squidex Headless CMS // ========================================================================== // Copyright (c) Squidex UG (haftungsbeschränkt) // All rights reserved. Licensed under the MIT license. // ========================================================================== using System; using System.Collections.Generic; using System.Collections.ObjectModel; using System.Linq; using System.Threading.Tasks; using FakeItEasy; using Squidex.Domain.Apps.Core.Contents; using Squidex.Domain.Apps.Core.HandleRules; using Squidex.Domain.Apps.Core.Rules.EnrichedEvents; using Squidex.Domain.Apps.Core.Rules.Triggers; using Squidex.Domain.Apps.Core.Scripting; using Squidex.Domain.Apps.Events; using Squidex.Domain.Apps.Events.Assets; using Squidex.Domain.Apps.Events.Contents; using Squidex.Infrastructure; using Squidex.Infrastructure.Caching; using Squidex.Infrastructure.EventSourcing; using Squidex.Infrastructure.Json.Objects; using Xunit; #pragma warning disable SA1401 // Fields must be private #pragma warning disable RECS0070 namespace Squidex.Domain.Apps.Entities.Contents { public class ContentChangedTriggerHandlerTests { private readonly IScriptEngine scriptEngine = A.Fake(); private readonly ILocalCache localCache = new AsyncLocalCache(); private readonly IContentLoader contentLoader = A.Fake(); private readonly ContentChangedTriggerHandler sut; private readonly IRuleTriggerHandler handler; private readonly Guid ruleId = Guid.NewGuid(); private static readonly NamedId SchemaMatch = NamedId.Of(Guid.NewGuid(), "my-schema1"); private static readonly NamedId SchemaNonMatch = NamedId.Of(Guid.NewGuid(), "my-schema2"); public ContentChangedTriggerHandlerTests() { A.CallTo(() => scriptEngine.Evaluate(A._, "true")) .Returns(true); A.CallTo(() => scriptEngine.Evaluate(A._, "false")) .Returns(false); sut = new ContentChangedTriggerHandler(scriptEngine, contentLoader, localCache); handler = sut; } public static IEnumerable TestEvents() { yield return new object[] { new ContentCreated(), EnrichedContentEventType.Created }; yield return new object[] { new ContentUpdated(), EnrichedContentEventType.Updated }; yield return new object[] { new ContentDeleted(), EnrichedContentEventType.Deleted }; yield return new object[] { new ContentStatusChanged { Change = StatusChange.Change }, EnrichedContentEventType.StatusChanged }; yield return new object[] { new ContentStatusChanged { Change = StatusChange.Published }, EnrichedContentEventType.Published }; yield return new object[] { new ContentStatusChanged { Change = StatusChange.Unpublished }, EnrichedContentEventType.Unpublished }; } [Fact] public async Task Should_resolve_reference_if_value_from_content_loader() { var referenceId = Guid.NewGuid(); var referenceValue = JsonValue.Array(referenceId); SetupReference(referenceId); var (handled, result) = sut.Format(null!, referenceValue, new[] { "data", "field1", "iv" }); Assert.True(handled); Assert.Equal("Hello", await result); } [Fact] public async Task Should_resolve_reference_only_once() { using (localCache.StartContext()) { var referenceId = Guid.NewGuid(); var referenceValue = JsonValue.Array(referenceId); SetupReference(referenceId); var (handled1, result1) = sut.Format(null!, referenceValue, new[] { "data", "field1", "iv" }); var (handled2, result2) = sut.Format(null!, referenceValue, new[] { "data", "field2", "iv" }); Assert.True(handled1); Assert.Equal("Hello", await result1); Assert.True(handled2); Assert.Equal("World", await result2); A.CallTo(() => contentLoader.GetAsync(A._, A._)) .MustHaveHappenedOnceExactly(); } } [Fact] public async Task Should_not_return_value_if_path_not_found_in_reference() { var referenceId = Guid.NewGuid(); var referenceValue = JsonValue.Array(referenceId); SetupReference(referenceId); var (handled, result) = sut.Format(null!, referenceValue, new[] { "data", "invalid", "iv" }); Assert.True(handled); Assert.Null(await result); } [Theory] [MemberData(nameof(TestEvents))] public async Task Should_create_enriched_events(ContentEvent @event, EnrichedContentEventType type) { var envelope = Envelope.Create(@event).SetEventStreamNumber(12); A.CallTo(() => contentLoader.GetAsync(@event.ContentId, 12)) .Returns(new ContentEntity { SchemaId = SchemaMatch }); var result = await handler.CreateEnrichedEventsAsync(envelope); var enrichedEvent = result.Single() as EnrichedContentEvent; Assert.Equal(type, enrichedEvent!.Type); } [Fact] public async Task Should_enrich_with_old_data_when_updated() { var @event = new ContentUpdated(); var envelope = Envelope.Create(@event).SetEventStreamNumber(12); var dataNow = new NamedContentData(); var dataOld = new NamedContentData(); A.CallTo(() => contentLoader.GetAsync(@event.ContentId, 12)) .Returns(new ContentEntity { SchemaId = SchemaMatch, Version = 12, Data = dataNow }); A.CallTo(() => contentLoader.GetAsync(@event.ContentId, 11)) .Returns(new ContentEntity { SchemaId = SchemaMatch, Version = 11, Data = dataOld }); var result = await handler.CreateEnrichedEventsAsync(envelope); var enrichedEvent = result.Single() as EnrichedContentEvent; Assert.Same(dataNow, enrichedEvent!.Data); Assert.Same(dataOld, enrichedEvent!.DataOld); } [Fact] public void Should_not_trigger_precheck_when_event_type_not_correct() { TestForTrigger(handleAll: true, schemaId: null, condition: null, action: trigger => { var result = handler.Trigger(new AssetCreated(), trigger, ruleId); Assert.False(result); }); } [Fact] public void Should_not_trigger_precheck_when_trigger_contains_no_schemas() { TestForTrigger(handleAll: false, schemaId: null, condition: null, action: trigger => { var result = handler.Trigger(new ContentCreated { SchemaId = SchemaMatch }, trigger, ruleId); Assert.False(result); }); } [Fact] public void Should_trigger_precheck_when_handling_all_events() { TestForTrigger(handleAll: true, schemaId: SchemaMatch, condition: null, action: trigger => { var result = handler.Trigger(new ContentCreated { SchemaId = SchemaMatch }, trigger, ruleId); Assert.True(result); }); } [Fact] public void Should_trigger_precheck_when_condition_is_empty() { TestForTrigger(handleAll: false, schemaId: SchemaMatch, condition: string.Empty, action: trigger => { var result = handler.Trigger(new ContentCreated { SchemaId = SchemaMatch }, trigger, ruleId); Assert.True(result); }); } [Fact] public void Should_not_trigger_precheck_when_schema_id_does_not_match() { TestForTrigger(handleAll: false, schemaId: SchemaNonMatch, condition: null, action: trigger => { var result = handler.Trigger(new ContentCreated { SchemaId = SchemaMatch }, trigger, ruleId); Assert.False(result); }); } [Fact] public void Should_not_trigger_check_when_event_type_not_correct() { TestForTrigger(handleAll: true, schemaId: null, condition: null, action: trigger => { var result = handler.Trigger(new EnrichedAssetEvent(), trigger); Assert.False(result); }); } [Fact] public void Should_not_trigger_check_when_trigger_contains_no_schemas() { TestForTrigger(handleAll: false, schemaId: null, condition: null, action: trigger => { var result = handler.Trigger(new EnrichedContentEvent { SchemaId = SchemaMatch }, trigger); Assert.False(result); }); } [Fact] public void Should_trigger_check_when_handling_all_events() { TestForTrigger(handleAll: true, schemaId: SchemaMatch, condition: null, action: trigger => { var result = handler.Trigger(new EnrichedContentEvent { SchemaId = SchemaMatch }, trigger); Assert.True(result); }); } [Fact] public void Should_trigger_check_when_condition_is_empty() { TestForTrigger(handleAll: false, schemaId: SchemaMatch, condition: string.Empty, action: trigger => { var result = handler.Trigger(new EnrichedContentEvent { SchemaId = SchemaMatch }, trigger); Assert.True(result); }); } [Fact] public void Should_trigger_check_when_condition_matchs() { TestForTrigger(handleAll: false, schemaId: SchemaMatch, condition: "true", action: trigger => { var result = handler.Trigger(new EnrichedContentEvent { SchemaId = SchemaMatch }, trigger); Assert.True(result); }); } [Fact] public void Should_not_trigger_check_when_schema_id_does_not_match() { TestForTrigger(handleAll: false, schemaId: SchemaNonMatch, condition: null, action: trigger => { var result = handler.Trigger(new EnrichedContentEvent { SchemaId = SchemaMatch }, trigger); Assert.False(result); }); } [Fact] public void Should_not_trigger_check_when_condition_does_not_matchs() { TestForTrigger(handleAll: false, schemaId: SchemaMatch, condition: "false", action: trigger => { var result = handler.Trigger(new EnrichedContentEvent { SchemaId = SchemaMatch }, trigger); Assert.False(result); }); } private void TestForTrigger(bool handleAll, NamedId? schemaId, string? condition, Action action) { var trigger = new ContentChangedTriggerV2 { HandleAll = handleAll }; if (schemaId != null) { trigger.Schemas = new ReadOnlyCollection(new List { new ContentChangedTriggerSchemaV2 { SchemaId = schemaId.Id, Condition = condition } }); } action(trigger); if (string.IsNullOrWhiteSpace(condition)) { A.CallTo(() => scriptEngine.Evaluate(A._, A._)) .MustNotHaveHappened(); } else { A.CallTo(() => scriptEngine.Evaluate(A._, condition)) .MustHaveHappened(); } } private void SetupReference(Guid referenceId) { A.CallTo(() => contentLoader.GetAsync(referenceId, EtagVersion.Any)) .Returns(new ContentEntity { Data = new NamedContentData() .AddField("field1", new ContentFieldData() .AddJsonValue(JsonValue.Create("Hello"))) .AddField("field2", new ContentFieldData() .AddJsonValue(JsonValue.Create("World"))) }); } } }