From d309f118d1e827dc2247cc9c8d6ffd0d285161c1 Mon Sep 17 00:00:00 2001 From: "Moore, Douglas S" Date: Thu, 7 Oct 2021 11:03:01 -0400 Subject: [PATCH 1/2] If caching a published document, make sure you use the published Name. Closes #11074. --- .../Strings/DefaultUrlSegmentProvider.cs | 7 +- .../PublishedContent/NuCacheRebuildTests.cs | 158 ++++++++++++++++++ src/Umbraco.Tests/Umbraco.Tests.csproj | 1 + 3 files changed, 165 insertions(+), 1 deletion(-) create mode 100644 src/Umbraco.Tests/PublishedContent/NuCacheRebuildTests.cs diff --git a/src/Umbraco.Core/Strings/DefaultUrlSegmentProvider.cs b/src/Umbraco.Core/Strings/DefaultUrlSegmentProvider.cs index a0f8c735486e..ccb1f1fc848b 100644 --- a/src/Umbraco.Core/Strings/DefaultUrlSegmentProvider.cs +++ b/src/Umbraco.Core/Strings/DefaultUrlSegmentProvider.cs @@ -24,7 +24,12 @@ private static string GetUrlSegmentSource(IContentBase content, string culture) if (content.HasProperty(Constants.Conventions.Content.UrlName)) source = (content.GetValue(Constants.Conventions.Content.UrlName, culture) ?? string.Empty).Trim(); if (string.IsNullOrWhiteSpace(source)) - source = content.GetCultureName(culture); + { + // If the name of a node has been updated, but it has not been published, the url should use the published name, not the current node name + source = (content is IContent document) && document.Edited + ? document.GetPublishName(culture) + : content.GetCultureName(culture); + } return source; } } diff --git a/src/Umbraco.Tests/PublishedContent/NuCacheRebuildTests.cs b/src/Umbraco.Tests/PublishedContent/NuCacheRebuildTests.cs new file mode 100644 index 000000000000..a13489827eb6 --- /dev/null +++ b/src/Umbraco.Tests/PublishedContent/NuCacheRebuildTests.cs @@ -0,0 +1,158 @@ +using System.Globalization; +using System.Linq; +using Moq; +using NUnit.Framework; +using Umbraco.Core; +using Umbraco.Core.Composing.CompositionExtensions; +using Umbraco.Core.Configuration.UmbracoSettings; +using Umbraco.Core.Services; +using Umbraco.Core.Dictionary; +using Umbraco.Core.Logging; +using Umbraco.Core.Models; +using Umbraco.Core.PropertyEditors; +using Umbraco.Core.Services.Implement; +using Umbraco.Tests.TestHelpers; +using Umbraco.Tests.TestHelpers.Entities; +using Umbraco.Web.Models.ContentEditing; +using Umbraco.Tests.Testing; +using Umbraco.Web.PropertyEditors; + +using Current = Umbraco.Web.Composing.Current; +using Umbraco.Web.PublishedCache; +using Umbraco.Web.PublishedCache.NuCache; +using System; +using Umbraco.Core.Cache; +using Umbraco.Web.Cache; +using Umbraco.Core.Sync; +using static Umbraco.Tests.Cache.DistributedCache.DistributedCacheTests; +using static Umbraco.Tests.Integration.ContentEventsTests; +using Umbraco.Tests.Services; +using Umbraco.Core.Persistence.Repositories.Implement; +using Umbraco.Tests.Testing.Objects.Accessors; +using Umbraco.Core.Persistence.Repositories; +using Umbraco.Core.Models.PublishedContent; +using Umbraco.Core.Configuration; +using Umbraco.Web.PublishedCache.NuCache.DataSource; +using Umbraco.Core.Strings; + +namespace Umbraco.Tests.PublishedContent +{ + [TestFixture] + [UmbracoTest(Database = UmbracoTestOptions.Database.NewSchemaPerTest, PublishedRepositoryEvents = true, WithApplication = true)] + public class NuCacheRebuildTests: TestWithDatabaseBase + { + private IContentTypeService _contentTypeService; + private IContentType _contentType; + + protected override void Initialize() + { + base.Initialize(); + + if (!(PublishedSnapshotService is PublishedSnapshotService)) + { + var options = new PublishedSnapshotServiceOptions { IgnoreLocalDb = true }; + var runtime = Mock.Of(); + PublishedSnapshotService = new PublishedSnapshotService( + options, + null, + runtime, + ServiceContext, + Factory.GetInstance(), + null, + new TestPublishedSnapshotAccessor(), + new TestVariationContextAccessor(), + Mock.Of(), + ScopeProvider, + Factory.GetInstance(), Factory.GetInstance(), Factory.GetInstance(), + DefaultCultureAccessor, + new DatabaseDataSource(new JsonContentNestedDataSerializerFactory()), + new GlobalSettings(), + Factory.GetInstance(), + Factory.GetInstance(), + new UrlSegmentProviderCollection(new[] { new DefaultUrlSegmentProvider() }), + new TestSyncBootStateAccessor(SyncBootState.WarmBoot), + new JsonContentNestedDataSerializerFactory() + ); + } + } + public override void SetUp() + { + base.SetUp(); + ContentRepositoryBase.ThrowOnWarning = true; + } + + public override void TearDown() + { + ContentRepositoryBase.ThrowOnWarning = false; + base.TearDown(); + } + + protected override void Compose() + { + base.Compose(); + + Composition.RegisterUnique(factory => Mock.Of()); + } + [Test] + public void UnpublishedNameChanges() + { + var urlSegmentProvider = new DefaultUrlSegmentProvider(); + + var contentType = MockedContentTypes.CreateTextPageContentType(); + ServiceContext.FileService.SaveTemplate(contentType.DefaultTemplate); + ServiceContext.ContentTypeService.Save(contentType); + + var content = MockedContent.CreateTextpageContent(contentType, "hello", Constants.System.Root); + + ServiceContext.ContentService.SaveAndPublish(content); + var cachedContent = ServiceContext.ContentService.GetById(content.Id); + var segment = urlSegmentProvider.GetUrlSegment(cachedContent); + + // Does a new node work? + + Assert.AreEqual("hello", segment); + + content.Name = "goodbye"; + cachedContent = ServiceContext.ContentService.GetById(content.Id); + segment = urlSegmentProvider.GetUrlSegment(cachedContent); + + // We didn't save anything, so all should still be the same + + Assert.AreEqual("hello", segment); + + ServiceContext.ContentService.Save(content); + cachedContent = ServiceContext.ContentService.GetById(content.Id); + segment = urlSegmentProvider.GetUrlSegment(cachedContent); + + // At this point we have saved the new name, but not published. The url should still be the previous name + + Assert.AreEqual("hello", segment); + + PublishedSnapshotService.Rebuild(); + + cachedContent = ServiceContext.ContentService.GetById(content.Id); + segment = urlSegmentProvider.GetUrlSegment(cachedContent); + + // After a rebuild, the unpublished name should still not be the url. + // This was previously incorrect, per #11074 + + Assert.AreEqual("hello", segment); + + ServiceContext.ContentService.SaveAndPublish(content); + cachedContent = ServiceContext.ContentService.GetById(content.Id); + segment = urlSegmentProvider.GetUrlSegment(cachedContent); + + // The page has now been published, so we should see the new url segment + Assert.AreEqual("goodbye", segment); + + PublishedSnapshotService.Rebuild(); + cachedContent = ServiceContext.ContentService.GetById(content.Id); + segment = urlSegmentProvider.GetUrlSegment(cachedContent); + + // Just double checking that things remain after a rebuild + Assert.AreEqual("goodbye", segment); + + } + + } +} diff --git a/src/Umbraco.Tests/Umbraco.Tests.csproj b/src/Umbraco.Tests/Umbraco.Tests.csproj index 3f4daa4df2d8..c00c67ca1ce7 100644 --- a/src/Umbraco.Tests/Umbraco.Tests.csproj +++ b/src/Umbraco.Tests/Umbraco.Tests.csproj @@ -157,6 +157,7 @@ + From 218cb169a3b2c1bac1a026ef187a2ae74e4c6c76 Mon Sep 17 00:00:00 2001 From: "Moore, Douglas S" Date: Thu, 7 Oct 2021 13:34:43 -0400 Subject: [PATCH 2/2] Fix case of new node --- src/Umbraco.Core/Strings/DefaultUrlSegmentProvider.cs | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/Umbraco.Core/Strings/DefaultUrlSegmentProvider.cs b/src/Umbraco.Core/Strings/DefaultUrlSegmentProvider.cs index ccb1f1fc848b..b5f7713fd3ed 100644 --- a/src/Umbraco.Core/Strings/DefaultUrlSegmentProvider.cs +++ b/src/Umbraco.Core/Strings/DefaultUrlSegmentProvider.cs @@ -26,7 +26,8 @@ private static string GetUrlSegmentSource(IContentBase content, string culture) if (string.IsNullOrWhiteSpace(source)) { // If the name of a node has been updated, but it has not been published, the url should use the published name, not the current node name - source = (content is IContent document) && document.Edited + // If this node has never been published (GetPublishName is null), use the unpublished name + source = (content is IContent document) && document.Edited && document.GetPublishName(culture) != null ? document.GetPublishName(culture) : content.GetCultureName(culture); }