// ========================================================================== // Squidex Headless CMS // ========================================================================== // Copyright (c) Squidex UG (haftungsbeschraenkt) // All rights reserved. Licensed under the MIT license. // ========================================================================== using Microsoft.Extensions.Logging; using Squidex.Domain.Apps.Core.EventSynchronization; using Squidex.Domain.Apps.Core.Schemas; using Squidex.Domain.Apps.Entities.Schemas.Commands; using Squidex.Domain.Apps.Entities.Schemas.DomainObject.Guards; using Squidex.Domain.Apps.Events; using Squidex.Domain.Apps.Events.Schemas; using Squidex.Infrastructure; using Squidex.Infrastructure.Commands; using Squidex.Infrastructure.EventSourcing; using Squidex.Infrastructure.Orleans; using Squidex.Infrastructure.Reflection; using Squidex.Infrastructure.States; namespace Squidex.Domain.Apps.Entities.Schemas.DomainObject { public sealed partial class SchemaDomainObject : DomainObject { public SchemaDomainObject(IPersistenceFactory persistence, ILogger log) : base(persistence, log) { } protected override bool IsDeleted(State snapshot) { return snapshot.IsDeleted; } protected override bool CanAcceptCreation(ICommand command) { return command is SchemaCommand; } protected override bool CanAccept(ICommand command) { return command is SchemaUpdateCommand schemaCommand && Equals(schemaCommand.AppId, Snapshot.AppId) && Equals(schemaCommand.SchemaId?.Id, Snapshot.Id); } public override Task ExecuteAsync(IAggregateCommand command) { switch (command) { case AddField addField: return UpdateReturn(addField, c => { GuardSchemaField.CanAdd(c, Snapshot.SchemaDef); AddField(c); return Snapshot; }); case CreateSchema createSchema: return CreateReturn(createSchema, c => { GuardSchema.CanCreate(c); Create(c); return Snapshot; }); case SynchronizeSchema synchronize: return UpdateReturn(synchronize, c => { GuardSchema.CanSynchronize(c); Synchronize(c); return Snapshot; }); case DeleteField deleteField: return UpdateReturn(deleteField, c => { GuardSchemaField.CanDelete(deleteField, Snapshot.SchemaDef); DeleteField(c); return Snapshot; }); case LockField lockField: return UpdateReturn(lockField, c => { GuardSchemaField.CanLock(lockField, Snapshot.SchemaDef); LockField(c); return Snapshot; }); case HideField hideField: return UpdateReturn(hideField, c => { GuardSchemaField.CanHide(c, Snapshot.SchemaDef); HideField(c); return Snapshot; }); case ShowField showField: return UpdateReturn(showField, c => { GuardSchemaField.CanShow(c, Snapshot.SchemaDef); ShowField(c); return Snapshot; }); case DisableField disableField: return UpdateReturn(disableField, c => { GuardSchemaField.CanDisable(c, Snapshot.SchemaDef); DisableField(c); return Snapshot; }); case EnableField enableField: return UpdateReturn(enableField, c => { GuardSchemaField.CanEnable(c, Snapshot.SchemaDef); EnableField(c); return Snapshot; }); case UpdateField updateField: return UpdateReturn(updateField, c => { GuardSchemaField.CanUpdate(c, Snapshot.SchemaDef); UpdateField(c); return Snapshot; }); case ReorderFields reorderFields: return UpdateReturn(reorderFields, c => { GuardSchema.CanReorder(c, Snapshot.SchemaDef); Reorder(c); return Snapshot; }); case ConfigureFieldRules configureFieldRules: return UpdateReturn(configureFieldRules, c => { GuardSchema.CanConfigureFieldRules(c); ConfigureFieldRules(c); return Snapshot; }); case ConfigurePreviewUrls configurePreviewUrls: return UpdateReturn(configurePreviewUrls, c => { GuardSchema.CanConfigurePreviewUrls(c); ConfigurePreviewUrls(c); return Snapshot; }); case ConfigureUIFields configureUIFields: return UpdateReturn(configureUIFields, c => { GuardSchema.CanConfigureUIFields(c, Snapshot.SchemaDef); ConfigureUIFields(c); return Snapshot; }); case ChangeCategory changeCategory: return UpdateReturn(changeCategory, c => { ChangeCategory(c); return Snapshot; }); case UpdateSchema update: return UpdateReturn(update, c => { Update(c); return Snapshot; }); case PublishSchema publish: return UpdateReturn(publish, c => { Publish(c); return Snapshot; }); case UnpublishSchema unpublish: return UpdateReturn(unpublish, c => { Unpublish(c); return Snapshot; }); case ConfigureScripts configureScripts: return UpdateReturn(configureScripts, c => { ConfigureScripts(c); return Snapshot; }); case DeleteSchema deleteSchema: return Update(deleteSchema, c => { Delete(c); }); default: throw new NotSupportedException(); } } private void Synchronize(SynchronizeSchema command) { var options = new SchemaSynchronizationOptions { NoFieldDeletion = command.NoFieldDeletion, NoFieldRecreation = command.NoFieldRecreation }; var schemaSource = Snapshot.SchemaDef; var schemaTarget = command.BuildSchema(schemaSource.Name, schemaSource.Type); var events = schemaSource.Synchronize(schemaTarget, () => Snapshot.SchemaFieldsTotal + 1, options); foreach (var @event in events) { Raise(command, @event); } } private void Create(CreateSchema command) { Raise(command, new SchemaCreated { SchemaId = NamedId.Of(command.SchemaId, command.Name), Schema = command.BuildSchema() }); } private void AddField(AddField command) { Raise(command, new FieldAdded { FieldId = CreateFieldId(command) }); } private void UpdateField(UpdateField command) { Raise(command, new FieldUpdated()); } private void LockField(LockField command) { Raise(command, new FieldLocked()); } private void HideField(HideField command) { Raise(command, new FieldHidden()); } private void ShowField(ShowField command) { Raise(command, new FieldShown()); } private void DisableField(DisableField command) { Raise(command, new FieldDisabled()); } private void EnableField(EnableField command) { Raise(command, new FieldEnabled()); } private void DeleteField(DeleteField command) { Raise(command, new FieldDeleted()); } private void Reorder(ReorderFields command) { Raise(command, new SchemaFieldsReordered()); } private void Publish(PublishSchema command) { Raise(command, new SchemaPublished()); } private void Unpublish(UnpublishSchema command) { Raise(command, new SchemaUnpublished()); } private void ConfigureScripts(ConfigureScripts command) { Raise(command, new SchemaScriptsConfigured()); } private void ConfigureFieldRules(ConfigureFieldRules command) { Raise(command, new SchemaFieldRulesConfigured { FieldRules = command.ToFieldRules() }); } private void ChangeCategory(ChangeCategory command) { Raise(command, new SchemaCategoryChanged()); } private void ConfigurePreviewUrls(ConfigurePreviewUrls command) { Raise(command, new SchemaPreviewUrlsConfigured()); } private void ConfigureUIFields(ConfigureUIFields command) { Raise(command, new SchemaUIFieldsConfigured()); } private void Update(UpdateSchema command) { Raise(command, new SchemaUpdated()); } private void Delete(DeleteSchema command) { Raise(command, new SchemaDeleted()); } private void Raise(T command, TEvent @event) where TEvent : SchemaEvent where T : class { SimpleMapper.Map(command, @event); NamedId? GetFieldId(long? id) { if (id != null && Snapshot.SchemaDef.FieldsById.TryGetValue(id.Value, out var field)) { return field.NamedId(); } return null; } if (command is ParentFieldCommand parentField && @event is ParentFieldEvent parentFieldEvent) { if (parentField.ParentFieldId != null) { if (Snapshot.SchemaDef.FieldsById.TryGetValue(parentField.ParentFieldId.Value, out var field)) { parentFieldEvent.ParentFieldId = field.NamedId(); if (command is FieldCommand fc && @event is FieldEvent fe) { if (field is IArrayField arrayField && arrayField.FieldsById.TryGetValue(fc.FieldId, out var nestedField)) { fe.FieldId = nestedField.NamedId(); } } } } else if (command is FieldCommand fc && @event is FieldEvent fe) { fe.FieldId = GetFieldId(fc.FieldId)!; } } SimpleMapper.Map(command, @event); @event.AppId ??= Snapshot.AppId; @event.SchemaId ??= Snapshot.NamedId(); RaiseEvent(Envelope.Create(@event)); } private NamedId CreateFieldId(AddField command) { return NamedId.Of(Snapshot.SchemaFieldsTotal + 1, command.Name); } public Task> GetStateAsync() { return J.AsTask(Snapshot); } } }