Files
Umbraco-CMS/tests/Umbraco.Tests.Integration/Umbraco.Infrastructure/Persistence/Repositories/NotificationsRepositoryTest.cs
Andy Butland d623476902 Use UTC for system dates in Umbraco (#19822)
* 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>
2025-08-22 11:59:23 +02:00

281 lines
9.9 KiB
C#

// Copyright (c) Umbraco.
// See LICENSE for more details.
using System.Globalization;
using System.Linq;
using Moq;
using NUnit.Framework;
using Umbraco.Cms.Core;
using Umbraco.Cms.Core.Models.Entities;
using Umbraco.Cms.Core.Models.Membership;
using Umbraco.Cms.Infrastructure.Persistence.Dtos;
using Umbraco.Cms.Infrastructure.Persistence.Repositories.Implement;
using Umbraco.Cms.Infrastructure.Scoping;
using Umbraco.Cms.Tests.Common.Testing;
using Umbraco.Cms.Tests.Integration.Testing;
namespace Umbraco.Cms.Tests.Integration.Umbraco.Infrastructure.Persistence.Repositories;
[TestFixture]
[UmbracoTest(Database = UmbracoTestOptions.Database.NewSchemaPerTest)]
internal sealed class NotificationsRepositoryTest : UmbracoIntegrationTest
{
[Test]
public void CreateNotification()
{
var provider = ScopeProvider;
using (var scope = provider.CreateScope())
{
var repo = new NotificationsRepository((IScopeAccessor)provider);
var node = new NodeDto // create bogus item so we can add a notification
{
CreateDate = DateTime.UtcNow,
Level = 1,
NodeObjectType = Constants.ObjectTypes.ContentItem,
ParentId = -1,
Path = "-1,123",
SortOrder = 1,
Text = "hello",
Trashed = false,
UniqueId = Guid.NewGuid(),
UserId = Constants.Security.SuperUserId
};
var result = ScopeAccessor.AmbientScope.Database.Insert(node);
var entity = Mock.Of<IEntity>(e => e.Id == node.NodeId);
var user = Mock.Of<IUser>(e => e.Id == node.UserId);
var notification = repo.CreateNotification(user, entity, "A");
Assert.AreEqual("A", notification.Action);
Assert.AreEqual(node.NodeId, notification.EntityId);
Assert.AreEqual(node.NodeObjectType, notification.EntityType);
Assert.AreEqual(node.UserId, notification.UserId);
}
}
[Test]
public void GetUserNotifications()
{
var provider = ScopeProvider;
using (var scope = provider.CreateScope())
{
var repo = new NotificationsRepository((IScopeAccessor)provider);
var userDto = new UserDto
{
Email = "test",
Login = "test",
Password = "test",
UserName = "test",
UserLanguage = "en",
CreateDate = DateTime.UtcNow,
UpdateDate = DateTime.UtcNow,
};
ScopeAccessor.AmbientScope.Database.Insert(userDto);
var userNew = Mock.Of<IUser>(e => e.Id == userDto.Id);
var userAdmin = Mock.Of<IUser>(e => e.Id == Constants.Security.SuperUserId);
for (var i = 0; i < 10; i++)
{
var node = new NodeDto
{
CreateDate = DateTime.UtcNow,
Level = 1,
NodeObjectType = Constants.ObjectTypes.ContentItem,
ParentId = -1,
Path = "-1," + i,
SortOrder = 1,
Text = "hello" + i,
Trashed = false,
UniqueId = Guid.NewGuid(),
UserId = -1
};
var result = ScopeAccessor.AmbientScope.Database.Insert(node);
var entity = Mock.Of<IEntity>(e => e.Id == node.NodeId);
var notification = repo.CreateNotification(i % 2 == 0 ? userAdmin : userNew, entity, i.ToString(CultureInfo.InvariantCulture));
}
var notifications = repo.GetUserNotifications(userAdmin);
Assert.AreEqual(5, notifications.Count());
}
}
[Test]
public void GetEntityNotifications()
{
var provider = ScopeProvider;
using (var scope = provider.CreateScope())
{
var repo = new NotificationsRepository((IScopeAccessor)provider);
var node1 = new NodeDto
{
CreateDate = DateTime.UtcNow,
Level = 1,
NodeObjectType = Constants.ObjectTypes.ContentItem,
ParentId = -1,
Path = "-1,1",
SortOrder = 1,
Text = "hello1",
Trashed = false,
UniqueId = Guid.NewGuid(),
UserId = -1
};
ScopeAccessor.AmbientScope.Database.Insert(node1);
var entity1 = Mock.Of<IEntity>(e => e.Id == node1.NodeId);
var node2 = new NodeDto
{
CreateDate = DateTime.UtcNow,
Level = 1,
NodeObjectType = Constants.ObjectTypes.ContentItem,
ParentId = -1,
Path = "-1,2",
SortOrder = 1,
Text = "hello2",
Trashed = false,
UniqueId = Guid.NewGuid(),
UserId = -1
};
ScopeAccessor.AmbientScope.Database.Insert(node2);
var entity2 = Mock.Of<IEntity>(e => e.Id == node2.NodeId);
for (var i = 0; i < 10; i++)
{
var userDto = new UserDto
{
Key = Guid.NewGuid(),
Email = "test" + i,
Login = "test" + i,
Password = "test",
UserName = "test" + i,
UserLanguage = "en",
CreateDate = DateTime.UtcNow,
UpdateDate = DateTime.UtcNow
};
ScopeAccessor.AmbientScope.Database.Insert(userDto);
var userNew = Mock.Of<IUser>(e => e.Id == userDto.Id);
var notification = repo.CreateNotification(userNew, i % 2 == 0 ? entity1 : entity2, i.ToString(CultureInfo.InvariantCulture));
}
var notifications = repo.GetEntityNotifications(entity1);
Assert.AreEqual(5, notifications.Count());
}
}
[Test]
public void Delete_By_Entity()
{
var provider = ScopeProvider;
using (var scope = provider.CreateScope())
{
var repo = new NotificationsRepository((IScopeAccessor)provider);
var node1 = new NodeDto
{
CreateDate = DateTime.UtcNow,
Level = 1,
NodeObjectType = Constants.ObjectTypes.ContentItem,
ParentId = -1,
Path = "-1,1",
SortOrder = 1,
Text = "hello1",
Trashed = false,
UniqueId = Guid.NewGuid(),
UserId = -1
};
ScopeAccessor.AmbientScope.Database.Insert(node1);
var entity1 = Mock.Of<IEntity>(e => e.Id == node1.NodeId);
var node2 = new NodeDto
{
CreateDate = DateTime.UtcNow,
Level = 1,
NodeObjectType = Constants.ObjectTypes.ContentItem,
ParentId = -1,
Path = "-1,2",
SortOrder = 1,
Text = "hello2",
Trashed = false,
UniqueId = Guid.NewGuid(),
UserId = -1
};
ScopeAccessor.AmbientScope.Database.Insert(node2);
var entity2 = Mock.Of<IEntity>(e => e.Id == node2.NodeId);
for (var i = 0; i < 10; i++)
{
var userDto = new UserDto
{
Key = Guid.NewGuid(),
Email = "test" + i,
Login = "test" + i,
Password = "test",
UserName = "test" + i,
UserLanguage = "en",
CreateDate = DateTime.UtcNow,
UpdateDate = DateTime.UtcNow
};
ScopeAccessor.AmbientScope.Database.Insert(userDto);
var userNew = Mock.Of<IUser>(e => e.Id == userDto.Id);
var notification = repo.CreateNotification(userNew, i % 2 == 0 ? entity1 : entity2, i.ToString(CultureInfo.InvariantCulture));
}
var delCount = repo.DeleteNotifications(entity1);
Assert.AreEqual(5, delCount);
}
}
[Test]
public void Delete_By_User()
{
var provider = ScopeProvider;
using (var scope = provider.CreateScope())
{
var repo = new NotificationsRepository((IScopeAccessor)provider);
var userDto = new UserDto
{
Email = "test",
Login = "test",
Password = "test",
UserName = "test",
UserLanguage = "en",
CreateDate = DateTime.UtcNow,
UpdateDate = DateTime.UtcNow
};
ScopeAccessor.AmbientScope.Database.Insert(userDto);
var userNew = Mock.Of<IUser>(e => e.Id == userDto.Id);
var userAdmin = Mock.Of<IUser>(e => e.Id == Constants.Security.SuperUserId);
for (var i = 0; i < 10; i++)
{
var node = new NodeDto
{
CreateDate = DateTime.UtcNow,
Level = 1,
NodeObjectType = Constants.ObjectTypes.ContentItem,
ParentId = -1,
Path = "-1," + i,
SortOrder = 1,
Text = "hello" + i,
Trashed = false,
UniqueId = Guid.NewGuid(),
UserId = -1
};
var result = ScopeAccessor.AmbientScope.Database.Insert(node);
var entity = Mock.Of<IEntity>(e => e.Id == node.NodeId);
var notification = repo.CreateNotification(i % 2 == 0 ? userAdmin : userNew, entity, i.ToString(CultureInfo.InvariantCulture));
}
var delCount = repo.DeleteNotifications(userAdmin);
Assert.AreEqual(5, delCount);
}
}
}