* Persist and expose Umbraco system dates as UTC (#19705) * Updated persistence DTOs defining default dates to use UTC. * Remove ForceToUtc = false from all persistence DTO attributes (default when not specified is true). * Removed use of SpecifyKind setting dates to local. * Removed unnecessary Utc suffixes on properties. * Persist current date time with UtcNow. * Removed further necessary Utc suffixes and fixed failing unit tests. * Added migration for SQL server to update database date default constraints. * Added comment justifying not providing a migration for SQLite default date constraints. * Ensure UTC for datetimes created from persistence DTOs. * Ensure UTC when creating dates for published content rendering in Razor and outputting in delivery API. * Fixed migration SQL syntax. * Introduced AuditItemFactory for creating entries for the backoffice document history, so we can control the UTC setting on the retrieved persisted dates. * Ensured UTC dates are retrieved for document versions. * Ensured UTC is returned for backoffice display of last edited and published for variant content. * Fixed SQLite syntax for default current datetime. * Apply suggestions from code review Co-authored-by: Laura Neto <12862535+lauraneto@users.noreply.github.com> * Further updates from code review. --------- Co-authored-by: Laura Neto <12862535+lauraneto@users.noreply.github.com> * Migrate system dates from local server time to UTC (#19798) * Add settings for the migration. * Add migration and implement for SQL server. * Implement for SQLite. * Fixes from testing with SQL Server. * Fixes from testing with SQLite. * Code tidy. * Cleaned up usings. * Removed audit log date from conversion. * Removed webhook log date from conversion. * Updated update date initialization on saving dictionary items. * Updated filter on log queries. * Use timezone ID instead of system name to work cross-culture. --------- Co-authored-by: Laura Neto <12862535+lauraneto@users.noreply.github.com>
1015 lines
39 KiB
C#
1015 lines
39 KiB
C#
// Copyright (c) Umbraco.
|
|
// See LICENSE for more details.
|
|
|
|
using System.Diagnostics;
|
|
using System.Globalization;
|
|
using System.Text.Json;
|
|
using Microsoft.Extensions.Logging;
|
|
using Microsoft.Extensions.Logging.Abstractions;
|
|
using Moq;
|
|
using NUnit.Framework;
|
|
using Umbraco.Cms.Core.Cache;
|
|
using Umbraco.Cms.Core.Logging;
|
|
using Umbraco.Cms.Core.Models;
|
|
using Umbraco.Cms.Core.Models.Entities;
|
|
using Umbraco.Cms.Core.PropertyEditors;
|
|
using Umbraco.Cms.Core.Services;
|
|
using Umbraco.Cms.Tests.Common.Builders;
|
|
using Umbraco.Cms.Tests.Common.Builders.Extensions;
|
|
using Umbraco.Cms.Tests.Common.Extensions;
|
|
using Umbraco.Cms.Tests.Common.TestHelpers.Stubs;
|
|
using Umbraco.Extensions;
|
|
|
|
namespace Umbraco.Cms.Tests.UnitTests.Umbraco.Core.Models;
|
|
|
|
[TestFixture]
|
|
public class ContentTests
|
|
{
|
|
private readonly IContentTypeService _contentTypeService = Mock.Of<IContentTypeService>();
|
|
|
|
private readonly PropertyEditorCollection _propertyEditorCollection = new (new DataEditorCollection(() => []));
|
|
|
|
[TestCase("name-fr", false)]
|
|
[TestCase("name-fr-updated", true)]
|
|
public void Variant_Culture_Names_Track_Dirty_Changes(string newName, bool expectedDirty)
|
|
{
|
|
var contentType = new ContentTypeBuilder()
|
|
.WithAlias("contentType")
|
|
.WithContentVariation(ContentVariation.Culture)
|
|
.Build();
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = new ContentBuilder()
|
|
.WithId(1)
|
|
.WithVersionId(1)
|
|
.WithName("content")
|
|
.WithContentType(contentType)
|
|
.Build();
|
|
|
|
const string langFr = "fr-FR";
|
|
|
|
Assert.IsFalse(content.IsPropertyDirty("CultureInfos")); // hasn't been changed
|
|
|
|
Thread.Sleep(500); // The "Date" wont be dirty if the test runs too fast since it will be the same date
|
|
content.SetCultureName("name-fr", langFr);
|
|
Assert.IsTrue(
|
|
content.IsPropertyDirty("CultureInfos")); // now it will be changed since the collection has changed
|
|
var frCultureName = content.CultureInfos[langFr];
|
|
Assert.IsTrue(frCultureName.IsPropertyDirty("Date"));
|
|
|
|
content.ResetDirtyProperties();
|
|
frCultureName.ResetDirtyProperties();
|
|
|
|
Assert.IsFalse(content.IsPropertyDirty("CultureInfos")); // it's been reset
|
|
Assert.IsTrue(content.WasPropertyDirty("CultureInfos"));
|
|
|
|
Thread.Sleep(500); // The "Date" wont be dirty if the test runs too fast since it will be the same date
|
|
content.SetCultureName(newName, langFr);
|
|
|
|
// dirty is only true if we updated the name
|
|
Assert.AreEqual(expectedDirty, frCultureName.IsPropertyDirty("Date"));
|
|
Assert.AreEqual(expectedDirty, content.IsPropertyDirty("CultureInfos"));
|
|
}
|
|
|
|
[Test]
|
|
public void Variant_Published_Culture_Names_Track_Dirty_Changes()
|
|
{
|
|
var contentType = new ContentTypeBuilder()
|
|
.WithAlias("contentType")
|
|
.WithContentVariation(ContentVariation.Culture)
|
|
.Build();
|
|
var content = new ContentBuilder()
|
|
.WithId(1)
|
|
.WithVersionId(1)
|
|
.WithName("content")
|
|
.WithContentType(contentType)
|
|
.Build();
|
|
|
|
const string langFr = "fr-FR";
|
|
|
|
content.ChangeContentType(contentType);
|
|
|
|
Assert.IsFalse(content.IsPropertyDirty("PublishCultureInfos")); // hasn't been changed
|
|
|
|
Thread.Sleep(500); // The "Date" wont be dirty if the test runs too fast since it will be the same date
|
|
content.SetCultureName("name-fr", langFr);
|
|
content.PublishCulture(CultureImpact.Explicit(langFr, false), DateTime.UtcNow, _propertyEditorCollection); // we've set the name, now we're publishing it
|
|
Assert.IsTrue(
|
|
content.IsPropertyDirty("PublishCultureInfos")); // now it will be changed since the collection has changed
|
|
var frCultureName = content.PublishCultureInfos[langFr];
|
|
Assert.IsTrue(frCultureName.IsPropertyDirty("Date"));
|
|
|
|
content.ResetDirtyProperties();
|
|
frCultureName.ResetDirtyProperties();
|
|
|
|
Assert.IsFalse(content.IsPropertyDirty("PublishCultureInfos")); // it's been reset
|
|
Assert.IsTrue(content.WasPropertyDirty("PublishCultureInfos"));
|
|
|
|
Thread.Sleep(500); // The "Date" wont be dirty if the test runs too fast since it will be the same date
|
|
content.SetCultureName("name-fr", langFr);
|
|
content.PublishCulture(CultureImpact.Explicit(langFr, false), DateTime.UtcNow, _propertyEditorCollection); // we've set the name, now we're publishing it
|
|
Assert.IsTrue(frCultureName.IsPropertyDirty("Date"));
|
|
Assert.IsTrue(content.IsPropertyDirty("PublishCultureInfos")); // it's true now since we've updated a name
|
|
}
|
|
|
|
[Test]
|
|
public void Get_Non_Grouped_Properties()
|
|
{
|
|
var contentType = ContentTypeBuilder.CreateSimpleContentType();
|
|
|
|
// Add non-grouped properties
|
|
var pt1 = new PropertyTypeBuilder()
|
|
.WithAlias("nonGrouped1")
|
|
.WithName("Non Grouped 1")
|
|
.Build();
|
|
var pt2 = new PropertyTypeBuilder()
|
|
.WithAlias("nonGrouped2")
|
|
.WithName("Non Grouped 2")
|
|
.Build();
|
|
contentType.AddPropertyType(pt1);
|
|
contentType.AddPropertyType(pt2);
|
|
|
|
// Ensure that nothing is marked as dirty
|
|
contentType.ResetDirtyProperties(false);
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateSimpleContent(contentType);
|
|
|
|
var nonGrouped = content.GetNonGroupedProperties();
|
|
|
|
Assert.AreEqual(2, nonGrouped.Count());
|
|
Assert.AreEqual(5, content.Properties.Count());
|
|
}
|
|
|
|
[Test]
|
|
public void All_Dirty_Properties_Get_Reset()
|
|
{
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
|
|
content.ResetDirtyProperties(false);
|
|
|
|
Assert.IsFalse(content.IsDirty());
|
|
foreach (var prop in content.Properties)
|
|
{
|
|
Assert.IsFalse(prop.IsDirty());
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Verify_Mocked_Content()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
|
|
// Act
|
|
|
|
// Assert
|
|
Assert.That(content, Is.Not.Null);
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Change_Property_Value()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
|
|
// Act
|
|
content.Properties["title"].SetValue("This is the new title");
|
|
|
|
// Assert
|
|
Assert.That(content.Properties.Any(), Is.True);
|
|
Assert.That(content.Properties["title"], Is.Not.Null);
|
|
Assert.That(content.Properties["title"].GetValue(), Is.EqualTo("This is the new title"));
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Set_Property_Value_As_String()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
|
|
// Act
|
|
content.SetValue("title", "This is the new title");
|
|
|
|
// Assert
|
|
Assert.That(content.Properties.Any(), Is.True);
|
|
Assert.That(content.Properties["title"], Is.Not.Null);
|
|
Assert.That(content.Properties["title"].GetValue(), Is.EqualTo("This is the new title"));
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Clone_Content_With_Reset_Identity()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
content.Id = 10;
|
|
content.Key = new Guid("29181B97-CB8F-403F-86DE-5FEB497F4800");
|
|
|
|
// Act
|
|
var clone = content.DeepCloneWithResetIdentities();
|
|
|
|
// Assert
|
|
Assert.AreNotSame(clone, content);
|
|
Assert.AreNotSame(clone.Id, content.Id);
|
|
Assert.AreNotSame(clone.VersionId, content.VersionId);
|
|
Assert.That(clone.HasIdentity, Is.False);
|
|
|
|
Assert.AreNotSame(content.Properties, clone.Properties);
|
|
}
|
|
|
|
private static IProfilingLogger GetTestProfilingLogger()
|
|
{
|
|
var logger = NullLoggerFactory.Instance.CreateLogger<ProfilingLogger>();
|
|
var profiler = new TestProfiler();
|
|
return new ProfilingLogger(logger, profiler);
|
|
}
|
|
|
|
[Ignore("TODO - ignored test")]
|
|
[Test]
|
|
public void Can_Deep_Clone_Perf_Test()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
contentType.Id = 99;
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
var i = 200;
|
|
foreach (var property in content.Properties)
|
|
{
|
|
property.Id = ++i;
|
|
}
|
|
|
|
content.Id = 10;
|
|
content.CreateDate = DateTime.UtcNow;
|
|
content.CreatorId = 22;
|
|
content.Key = Guid.NewGuid();
|
|
content.Level = 3;
|
|
content.Path = "-1,4,10";
|
|
content.SortOrder = 5;
|
|
content.TemplateId = 88;
|
|
content.Trashed = false;
|
|
content.UpdateDate = DateTime.UtcNow;
|
|
content.WriterId = 23;
|
|
|
|
var runtimeCache = new ObjectCacheAppCache();
|
|
runtimeCache.Insert(content.Id.ToString(CultureInfo.InvariantCulture), () => content);
|
|
|
|
var proflog = GetTestProfilingLogger();
|
|
|
|
using (proflog.DebugDuration<ContentTests>("STARTING PERF TEST WITH RUNTIME CACHE"))
|
|
{
|
|
for (var j = 0; j < 1000; j++)
|
|
{
|
|
var clone = runtimeCache.Get(content.Id.ToString(CultureInfo.InvariantCulture));
|
|
}
|
|
}
|
|
|
|
using (proflog.DebugDuration<ContentTests>("STARTING PERF TEST WITHOUT RUNTIME CACHE"))
|
|
{
|
|
for (var j = 0; j < 1000; j++)
|
|
{
|
|
var clone = (ContentType)contentType.DeepClone();
|
|
}
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Deep_Clone()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
contentType.Id = 99;
|
|
contentType.Variations = ContentVariation.Culture;
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
|
|
content.SetCultureName("Hello", "en-US");
|
|
content.SetCultureName("World", "es-ES");
|
|
content.PublishCulture(CultureImpact.All, DateTime.UtcNow, _propertyEditorCollection);
|
|
|
|
// should not try to clone something that's not Published or Unpublished
|
|
// (and in fact it will not work)
|
|
// but we cannot directly set the state to Published - hence this trick
|
|
// content.ChangePublishedState(PublishedState.Publishing);
|
|
content.ResetDirtyProperties(false); // => .Published
|
|
|
|
var i = 200;
|
|
foreach (var property in content.Properties)
|
|
{
|
|
property.Id = ++i;
|
|
}
|
|
|
|
content.Id = 10;
|
|
content.CreateDate = DateTime.UtcNow;
|
|
content.CreatorId = 22;
|
|
content.Key = Guid.NewGuid();
|
|
content.Level = 3;
|
|
content.Path = "-1,4,10";
|
|
content.SortOrder = 5;
|
|
content.TemplateId = 88;
|
|
content.Trashed = false;
|
|
content.UpdateDate = DateTime.UtcNow;
|
|
content.WriterId = 23;
|
|
|
|
// Act
|
|
var clone = (Content)content.DeepClone();
|
|
|
|
// Assert
|
|
Assert.AreNotSame(clone, content);
|
|
Assert.AreEqual(clone, content);
|
|
Assert.AreEqual(clone.Id, content.Id);
|
|
Assert.AreEqual(clone.VersionId, content.VersionId);
|
|
Assert.AreEqual(clone.ContentType, content.ContentType);
|
|
Assert.AreEqual(clone.ContentTypeId, content.ContentTypeId);
|
|
Assert.AreEqual(clone.CreateDate, content.CreateDate);
|
|
Assert.AreEqual(clone.CreatorId, content.CreatorId);
|
|
Assert.AreEqual(clone.Key, content.Key);
|
|
Assert.AreEqual(clone.Level, content.Level);
|
|
Assert.AreEqual(clone.Path, content.Path);
|
|
Assert.AreEqual(clone.Published, content.Published);
|
|
Assert.AreEqual(clone.PublishedState, content.PublishedState);
|
|
Assert.AreEqual(clone.SortOrder, content.SortOrder);
|
|
Assert.AreEqual(clone.PublishedState, content.PublishedState);
|
|
Assert.AreNotSame(clone.TemplateId, content.TemplateId);
|
|
Assert.AreEqual(clone.TemplateId, content.TemplateId);
|
|
Assert.AreEqual(clone.Trashed, content.Trashed);
|
|
Assert.AreEqual(clone.UpdateDate, content.UpdateDate);
|
|
Assert.AreEqual(clone.VersionId, content.VersionId);
|
|
Assert.AreEqual(clone.WriterId, content.WriterId);
|
|
Assert.AreNotSame(clone.Properties, content.Properties);
|
|
Assert.AreEqual(clone.Properties.Count(), content.Properties.Count());
|
|
for (var index = 0; index < content.Properties.Count; index++)
|
|
{
|
|
Assert.AreNotSame(clone.Properties[index], content.Properties[index]);
|
|
Assert.AreEqual(clone.Properties[index], content.Properties[index]);
|
|
}
|
|
|
|
Assert.AreNotSame(clone.PublishCultureInfos, content.PublishCultureInfos);
|
|
Assert.AreEqual(clone.PublishCultureInfos.Count, content.PublishCultureInfos.Count);
|
|
foreach (var key in content.PublishCultureInfos.Keys)
|
|
{
|
|
Assert.AreNotSame(clone.PublishCultureInfos[key], content.PublishCultureInfos[key]);
|
|
Assert.AreEqual(clone.PublishCultureInfos[key], content.PublishCultureInfos[key]);
|
|
}
|
|
|
|
Assert.AreNotSame(clone.CultureInfos, content.CultureInfos);
|
|
Assert.AreEqual(clone.CultureInfos.Count, content.CultureInfos.Count);
|
|
foreach (var key in content.CultureInfos.Keys)
|
|
{
|
|
Assert.AreNotSame(clone.CultureInfos[key], content.CultureInfos[key]);
|
|
Assert.AreEqual(clone.CultureInfos[key], content.CultureInfos[key]);
|
|
}
|
|
|
|
// This double verifies by reflection
|
|
var allProps = clone.GetType().GetProperties();
|
|
foreach (var propertyInfo in allProps)
|
|
{
|
|
Assert.AreEqual(propertyInfo.GetValue(clone, null), propertyInfo.GetValue(content, null));
|
|
}
|
|
|
|
// Need to ensure the event handlers are wired
|
|
var asDirty = (ICanBeDirty)clone;
|
|
|
|
Assert.IsFalse(asDirty.IsPropertyDirty("Properties"));
|
|
var propertyType = new PropertyTypeBuilder()
|
|
.WithAlias("blah")
|
|
.Build();
|
|
var newProperty = new PropertyBuilder()
|
|
.WithId(1)
|
|
.WithPropertyType(propertyType)
|
|
.Build();
|
|
newProperty.SetValue("blah");
|
|
clone.Properties.Add(newProperty);
|
|
|
|
Assert.IsTrue(asDirty.IsPropertyDirty("Properties"));
|
|
}
|
|
|
|
[Test]
|
|
public void Remember_Dirty_Properties()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
contentType.Id = 99;
|
|
contentType.Variations = ContentVariation.Culture;
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
|
|
content.SetCultureName("Hello", "en-US");
|
|
content.SetCultureName("World", "es-ES");
|
|
content.PublishCulture(CultureImpact.All, DateTime.UtcNow, _propertyEditorCollection);
|
|
|
|
var i = 200;
|
|
foreach (var property in content.Properties)
|
|
{
|
|
property.Id = ++i;
|
|
}
|
|
|
|
content.Id = 10;
|
|
content.CreateDate = DateTime.UtcNow;
|
|
content.CreatorId = 22;
|
|
content.Key = Guid.NewGuid();
|
|
content.Level = 3;
|
|
content.Path = "-1,4,10";
|
|
content.SortOrder = 5;
|
|
content.TemplateId = 88;
|
|
content.Trashed = true;
|
|
content.UpdateDate = DateTime.UtcNow;
|
|
content.WriterId = 23;
|
|
|
|
// Act
|
|
content.ResetDirtyProperties();
|
|
|
|
// Assert
|
|
Assert.IsTrue(content.WasDirty());
|
|
Assert.IsTrue(content.WasPropertyDirty(nameof(Content.Id)));
|
|
Assert.IsTrue(content.WasPropertyDirty(nameof(Content.CreateDate)));
|
|
Assert.IsTrue(content.WasPropertyDirty(nameof(Content.CreatorId)));
|
|
Assert.IsTrue(content.WasPropertyDirty(nameof(Content.Key)));
|
|
Assert.IsTrue(content.WasPropertyDirty(nameof(Content.Level)));
|
|
Assert.IsTrue(content.WasPropertyDirty(nameof(Content.Path)));
|
|
Assert.IsTrue(content.WasPropertyDirty(nameof(Content.SortOrder)));
|
|
Assert.IsTrue(content.WasPropertyDirty(nameof(Content.TemplateId)));
|
|
Assert.IsTrue(content.WasPropertyDirty(nameof(Content.Trashed)));
|
|
Assert.IsTrue(content.WasPropertyDirty(nameof(Content.UpdateDate)));
|
|
Assert.IsTrue(content.WasPropertyDirty(nameof(Content.WriterId)));
|
|
foreach (var prop in content.Properties)
|
|
{
|
|
Assert.IsTrue(prop.WasDirty());
|
|
Assert.IsTrue(prop.WasPropertyDirty("Id"));
|
|
}
|
|
|
|
Assert.IsTrue(content.WasPropertyDirty("CultureInfos"));
|
|
foreach (var culture in content.CultureInfos)
|
|
{
|
|
Assert.IsTrue(culture.WasDirty());
|
|
Assert.IsTrue(culture.WasPropertyDirty("Name"));
|
|
Assert.IsTrue(culture.WasPropertyDirty("Date"));
|
|
}
|
|
|
|
Assert.IsTrue(content.WasPropertyDirty("PublishCultureInfos"));
|
|
foreach (var culture in content.PublishCultureInfos)
|
|
{
|
|
Assert.IsTrue(culture.WasDirty());
|
|
Assert.IsTrue(culture.WasPropertyDirty("Name"));
|
|
Assert.IsTrue(culture.WasPropertyDirty("Date"));
|
|
}
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Serialize_Without_Error()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
contentType.Id = 99;
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
var i = 200;
|
|
foreach (var property in content.Properties)
|
|
{
|
|
property.Id = ++i;
|
|
}
|
|
|
|
content.Id = 10;
|
|
content.CreateDate = DateTime.UtcNow;
|
|
content.CreatorId = 22;
|
|
content.Key = Guid.NewGuid();
|
|
content.Level = 3;
|
|
content.Path = "-1,4,10";
|
|
content.SortOrder = 5;
|
|
content.TemplateId = 88;
|
|
content.Trashed = false;
|
|
content.UpdateDate = DateTime.UtcNow;
|
|
content.WriterId = 23;
|
|
|
|
var json = JsonSerializer.Serialize(content);
|
|
Debug.Print(json);
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Change_Property_Value_Through_Anonymous_Object()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
|
|
// Act
|
|
content.PropertyValues(new { title = "This is the new title" });
|
|
|
|
// Assert
|
|
Assert.That(content.Properties.Any(), Is.True);
|
|
Assert.That(content.Properties["title"], Is.Not.Null);
|
|
Assert.That(content.Properties["title"].Alias, Is.EqualTo("title"));
|
|
Assert.That(content.Properties["title"].GetValue(), Is.EqualTo("This is the new title"));
|
|
Assert.That(content.Properties["description"].GetValue(), Is.EqualTo("This is the meta description for a textpage"));
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Verify_Dirty_Property_On_Content()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
|
|
// Act
|
|
content.ResetDirtyProperties();
|
|
content.Name = "New Home";
|
|
|
|
// Assert
|
|
Assert.That(content.Name, Is.EqualTo("New Home"));
|
|
Assert.That(content.IsPropertyDirty("Name"), Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Add_PropertyGroup_On_ContentType()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
|
|
// Act
|
|
contentType.PropertyGroups.Add(
|
|
new PropertyGroup(true) { Alias = "testGroup", Name = "Test Group", SortOrder = 3 });
|
|
|
|
// Assert
|
|
Assert.That(contentType.PropertyGroups.Count, Is.EqualTo(3));
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Remove_PropertyGroup_From_ContentType()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
contentType.ResetDirtyProperties();
|
|
|
|
// Act
|
|
contentType.PropertyGroups.Remove("content");
|
|
|
|
// Assert
|
|
Assert.That(contentType.PropertyGroups.Count, Is.EqualTo(1));
|
|
//// Assert.That(contentType.IsPropertyDirty("PropertyGroups"), Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Add_PropertyType_To_Group_On_ContentType()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
|
|
// Act
|
|
var propertyType = new PropertyTypeBuilder()
|
|
.WithAlias("subtitle")
|
|
.WithName("Subtitle")
|
|
.Build();
|
|
contentType.PropertyGroups["content"].PropertyTypes.Add(propertyType);
|
|
|
|
// Assert
|
|
Assert.That(contentType.PropertyGroups["content"].PropertyTypes.Count, Is.EqualTo(3));
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Add_New_Property_To_New_PropertyType()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
|
|
// Act
|
|
var propertyType = new PropertyTypeBuilder()
|
|
.WithAlias("subtitle")
|
|
.WithName("Subtitle")
|
|
.Build();
|
|
contentType.PropertyGroups["content"].PropertyTypes.Add(propertyType);
|
|
var newProperty = new Property(propertyType);
|
|
newProperty.SetValue("This is a subtitle Test");
|
|
content.Properties.Add(newProperty);
|
|
|
|
// Assert
|
|
Assert.That(content.Properties.Contains("subtitle"), Is.True);
|
|
Assert.That(content.Properties["subtitle"].GetValue(), Is.EqualTo("This is a subtitle Test"));
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Add_New_Property_To_New_PropertyType_In_New_PropertyGroup()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
|
|
// Act
|
|
var propertyType = new PropertyTypeBuilder()
|
|
.WithAlias("subtitle")
|
|
.WithName("Subtitle")
|
|
.Build();
|
|
var propertyGroup = new PropertyGroup(true) { Alias = "testGroup", Name = "Test Group", SortOrder = 3 };
|
|
propertyGroup.PropertyTypes.Add(propertyType);
|
|
contentType.PropertyGroups.Add(propertyGroup);
|
|
var newProperty = new Property(propertyType);
|
|
newProperty.SetValue("Subtitle Test");
|
|
content.Properties.Add(newProperty);
|
|
|
|
// Assert
|
|
Assert.That(content.Properties.Count, Is.EqualTo(5));
|
|
Assert.That(content.Properties["subtitle"].GetValue(), Is.EqualTo("Subtitle Test"));
|
|
Assert.That(content.Properties["title"].GetValue(), Is.EqualTo("Textpage textpage"));
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Update_PropertyType_Through_Content_Properties()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
|
|
// Act - note that the PropertyType's properties like SortOrder is not updated through the Content object
|
|
var propertyType = new PropertyTypeBuilder()
|
|
.WithAlias("title")
|
|
.WithName("Title")
|
|
.Build();
|
|
content.Properties.Add(new Property(propertyType));
|
|
|
|
// Assert
|
|
Assert.That(content.Properties.Count, Is.EqualTo(4));
|
|
Assert.That(contentType.PropertyTypes.First(x => x.Alias == "title").SortOrder, Is.EqualTo(1));
|
|
Assert.That(content.Properties["title"].GetValue(), Is.EqualTo("Textpage textpage"));
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Change_ContentType_On_Content()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
var simpleContentType = ContentTypeBuilder.CreateSimpleContentType();
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
|
|
// Act
|
|
content.ChangeContentType(simpleContentType);
|
|
|
|
// Assert
|
|
Assert.That(content.Properties.Contains("author"), Is.True);
|
|
|
|
// Note: There were 4 properties, after changing ContentType 1 has been added (no properties are deleted).
|
|
Assert.That(content.Properties.Count, Is.EqualTo(5));
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Change_ContentType_On_Content_And_Set_Property_Value()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
var simpleContentType = ContentTypeBuilder.CreateSimpleContentType();
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
|
|
// Act
|
|
content.ChangeContentType(simpleContentType);
|
|
content.SetValue("author", "John Doe");
|
|
|
|
// Assert
|
|
Assert.That(content.Properties.Contains("author"), Is.True);
|
|
Assert.That(content.Properties["author"].GetValue(), Is.EqualTo("John Doe"));
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Change_ContentType_On_Content_And_Still_Get_Old_Properties()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
var simpleContentType = ContentTypeBuilder.CreateSimpleContentType();
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
|
|
// Act
|
|
content.ChangeContentType(simpleContentType);
|
|
|
|
// Assert
|
|
Assert.That(content.Properties.Contains("author"), Is.True);
|
|
Assert.That(content.Properties.Contains("keywords"), Is.True);
|
|
Assert.That(content.Properties.Contains("description"), Is.True);
|
|
Assert.That(content.Properties["keywords"].GetValue(), Is.EqualTo("text,page,meta"));
|
|
Assert.That(content.Properties["description"].GetValue(), Is.EqualTo("This is the meta description for a textpage"));
|
|
}
|
|
|
|
[Test]
|
|
[Ignore("Need to reimplement this logic for v8")]
|
|
public void Can_Change_ContentType_On_Content_And_Clear_Old_PropertyTypes() => throw new NotImplementedException();
|
|
|
|
//// Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>())).Returns(contentType);
|
|
////// Arrange
|
|
//// var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
//// var simpleContentType = ContentTypeBuilder.CreateSimpleContentType();
|
|
//// var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
////// Act
|
|
//// content.ChangeContentType(simpleContentType, true);
|
|
////// Assert
|
|
//// Assert.That(content.Properties.Contains("author"), Is.True);
|
|
//// Assert.That(content.Properties.Contains("keywords"), Is.False);
|
|
//// Assert.That(content.Properties.Contains("description"), Is.False);
|
|
[Test]
|
|
public void Can_Verify_Content_Is_Published()
|
|
{
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "Textpage", -1);
|
|
|
|
content.ResetDirtyProperties();
|
|
content.PublishedState = PublishedState.Publishing;
|
|
|
|
Assert.IsFalse(content.IsPropertyDirty("Published"));
|
|
Assert.IsFalse(content.Published);
|
|
Assert.IsFalse(content.IsPropertyDirty("Name"));
|
|
Assert.AreEqual(PublishedState.Publishing, content.PublishedState);
|
|
|
|
// the repo would do
|
|
content.Published = true;
|
|
|
|
// and then
|
|
Assert.IsTrue(content.IsPropertyDirty("Published"));
|
|
Assert.IsTrue(content.Published);
|
|
Assert.IsFalse(content.IsPropertyDirty("Name"));
|
|
Assert.AreEqual(PublishedState.Published, content.PublishedState);
|
|
|
|
// and before returning,
|
|
content.ResetDirtyProperties();
|
|
|
|
// and then
|
|
Assert.IsFalse(content.IsPropertyDirty("Published"));
|
|
Assert.IsTrue(content.Published);
|
|
Assert.IsFalse(content.IsPropertyDirty("Name"));
|
|
Assert.AreEqual(PublishedState.Published, content.PublishedState);
|
|
}
|
|
|
|
[Test]
|
|
public void Adding_PropertyGroup_To_ContentType_Results_In_Dirty_Entity()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
contentType.ResetDirtyProperties();
|
|
|
|
// Act
|
|
var propertyGroup = new PropertyGroup(true) { Alias = "testGroup", Name = "Test Group", SortOrder = 3 };
|
|
contentType.PropertyGroups.Add(propertyGroup);
|
|
|
|
// Assert
|
|
Assert.That(contentType.IsDirty(), Is.True);
|
|
Assert.That(contentType.PropertyGroups.Any(x => x.Name == "Test Group"), Is.True);
|
|
//// Assert.That(contentType.IsPropertyDirty("PropertyGroups"), Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void After_Committing_Changes_Was_Dirty_Is_True()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
contentType.ResetDirtyProperties(); // reset
|
|
|
|
// Act
|
|
contentType.Alias = "newAlias";
|
|
contentType.ResetDirtyProperties(); // this would be like committing the entity
|
|
|
|
// Assert
|
|
Assert.That(contentType.IsDirty(), Is.False);
|
|
Assert.That(contentType.WasDirty(), Is.True);
|
|
Assert.That(contentType.WasPropertyDirty("Alias"), Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void After_Committing_Changes_Was_Dirty_Is_True_On_Changed_Property()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
contentType.ResetDirtyProperties(); // reset
|
|
Mock.Get(_contentTypeService).As<IContentTypeBaseService>().Setup(x => x.Get(It.IsAny<int>()))
|
|
.Returns(contentType);
|
|
|
|
var content = ContentBuilder.CreateTextpageContent(contentType, "test", -1);
|
|
content.ResetDirtyProperties();
|
|
|
|
// Act
|
|
content.SetValue("title", "new title");
|
|
Assert.That(content.IsEntityDirty(), Is.False);
|
|
Assert.That(content.IsDirty(), Is.True);
|
|
Assert.That(content.IsPropertyDirty("title"), Is.True);
|
|
Assert.That(content.IsAnyUserPropertyDirty(), Is.True);
|
|
Assert.That(content.GetDirtyUserProperties().Count(), Is.EqualTo(1));
|
|
Assert.That(content.Properties[0].IsDirty(), Is.True);
|
|
Assert.That(content.Properties["title"].IsDirty(), Is.True);
|
|
|
|
content.ResetDirtyProperties(); // this would be like committing the entity
|
|
|
|
// Assert
|
|
Assert.That(content.WasDirty(), Is.True);
|
|
Assert.That(content.Properties[0].WasDirty(), Is.True);
|
|
|
|
Assert.That(content.WasPropertyDirty("title"), Is.True);
|
|
Assert.That(content.Properties["title"].IsDirty(), Is.False);
|
|
Assert.That(content.Properties["title"].WasDirty(), Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void If_Not_Committed_Was_Dirty_Is_False()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
|
|
// Act
|
|
contentType.Alias = "newAlias";
|
|
|
|
// Assert
|
|
Assert.That(contentType.IsDirty(), Is.True);
|
|
Assert.That(contentType.WasDirty(), Is.False);
|
|
}
|
|
|
|
[Test]
|
|
public void Detect_That_A_Property_Is_Removed()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
Assert.That(contentType.WasPropertyDirty("HasPropertyTypeBeenRemoved"), Is.False);
|
|
|
|
// Act
|
|
contentType.RemovePropertyType("title");
|
|
|
|
// Assert
|
|
Assert.That(contentType.IsPropertyDirty("HasPropertyTypeBeenRemoved"), Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void Adding_PropertyType_To_PropertyGroup_On_ContentType_Results_In_Dirty_Entity()
|
|
{
|
|
// Arrange
|
|
var contentType = ContentTypeBuilder.CreateTextPageContentType();
|
|
contentType.ResetDirtyProperties();
|
|
|
|
// Act
|
|
var propertyType = new PropertyTypeBuilder()
|
|
.WithAlias("subtitle")
|
|
.WithName("Subtitle")
|
|
.Build();
|
|
contentType.PropertyGroups["content"].PropertyTypes.Add(propertyType);
|
|
|
|
// Assert
|
|
Assert.That(contentType.PropertyGroups["content"].IsDirty(), Is.True);
|
|
Assert.That(contentType.PropertyGroups["content"].IsPropertyDirty("PropertyTypes"), Is.True);
|
|
Assert.That(contentType.PropertyGroups.Any(x => x.IsDirty()), Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Compose_Composite_ContentType_Collection()
|
|
{
|
|
// Arrange
|
|
var simpleContentType = ContentTypeBuilder.CreateSimpleContentType();
|
|
var propertyType = new PropertyTypeBuilder()
|
|
.WithAlias("coauthor")
|
|
.WithName("Co-author")
|
|
.Build();
|
|
var simple2ContentType = ContentTypeBuilder.CreateSimpleContentType(
|
|
"anotherSimple",
|
|
"Another Simple Page",
|
|
propertyTypeCollection: new PropertyTypeCollection(true, new List<PropertyType> { propertyType }));
|
|
|
|
// Act
|
|
var added = simpleContentType.AddContentType(simple2ContentType);
|
|
var compositionPropertyGroups = simpleContentType.CompositionPropertyGroups;
|
|
var compositionPropertyTypes = simpleContentType.CompositionPropertyTypes;
|
|
|
|
// Assert
|
|
Assert.That(added, Is.True);
|
|
Assert.That(compositionPropertyGroups.Count(), Is.EqualTo(1));
|
|
Assert.That(compositionPropertyTypes.Count(), Is.EqualTo(4));
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Compose_Nested_Composite_ContentType_Collection()
|
|
{
|
|
// Arrange
|
|
var metaContentType = ContentTypeBuilder.CreateMetaContentType();
|
|
var simpleContentType = ContentTypeBuilder.CreateSimpleContentType();
|
|
var propertyType = new PropertyTypeBuilder()
|
|
.WithAlias("coauthor")
|
|
.WithName("Co-author")
|
|
.Build();
|
|
var simple2ContentType = ContentTypeBuilder.CreateSimpleContentType(
|
|
"anotherSimple",
|
|
"Another Simple Page",
|
|
propertyTypeCollection: new PropertyTypeCollection(true, new List<PropertyType> { propertyType }));
|
|
|
|
// Act
|
|
var addedMeta = simple2ContentType.AddContentType(metaContentType);
|
|
var added = simpleContentType.AddContentType(simple2ContentType);
|
|
var compositionPropertyGroups = simpleContentType.CompositionPropertyGroups;
|
|
var compositionPropertyTypes = simpleContentType.CompositionPropertyTypes;
|
|
|
|
// Assert
|
|
Assert.That(addedMeta, Is.True);
|
|
Assert.That(added, Is.True);
|
|
Assert.That(compositionPropertyGroups.Count(), Is.EqualTo(2));
|
|
Assert.That(compositionPropertyTypes.Count(), Is.EqualTo(6));
|
|
Assert.That(simpleContentType.ContentTypeCompositionExists("meta"), Is.True);
|
|
}
|
|
|
|
[Test]
|
|
public void Can_Avoid_Circular_Dependencies_In_Composition()
|
|
{
|
|
var textPage = ContentTypeBuilder.CreateTextPageContentType();
|
|
var parent = ContentTypeBuilder.CreateSimpleContentType("parent", "Parent", null, randomizeAliases: true);
|
|
var meta = ContentTypeBuilder.CreateMetaContentType();
|
|
var propertyType1 = new PropertyTypeBuilder()
|
|
.WithAlias("coauthor")
|
|
.WithName("Co-author")
|
|
.Build();
|
|
var mixin1 = ContentTypeBuilder.CreateSimpleContentType(
|
|
"mixin1",
|
|
"Mixin1",
|
|
propertyTypeCollection: new PropertyTypeCollection(true, new List<PropertyType> { propertyType1 }));
|
|
var propertyType2 = new PropertyTypeBuilder()
|
|
.WithAlias("author")
|
|
.WithName("Author")
|
|
.Build();
|
|
var mixin2 = ContentTypeBuilder.CreateSimpleContentType(
|
|
"mixin2",
|
|
"Mixin2",
|
|
propertyTypeCollection: new PropertyTypeCollection(true, new List<PropertyType> { propertyType2 }));
|
|
|
|
// Act
|
|
var addedMetaMixin2 = mixin2.AddContentType(meta);
|
|
var addedMixin2 = mixin1.AddContentType(mixin2);
|
|
var addedMeta = parent.AddContentType(meta);
|
|
|
|
var addedMixin1 = parent.AddContentType(mixin1);
|
|
|
|
var addedMixin1Textpage = textPage.AddContentType(mixin1);
|
|
var addedTextpageParent = parent.AddContentType(textPage);
|
|
|
|
var aliases = textPage.CompositionAliases();
|
|
var propertyTypes = textPage.CompositionPropertyTypes;
|
|
var propertyGroups = textPage.CompositionPropertyGroups;
|
|
|
|
// Assert
|
|
Assert.That(mixin2.ContentTypeCompositionExists("meta"), Is.True);
|
|
Assert.That(mixin1.ContentTypeCompositionExists("meta"), Is.True);
|
|
Assert.That(parent.ContentTypeCompositionExists("meta"), Is.True);
|
|
Assert.That(textPage.ContentTypeCompositionExists("meta"), Is.True);
|
|
|
|
Assert.That(aliases.Count(), Is.EqualTo(3));
|
|
Assert.That(propertyTypes.Count(), Is.EqualTo(8));
|
|
Assert.That(propertyGroups.Count(), Is.EqualTo(2));
|
|
|
|
Assert.That(addedMeta, Is.True);
|
|
Assert.That(addedMetaMixin2, Is.True);
|
|
Assert.That(addedMixin2, Is.True);
|
|
Assert.That(addedMixin1, Is.False);
|
|
Assert.That(addedMixin1Textpage, Is.True);
|
|
Assert.That(addedTextpageParent, Is.False);
|
|
}
|
|
}
|