diff --git a/CREDITS.txt b/CREDITS.txt index eca307584..216e1d712 100644 --- a/CREDITS.txt +++ b/CREDITS.txt @@ -191,11 +191,11 @@ Website: https://lucenenet.apache.org/ Copyright: Copyright (c) 2009 Apache Software Foundation License: Apache Software Foundation License 2.0 -MarkdownSharp +Markdig ----- -Website: http://code.google.com/p/markdownsharp/ -Copyright: Copyright (c) 2009-2011 Jeff Atwood -License: MIT +Website: https://github.com/lunet-io/markdig +Copyright: Copyright (c) 2018-2019, Alexandre Mutel +License: BSD 2-Clause Mono Class Library ----- diff --git a/src/NuGet.config b/src/NuGet.config new file mode 100644 index 000000000..04f49beab --- /dev/null +++ b/src/NuGet.config @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/src/Orchard.Web/Core/Contents/Handlers/ContentsHandler.cs b/src/Orchard.Web/Core/Contents/Handlers/ContentsHandler.cs index abd4a7b9d..f88f98d17 100644 --- a/src/Orchard.Web/Core/Contents/Handlers/ContentsHandler.cs +++ b/src/Orchard.Web/Core/Contents/Handlers/ContentsHandler.cs @@ -4,6 +4,9 @@ using Orchard.ContentManagement.Handlers; namespace Orchard.Core.Contents.Handlers { public class ContentsHandler : ContentHandlerBase { public override void GetContentItemMetadata(GetContentItemMetadataContext context) { + if (string.IsNullOrWhiteSpace(context.Metadata.DisplayText)) + context.Metadata.DisplayText = context.ContentItem.ContentType; + if (context.Metadata.CreateRouteValues == null) { context.Metadata.CreateRouteValues = new RouteValueDictionary { {"Area", "Contents"}, @@ -12,6 +15,7 @@ namespace Orchard.Core.Contents.Handlers { {"Id", context.ContentItem.ContentType} }; } + if (context.Metadata.EditorRouteValues == null) { context.Metadata.EditorRouteValues = new RouteValueDictionary { {"Area", "Contents"}, @@ -20,6 +24,7 @@ namespace Orchard.Core.Contents.Handlers { {"Id", context.ContentItem.Id} }; } + if (context.Metadata.DisplayRouteValues == null) { context.Metadata.DisplayRouteValues = new RouteValueDictionary { {"Area", "Contents"}, @@ -28,6 +33,7 @@ namespace Orchard.Core.Contents.Handlers { {"Id", context.ContentItem.Id} }; } + if (context.Metadata.RemoveRouteValues == null) { context.Metadata.RemoveRouteValues = new RouteValueDictionary { {"Area", "Contents"}, diff --git a/src/Orchard.Web/Modules/Markdown/Markdown.csproj b/src/Orchard.Web/Modules/Markdown/Markdown.csproj index 78e9bbf28..d2f55aa7c 100644 --- a/src/Orchard.Web/Modules/Markdown/Markdown.csproj +++ b/src/Orchard.Web/Modules/Markdown/Markdown.csproj @@ -54,9 +54,8 @@ false - - ..\..\..\packages\StackExchange.MarkdownSharp.1.5.1.0\lib\net35\MarkdownSharp.dll - True + + ..\..\..\packages\Markdig.Signed.0.18.0\lib\net40\Markdig.Signed.dll ..\..\..\packages\Microsoft.CodeDom.Providers.DotNetCompilerPlatform.2.0.1\lib\net45\Microsoft.CodeDom.Providers.DotNetCompilerPlatform.dll diff --git a/src/Orchard.Web/Modules/Markdown/Services/MarkdownFilter.cs b/src/Orchard.Web/Modules/Markdown/Services/MarkdownFilter.cs index e65d74eb9..cf5d1db3e 100644 --- a/src/Orchard.Web/Modules/Markdown/Services/MarkdownFilter.cs +++ b/src/Orchard.Web/Modules/Markdown/Services/MarkdownFilter.cs @@ -1,5 +1,4 @@ using System; -using System.Web.ApplicationServices; using Orchard.Services; namespace Markdown.Services { @@ -12,9 +11,7 @@ namespace Markdown.Services { if (string.IsNullOrEmpty(text)) return string.Empty; - var markdown = new MarkdownSharp.Markdown(); - - return markdown.Transform(text); + return Markdig.Markdown.ToHtml(text); } } -} \ No newline at end of file +} diff --git a/src/Orchard.Web/Modules/Markdown/packages.config b/src/Orchard.Web/Modules/Markdown/packages.config index 6c607bdbf..3f81f03ed 100644 --- a/src/Orchard.Web/Modules/Markdown/packages.config +++ b/src/Orchard.Web/Modules/Markdown/packages.config @@ -1,9 +1,9 @@  + - \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.AntiSpam/Drivers/ReCaptchaPartDriver.cs b/src/Orchard.Web/Modules/Orchard.AntiSpam/Drivers/ReCaptchaPartDriver.cs index fbf2db7e7..a3624cc17 100644 --- a/src/Orchard.Web/Modules/Orchard.AntiSpam/Drivers/ReCaptchaPartDriver.cs +++ b/src/Orchard.Web/Modules/Orchard.AntiSpam/Drivers/ReCaptchaPartDriver.cs @@ -13,6 +13,7 @@ using Orchard.Logging; using Orchard.UI.Admin; using Orchard.UI.Notify; using Orchard.Services; +using System.Collections.Generic; namespace Orchard.AntiSpam.Drivers { public class ReCaptchaPartDriver : ContentPartDriver { @@ -35,14 +36,16 @@ namespace Orchard.AntiSpam.Drivers { public Localizer T { get; set; } public ILogger Logger { get; set; } protected override DriverResult Editor(ReCaptchaPart part, dynamic shapeHelper) { - var workContext = _workContextAccessor.GetContext(); - - // don't display the part in the admin - if (AdminFilter.IsApplied(workContext.HttpContext.Request.RequestContext)) { - return null; - } - + + // we want to be returning a shape even when it should display nothing, because + // other features may need the Shape's type, or some other of its properties return ContentShape("Parts_ReCaptcha_Fields", () => { + var workContext = _workContextAccessor.GetContext(); + // don't display the part in the admin + if (AdminFilter.IsApplied(workContext.HttpContext.Request.RequestContext)) { + return null; + } + var settings = workContext.CurrentSite.As(); if (settings.TrustAuthenticatedUsers && workContext.CurrentUser != null) { @@ -73,7 +76,7 @@ namespace Orchard.AntiSpam.Drivers { var context = workContext.HttpContext; try { - var result = ExecuteValidateRequest( + var result = ValidateRequest(//ExecuteValidateRequest( settings.PrivateKey, context.Request.ServerVariables["REMOTE_ADDR"], context.Request.Form["g-recaptcha-response"] @@ -104,13 +107,27 @@ namespace Orchard.AntiSpam.Drivers { return Editor(part, shapeHelper); } - private static string ExecuteValidateRequest(string privateKey, string remoteip, string response) { - var postData = String.Format(CultureInfo.InvariantCulture, + // temporarily save pairs, to prevent sending the same exact request + // more than once in a single Request. + private Dictionary ValidationResponse; + + private string ValidateRequest(string privateKey, string remoteip, string response) { + if (ValidationResponse == null) { + ValidationResponse = new Dictionary(); + } + var postData = string.Format(CultureInfo.InvariantCulture, "secret={0}&response={1}&remoteip={2}", privateKey, response, remoteip ); + if (!ValidationResponse.ContainsKey(postData)) { + ValidationResponse.Add(postData, ExecuteValidateRequest(postData)); + } + return ValidationResponse[postData]; + } + + private static string ExecuteValidateRequest(string postData) { WebRequest request = WebRequest.Create(ReCaptchaSecureUrl + "?" + postData); request.Method = "GET"; @@ -123,5 +140,6 @@ namespace Orchard.AntiSpam.Drivers { } } } + } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.ArchiveLater/Handlers/ArchiveLaterPartHandler.cs b/src/Orchard.Web/Modules/Orchard.ArchiveLater/Handlers/ArchiveLaterPartHandler.cs index b77d6b962..44d867039 100644 --- a/src/Orchard.Web/Modules/Orchard.ArchiveLater/Handlers/ArchiveLaterPartHandler.cs +++ b/src/Orchard.Web/Modules/Orchard.ArchiveLater/Handlers/ArchiveLaterPartHandler.cs @@ -11,10 +11,12 @@ namespace Orchard.ArchiveLater.Handlers { OnLoading((context, part) => LazyLoadHandlers(part)); OnVersioning((context, part, newVersionPart) => LazyLoadHandlers(newVersionPart)); + OnRemoved((context, part) => _archiveLaterService.RemoveArchiveLaterTasks(part.ContentItem)); + OnDestroyed((context, part) => _archiveLaterService.RemoveArchiveLaterTasks(part.ContentItem)); } protected void LazyLoadHandlers(ArchiveLaterPart part) { part.ScheduledArchiveUtc.Loader(() => _archiveLaterService.GetScheduledArchiveUtc(part)); } } -} \ No newline at end of file +} diff --git a/src/Orchard.Web/Modules/Orchard.Blogs/Handlers/BlogPartArchiveHandler.cs b/src/Orchard.Web/Modules/Orchard.Blogs/Handlers/BlogPartArchiveHandler.cs index 3a53ceddb..5b3b04494 100644 --- a/src/Orchard.Web/Modules/Orchard.Blogs/Handlers/BlogPartArchiveHandler.cs +++ b/src/Orchard.Web/Modules/Orchard.Blogs/Handlers/BlogPartArchiveHandler.cs @@ -15,10 +15,10 @@ namespace Orchard.Blogs.Handlers { private readonly IWorkContextAccessor _workContextAccessor; private readonly IContentManager _contentManager; // contains the creation time of a blog part before it has been changed - private readonly Dictionary _previousCreatedUtc = new Dictionary(); + private readonly Dictionary _previousCreatedUtc = new Dictionary(); public BlogPartArchiveHandler( - IRepository blogArchiveRepository, + IRepository blogArchiveRepository, IBlogPostService blogPostService, IWorkContextAccessor workContextAccessor, IContentManager contentManager) { @@ -26,11 +26,11 @@ namespace Orchard.Blogs.Handlers { _workContextAccessor = workContextAccessor; _contentManager = contentManager; - OnUpdating((context, cp) => { if(context.ContentItem.Has()) SavePreviousCreatedDate(context.Id);}); + OnUpdating((context, cp) => { if (context.ContentItem.Has()) SavePreviousCreatedDate(context.Id); }); OnRemoving((context, bp) => SavePreviousCreatedDate(context.Id)); OnUnpublishing((context, bp) => SavePreviousCreatedDate(context.Id)); - OnPublished((context, bp) => IncreaseBlogArchive(bp)); + OnPublished((context, bp) => ManageBlogArchiveSync(bp)); OnUnpublished((context, bp) => ReduceBlogArchive(bp)); OnRemoved((context, bp) => ReduceBlogArchive(bp)); } @@ -71,58 +71,56 @@ namespace Orchard.Blogs.Handlers { && x.Month == datetime.Month && x.Year == datetime.Year); - if(previousArchiveRecord == null) + if (previousArchiveRecord == null) return; if (previousArchiveRecord.PostCount > 1) previousArchiveRecord.PostCount--; else _blogArchiveRepository.Delete(previousArchiveRecord); + + blogPostPart.ArchiveSync = null; } + private void ReduceBlogArchiveByDate(int BlogPartId, DateTime? dateBlogPost) { + _blogArchiveRepository.Flush(); + + if (BlogPartId!= 0 && dateBlogPost.HasValue) { + var previousArchiveRecord = _blogArchiveRepository.Table + .FirstOrDefault(x => x.BlogPart.Id == BlogPartId + && x.Month == dateBlogPost.Value.Month + && x.Year == dateBlogPost.Value.Year); + + if (previousArchiveRecord == null) + return; + + if (previousArchiveRecord.PostCount > 1) + previousArchiveRecord.PostCount--; + else + _blogArchiveRepository.Delete(previousArchiveRecord); + } + else { + return; + } + + } + private void IncreaseBlogArchive(BlogPostPart blogPostPart) { _blogArchiveRepository.Flush(); - + var commonPart = blogPostPart.As(); - if(commonPart == null || !commonPart.CreatedUtc.HasValue) + if (commonPart == null || !commonPart.CreatedUtc.HasValue) return; // get the time zone for the current request var timeZone = _workContextAccessor.GetContext().CurrentTimeZone; - var previousCreatedUtc = _previousCreatedUtc.ContainsKey(blogPostPart.Id) ? _previousCreatedUtc[blogPostPart.Id] : DateTime.MinValue; - previousCreatedUtc = TimeZoneInfo.ConvertTimeFromUtc(previousCreatedUtc, timeZone); - - var previousMonth = previousCreatedUtc.Month; - var previousYear = previousCreatedUtc.Year; - var newCreatedUtc = commonPart.CreatedUtc; newCreatedUtc = TimeZoneInfo.ConvertTimeFromUtc(newCreatedUtc.Value, timeZone); var newMonth = newCreatedUtc.Value.Month; var newYear = newCreatedUtc.Value.Year; - // if archives are the same there is nothing to do - if (previousMonth == newMonth && previousYear == newYear) { - return; - } - - // decrement previous archive record - var previousArchiveRecord = _blogArchiveRepository - .Table - .FirstOrDefault(x => x.BlogPart.Id == blogPostPart.BlogPart.Id - && x.Month == previousMonth - && x.Year == previousYear); - - if (previousArchiveRecord != null && previousArchiveRecord.PostCount > 0) { - previousArchiveRecord.PostCount--; - } - - // if previous count is now zero, delete the record - if (previousArchiveRecord != null && previousArchiveRecord.PostCount == 0) { - _blogArchiveRepository.Delete(previousArchiveRecord); - } - // increment new archive record var newArchiveRecord = _blogArchiveRepository .Table @@ -136,7 +134,31 @@ namespace Orchard.Blogs.Handlers { _blogArchiveRepository.Create(newArchiveRecord); } - newArchiveRecord.PostCount++; + newArchiveRecord.PostCount++; + } + + private void ManageBlogArchiveSync(BlogPostPart blogPostPart) { + var commonPart = blogPostPart.As(); + if (commonPart == null || !commonPart.CreatedUtc.HasValue) + return; + + var timeZone = _workContextAccessor.GetContext().CurrentTimeZone; + var creationDate = TimeZoneInfo.ConvertTimeFromUtc(commonPart.CreatedUtc.Value, timeZone); + + if (blogPostPart.ArchiveSync == null) { + IncreaseBlogArchive(blogPostPart); + blogPostPart.ArchiveSync = creationDate; + } + else { + if (creationDate == blogPostPart.ArchiveSync) { + return; + } + else { + ReduceBlogArchiveByDate(blogPostPart.BlogPart.Id, blogPostPart.ArchiveSync); + IncreaseBlogArchive(blogPostPart); + blogPostPart.ArchiveSync = creationDate; + } + } } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Blogs/Models/BlogPostPart.cs b/src/Orchard.Web/Modules/Orchard.Blogs/Models/BlogPostPart.cs index 754e3e99d..d495fb3a9 100644 --- a/src/Orchard.Web/Modules/Orchard.Blogs/Models/BlogPostPart.cs +++ b/src/Orchard.Web/Modules/Orchard.Blogs/Models/BlogPostPart.cs @@ -49,5 +49,10 @@ namespace Orchard.Blogs.Models { public DateTime? PublishedUtc { get { return this.As().PublishedUtc; } } + + public DateTime? ArchiveSync { + get { return this.Retrieve(x => x.ArchiveSync); } + set { this.Store(x => x.ArchiveSync, value); } + } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Indexing/Services/IndexingTaskExecutor.cs b/src/Orchard.Web/Modules/Orchard.Indexing/Services/IndexingTaskExecutor.cs index 6d4343a9c..93fa1eb9b 100644 --- a/src/Orchard.Web/Modules/Orchard.Indexing/Services/IndexingTaskExecutor.cs +++ b/src/Orchard.Web/Modules/Orchard.Indexing/Services/IndexingTaskExecutor.cs @@ -223,11 +223,25 @@ namespace Orchard.Indexing.Services { .OrderBy(x => x.Id) .Take(ContentItemsPerLoop) .ToList() - // In some rare cases a ContentItemRecord without a ContentType can end up in the DB. - // We need to filter out such records, otherwise they will crash the ContentManager. - .Where(x => x.ContentItemRecord != null && x.ContentItemRecord.ContentType != null) + .Where(x => x.ContentItemRecord != null) .GroupBy(x => x.ContentItemRecord.Id) - .Select(group => new { TaskId = group.Max(task => task.Id), Delete = group.Last().Action == IndexingTaskRecord.Delete, Id = group.Key, ContentItem = _contentManager.Get(group.Key, VersionOptions.Latest) }) + .Select(group => new { + TaskId = group.Max(task => task.Id), + Delete = group.Last().Action == IndexingTaskRecord.Delete, + Id = group.Key, + // We can only have a ContentItem if the ContentItemRecord matches + // something that still exists in our records. + ContentItem = group.All(x => { + try { + return x.ContentItemRecord != null + // ContentType is required to build the ContentItem + && x.ContentItemRecord.ContentType != null; + } + catch { + return false; + } + }) ? _contentManager.Get(group.Key, VersionOptions.Latest) : null // Set to null to "tell" it's a delete/destroy + }) .OrderBy(x => x.TaskId) .ToArray(); @@ -274,7 +288,7 @@ namespace Orchard.Indexing.Services { } } while (loop); - } + } // save current state of the index indexSettings.LastIndexedUtc = _clock.UtcNow; diff --git a/src/Orchard.Web/Modules/Orchard.Lists/Controllers/AdminController.cs b/src/Orchard.Web/Modules/Orchard.Lists/Controllers/AdminController.cs index 8b4c65602..c4d0acf98 100644 --- a/src/Orchard.Web/Modules/Orchard.Lists/Controllers/AdminController.cs +++ b/src/Orchard.Web/Modules/Orchard.Lists/Controllers/AdminController.cs @@ -9,7 +9,6 @@ using Orchard.Core.Common.Models; using Orchard.Core.Containers.Models; using Orchard.Core.Containers.Services; using Orchard.Core.Containers.ViewModels; -using Orchard.Core.Contents; using Orchard.Core.Contents.ViewModels; using Orchard.Core.Title.Models; using Orchard.Data; @@ -39,7 +38,7 @@ namespace Orchard.Lists.Controllers { IOrchardServices services, IContentDefinitionManager contentDefinitionManager, IShapeFactory shapeFactory, - IContainerService containerService, + IContainerService containerService, IListViewService listViewService, ITransactionManager transactionManager) { @@ -92,7 +91,7 @@ namespace Orchard.Lists.Controllers { var pager = new Pager(_services.WorkContext.CurrentSite, pagerParameters); var pagerShape = Shape.Pager(pager).TotalItemCount(query.Count()); var pageOfLists = query.Slice(pager.GetStartIndex(), pager.PageSize); - + var listsShape = Shape.List(); listsShape.AddRange(pageOfLists.Select(x => _contentManager.BuildDisplay(x, "SummaryAdmin")).ToList()); var viewModel = Shape.ViewModel() @@ -171,10 +170,10 @@ namespace Orchard.Lists.Controllers { if (containerTypes.Count > 1) { return RedirectToAction("SelectType"); } - return RedirectToAction("Create", new {id = containerTypes.First().Name}); + return RedirectToAction("Create", new { id = containerTypes.First().Name }); } - return RedirectToAction("Create", "Admin", new {area = "Contents", id, returnUrl = Url.Action("Index", "Admin", new { area = "Orchard.Lists" })}); + return RedirectToAction("Create", "Admin", new { area = "Contents", id, returnUrl = Url.Action("Index", "Admin", new { area = "Orchard.Lists" }) }); } public ActionResult SelectType() { @@ -312,11 +311,11 @@ namespace Orchard.Lists.Controllers { LocalizedString message; if (previousItemContainer == null) { - message = T("{0} was moved to {2}", itemMetadata.DisplayText, Url.RouteUrl(containerMetadata.AdminRouteValues), containerMetadata.DisplayText); + message = T("{0} was moved to {2}.", itemMetadata.DisplayText, Url.RouteUrl(containerMetadata.AdminRouteValues), containerMetadata.DisplayText); } else if (previousItemContainer.Id != containerId) { var previousItemContainerMetadata = _contentManager.GetItemMetadata(commonPart.Container); - message = T("{0} was moved from {4} to {2}", + message = T("{0} was moved from {4} to {2}.", itemMetadata.DisplayText, Url.RouteUrl(containerMetadata.AdminRouteValues), containerMetadata.DisplayText, @@ -374,7 +373,7 @@ namespace Orchard.Lists.Controllers { break; } - return RedirectToAction("List", new {containerId, page = pagerParameters.Page, pageSize = pagerParameters.PageSize}); + return RedirectToAction("List", new { containerId, page = pagerParameters.Page, pageSize = pagerParameters.PageSize }); } [HttpPost, ActionName("List")] @@ -393,7 +392,7 @@ namespace Orchard.Lists.Controllers { /// Only publishes the content if it is already published. /// private void RePublish(IContent content) { - if(content.ContentItem.VersionRecord.Published) + if (content.ContentItem.VersionRecord.Published) _contentManager.Publish(content.ContentItem); } @@ -439,7 +438,7 @@ namespace Orchard.Lists.Controllers { if (!_services.Authorizer.Authorize(Orchard.Core.Contents.Permissions.EditContent, item, T("Couldn't move selected content."))) { return false; } - + // Ensure the item can be in that container. if (itemContentTypes.Any() && itemContentTypes.All(x => x.Name != item.ContentItem.ContentType)) { _services.TransactionManager.Cancel(); diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Styles/media-library-picker-admin.css b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Styles/media-library-picker-admin.css index e939618b8..8cd94d641 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Styles/media-library-picker-admin.css +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Styles/media-library-picker-admin.css @@ -32,6 +32,12 @@ clear: both; } + .media-library-picker .media-thumbnail-image { + background-image: url(""); + background-size: 15%; + background-repeat: repeat; + } + .overlay { position: absolute; bottom: 20px; diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Styles/menu.media-library-admin.css b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Styles/menu.media-library-admin.css index a1076ccb1..dbab41574 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Styles/menu.media-library-admin.css +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Styles/menu.media-library-admin.css @@ -6,4 +6,10 @@ background-position:0 -30px !important; } .summary .media-thumbnail { display: inline-block; +} + +.summary .media-thumbnail-image { + background-image: url(""); + background-size: 15%; + background-repeat: repeat; } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Styles/orchard-medialibrary-admin.css b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Styles/orchard-medialibrary-admin.css index 9831fb6a7..c93c2f202 100644 --- a/src/Orchard.Web/Modules/Orchard.MediaLibrary/Styles/orchard-medialibrary-admin.css +++ b/src/Orchard.Web/Modules/Orchard.MediaLibrary/Styles/orchard-medialibrary-admin.css @@ -267,13 +267,18 @@ #media-library-main-list .media-thumbnail { width: 200px; height: 200px; - overflow:hidden; + overflow: hidden; +} +#media-library-main-editor-focus .media-item-summary-admin img { + background-image: url(""); + background-size: 15%; + background-repeat: repeat; } #media-library-main-selection .media-thumbnail { width: 120px; height: 120px; - overflow:hidden; + overflow: hidden; } .media-thumbnail-o-embed { @@ -294,6 +299,9 @@ -o-background-size: cover; -webkit-background-size: cover; background-size: cover; + background-image: url(""); + background-size: 15%; + background-repeat: repeat; } .media-thumbnail-video { diff --git a/src/Orchard.Web/Modules/Orchard.Projections/Handlers/FieldIndexPartHandler.cs b/src/Orchard.Web/Modules/Orchard.Projections/Handlers/FieldIndexPartHandler.cs index 528ee3407..fed0851a1 100644 --- a/src/Orchard.Web/Modules/Orchard.Projections/Handlers/FieldIndexPartHandler.cs +++ b/src/Orchard.Web/Modules/Orchard.Projections/Handlers/FieldIndexPartHandler.cs @@ -45,7 +45,10 @@ namespace Orchard.Projections.Handlers { } } private void Updated(UpdateContentContext context, FieldIndexPart fieldIndexPart) { - if (context.UpdatingItemVersionRecord.Latest) { // updates projection draft indexes only if it is the latest version + // there are two different item types: saved in memory and saved to db + // those saved in memory don't have correctly the populated record and this generate NullReferenceException + if (context.UpdatingItemVersionRecord != null && context.UpdatingItemVersionRecord.Latest) { + // updates projection draft indexes only if it is the latest version DescribeValuesToIndex(fieldIndexPart, (indexServiceContext) => { _fieldIndexService.Set( fieldIndexPart, @@ -114,4 +117,4 @@ namespace Orchard.Projections.Handlers { } } -} \ No newline at end of file +} diff --git a/src/Orchard.Web/Modules/Orchard.Recipes/Orchard.Recipes.csproj b/src/Orchard.Web/Modules/Orchard.Recipes/Orchard.Recipes.csproj index 6183f7e56..6d50fb1e8 100644 --- a/src/Orchard.Web/Modules/Orchard.Recipes/Orchard.Recipes.csproj +++ b/src/Orchard.Web/Modules/Orchard.Recipes/Orchard.Recipes.csproj @@ -131,6 +131,7 @@ + diff --git a/src/Orchard.Web/Modules/Orchard.Recipes/Providers/Executors/RemoveFromContentTypeStep.cs b/src/Orchard.Web/Modules/Orchard.Recipes/Providers/Executors/RemoveFromContentTypeStep.cs new file mode 100644 index 000000000..06a7514e1 --- /dev/null +++ b/src/Orchard.Web/Modules/Orchard.Recipes/Providers/Executors/RemoveFromContentTypeStep.cs @@ -0,0 +1,89 @@ +using System; +using System.Xml; +using Orchard.ContentManagement.MetaData; +using Orchard.ContentTypes.Events; +using Orchard.Localization; +using Orchard.Logging; +using Orchard.Recipes.Models; +using Orchard.Recipes.Services; + +namespace Orchard.Recipes.Providers.Executors { + public class RemoveFromContentTypeStep : RecipeExecutionStep { + private readonly IContentDefinitionManager _contentDefinitionManager; + private readonly IContentDefinitionEventHandler _contentDefinitionEventHandlers; + + public RemoveFromContentTypeStep(RecipeExecutionLogger logger, IContentDefinitionManager contentDefinitionManager, + IContentDefinitionEventHandler contentDefinitonEventHandlers) : base(logger) { + _contentDefinitionManager = contentDefinitionManager; + _contentDefinitionEventHandlers = contentDefinitonEventHandlers; + } + + public override string Name { + get { return "RemoveFromContentType"; } + } + + public override LocalizedString DisplayName { + get { return T("Remove From Content Type"); } + } + + public override LocalizedString Description { + get { return T("Removes a list of parts and fields from a content type."); } + } + + // + // + // + // + // + // + // + // + public override void Execute(RecipeExecutionContext context) { + foreach (var metadataElementType in context.RecipeStep.Step.Elements()) { + Logger.Debug("Processing element '{0}'.", metadataElementType.Name.LocalName); + var typeName = XmlConvert.DecodeName(metadataElementType.Name.LocalName); + + foreach (var metadataElement in metadataElementType.Elements()) { + switch (metadataElement.Name.LocalName) { + case "Parts": + foreach (var element in metadataElement.Elements()) { + var partName = XmlConvert.DecodeName(element.Name.LocalName); + + Logger.Information("Removing content part '{0}' from content type '{1}'.", partName, typeName); + try { + _contentDefinitionManager.AlterTypeDefinition(typeName, typeBuilder => typeBuilder.RemovePart(partName)); + _contentDefinitionEventHandlers.ContentPartDetached(new ContentPartDetachedContext { ContentTypeName = typeName, ContentPartName = partName }); + } + catch (Exception ex) { + Logger.Error(ex, "Error while removing content part '{0}' from content type'{1}'.", partName, typeName); + throw; + } + } + break; + + case "Fields": + foreach (var element in metadataElement.Elements()) { + var fieldName = XmlConvert.DecodeName(element.Name.LocalName); + + Logger.Information("Removing content field '{0}' from content type '{1}'.", fieldName, typeName); + try { + _contentDefinitionManager.AlterPartDefinition(typeName, typeBuilder => typeBuilder.RemoveField(fieldName)); + _contentDefinitionEventHandlers.ContentFieldDetached(new ContentFieldDetachedContext { ContentPartName = typeName, ContentFieldName = fieldName }); + } + catch (Exception ex) { + Logger.Error(ex, "Error while removing content field '{0}' from content type'{1}'.", fieldName, typeName); + throw; + } + } + break; + + default: + Logger.Warning("Unrecognized element '{0}' encountered; skipping", + metadataElement.Name.LocalName); + break; + } + } + } + } + } +} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Recipes/Views/Admin/Index.cshtml b/src/Orchard.Web/Modules/Orchard.Recipes/Views/Admin/Index.cshtml index 836a1af24..9890fc789 100644 --- a/src/Orchard.Web/Modules/Orchard.Recipes/Views/Admin/Index.cshtml +++ b/src/Orchard.Web/Modules/Orchard.Recipes/Views/Admin/Index.cshtml @@ -29,7 +29,7 @@

@recipe.Name - @Html.ActionLink(T("Execute").Text, "Recipes", "Admin", new { area = "Orchard.Recipes", moduleId = descriptor.Id, name = recipe.Name }, new { itemprop = "UnsafeUrl" })

-

@recipe.Description

+

@recipe.Description

} diff --git a/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Handlers/SslSettingsPartHandler.cs b/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Handlers/SslSettingsPartHandler.cs index 8d9e08d07..ab93bdd4b 100644 --- a/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Handlers/SslSettingsPartHandler.cs +++ b/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Handlers/SslSettingsPartHandler.cs @@ -3,12 +3,24 @@ using Orchard.Data; using Orchard.ContentManagement.Handlers; using Orchard.Localization; using Orchard.SecureSocketsLayer.Models; +using Orchard.Caching; namespace Orchard.SecureSocketsLayer.Handlers { public class SslSettingsPartHandler : ContentHandler { - public SslSettingsPartHandler() { + private readonly ISignals _signals; + + public SslSettingsPartHandler(ISignals signals) { + + _signals = signals; + T = NullLocalizer.Instance; + Filters.Add(new ActivatingFilter("Site")); + + // Evict cached content when updated, removed or destroyed. + OnPublished((context, part) => Invalidate(part)); + OnRemoved((context, part) => Invalidate(part)); + OnDestroyed((context, part) => Invalidate(part)); } public Localizer T { get; set; } @@ -22,5 +34,10 @@ namespace Orchard.SecureSocketsLayer.Handlers { Position = "2" }); } + + private void Invalidate(SslSettingsPart content) { + _signals.Trigger($"SslSettingsPart_{content.Id}"); + _signals.Trigger("SslSettingsPart_EvictAll"); + } } } \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Services/StrictTransportSecurityMiddlewareProvider.cs b/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Services/StrictTransportSecurityMiddlewareProvider.cs index 2b3d8c329..3e46e2c6d 100644 --- a/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Services/StrictTransportSecurityMiddlewareProvider.cs +++ b/src/Orchard.Web/Modules/Orchard.SecureSocketsLayer/Services/StrictTransportSecurityMiddlewareProvider.cs @@ -1,4 +1,5 @@ using System.Collections.Generic; +using Orchard.Caching; using Orchard.ContentManagement; using Orchard.Logging; using Orchard.Owin; @@ -8,11 +9,19 @@ using Owin; namespace Orchard.SecureSocketsLayer.Services { public class StrictTransportSecurityMiddlewareProvider : IOwinMiddlewareProvider { private readonly IWorkContextAccessor _wca; + private readonly ICacheManager _cacheManager; + private readonly ISignals _signals; public ILogger Logger { get; set; } - public StrictTransportSecurityMiddlewareProvider(IWorkContextAccessor wca) { + public StrictTransportSecurityMiddlewareProvider( + IWorkContextAccessor wca, + ICacheManager cacheManager, + ISignals signals) { + _wca = wca; + _cacheManager = cacheManager; + _signals = signals; Logger = NullLogger.Instance; } @@ -22,7 +31,13 @@ namespace Orchard.SecureSocketsLayer.Services { new OwinMiddlewareRegistration { Configure = app => app.Use(async (context, next) => { - var sslSettings = _wca.GetContext().CurrentSite.As(); + var cacheKey = "Orchard.SecureSocketsLayer.Services.StrictTransportSecurityMiddlewareProvider.GetOwinMiddlewares"; + var sslSettings = _cacheManager.Get(cacheKey, true, ctx =>{ + // check whether the cache should be invalidated + ctx.Monitor(_signals.When("SslSettingsPart_EvictAll")); + // cache this and save recomputing it each call + return _wca.GetContext().CurrentSite.As(); + }); if (sslSettings.SendStrictTransportSecurityHeaders) { string responseValue = "max-age=" + sslSettings.StrictTransportSecurityMaxAge; diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Views/Admin/Index.cshtml b/src/Orchard.Web/Modules/Orchard.Taxonomies/Views/Admin/Index.cshtml index 1b4598d0d..dffaddfc7 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Views/Admin/Index.cshtml +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Views/Admin/Index.cshtml @@ -56,6 +56,7 @@ @if (!taxonomyEntry.IsInternal) { + @Html.ItemDisplayLink(T("View").Text, taxonomyEntry.ContentItem) | if (Authorizer.Authorize(Permissions.CreateTaxonomy)) { @Html.ItemEditLink(T("Edit").Text, taxonomyEntry.ContentItem, new { ReturnUrl = Request.RawUrl }) | } diff --git a/src/Orchard.Web/Modules/Orchard.Taxonomies/Views/TermAdmin/Index.cshtml b/src/Orchard.Web/Modules/Orchard.Taxonomies/Views/TermAdmin/Index.cshtml index a84438645..3a64ff264 100644 --- a/src/Orchard.Web/Modules/Orchard.Taxonomies/Views/TermAdmin/Index.cshtml +++ b/src/Orchard.Web/Modules/Orchard.Taxonomies/Views/TermAdmin/Index.cshtml @@ -41,14 +41,20 @@ - @* Tabs for levels *@ @for (var i = 1; i <= termEntry.GetLevels(); i++) {   } + @* Tabs for levels *@ @for (var i = 1; i <= termEntry.GetLevels(); i++) { } - @Html.ItemDisplayLink(termEntry.Name, termEntry.ContentItem) - @if (termEntry.HasDraft) { - @T(" (Draft)") + @if (Authorizer.Authorize(Permissions.EditTerm)) { + @Html.ItemEditLink(termEntry.Name, termEntry.ContentItem, new { returnUrl = Url.Action("Index", "TermAdmin", new { taxonomyId = Model.Taxonomy.Id }) }) + } + else { + @Html.ItemDisplayText(termEntry.ContentItem) + @if (termEntry.HasDraft) { + @T(" (Draft)") + } } + @Html.ItemDisplayLink(T("View").Text, termEntry.ContentItem) | @if (Authorizer.Authorize(Permissions.EditTerm)) { @Html.ItemEditLink(T("Edit").Text, termEntry.ContentItem, new { returnUrl = Url.Action("Index", "TermAdmin", new { taxonomyId = Model.Taxonomy.Id }) }) | @Html.ActionLink(T("Move").ToString(), "MoveTerm", new { taxonomyId = Model.Taxonomy.Id, termIds = termEntry.Id }) diff --git a/src/Orchard.sln b/src/Orchard.sln index cc2b05fa7..610e3fae9 100644 --- a/src/Orchard.sln +++ b/src/Orchard.sln @@ -269,6 +269,7 @@ Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "Solution Items", "Solution ..\.deployment = ..\.deployment .editorconfig = .editorconfig ..\deploy.ps1 = ..\deploy.ps1 + NuGet.config = NuGet.config ..\DeploymentUtility.psm1 = ..\DeploymentUtility.psm1 Rebracer.xml = Rebracer.xml WebEssentials-Settings.json = WebEssentials-Settings.json diff --git a/src/Orchard/Localization/Services/SiteCultureSelector.cs b/src/Orchard/Localization/Services/SiteCultureSelector.cs index 66be0e6ec..dcfcbc751 100644 --- a/src/Orchard/Localization/Services/SiteCultureSelector.cs +++ b/src/Orchard/Localization/Services/SiteCultureSelector.cs @@ -1,22 +1,36 @@ using System; using System.Web; +using Orchard.Caching; namespace Orchard.Localization.Services { public class SiteCultureSelector : ICultureSelector { private readonly IWorkContextAccessor _workContextAccessor; + private readonly ICacheManager _cacheManager; + private readonly ISignals _signals; + + public SiteCultureSelector( + IWorkContextAccessor workContextAccessor, + ICacheManager cacheManager, + ISignals signals) { - public SiteCultureSelector(IWorkContextAccessor workContextAccessor) { _workContextAccessor = workContextAccessor; + _cacheManager = cacheManager; + _signals = signals; } public CultureSelectorResult GetCulture(HttpContextBase context) { - string currentCultureName = _workContextAccessor.GetContext().CurrentSite.SiteCulture; + var cacheKey = "Orchard.Localization.Services.SiteCultureSelector.GetCulture"; + return _cacheManager.Get(cacheKey, true, ctx => { + // this is the same signal used in Orchard.Framework.DefaultCultureManager + ctx.Monitor(_signals.When("culturesChanged")); - if (String.IsNullOrEmpty(currentCultureName)) { - return null; - } - - return new CultureSelectorResult { Priority = -5, CultureName = currentCultureName }; + string currentCultureName = _workContextAccessor.GetContext().CurrentSite.SiteCulture; + if (String.IsNullOrEmpty(currentCultureName)) { + return null; + } + return new CultureSelectorResult { Priority = -5, CultureName = currentCultureName }; + }); + } } } diff --git a/src/Orchard/Mvc/Html/ContentItemExtensions.cs b/src/Orchard/Mvc/Html/ContentItemExtensions.cs index 59fd7de07..eca400f8c 100644 --- a/src/Orchard/Mvc/Html/ContentItemExtensions.cs +++ b/src/Orchard/Mvc/Html/ContentItemExtensions.cs @@ -44,7 +44,7 @@ namespace Orchard.Mvc.Html { NonNullOrEmpty(linkText, metadata.DisplayText, "view"), Convert.ToString(metadata.DisplayRouteValues["action"]), metadata.DisplayRouteValues, - new RouteValueDictionary(htmlAttributes)); + HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes)); } public static string ItemDisplayUrl(this UrlHelper urlHelper, IContent content) { @@ -111,7 +111,7 @@ namespace Orchard.Mvc.Html { NonNullOrEmpty(linkText, metadata.DisplayText, content.ContentItem.TypeDefinition.DisplayName), Convert.ToString(metadata.EditorRouteValues["action"]), metadata.EditorRouteValues.Merge(additionalRouteValues), - new RouteValueDictionary(htmlAttributes)); + HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes)); } public static MvcHtmlString ItemAdminLink(this HtmlHelper html, IContent content) { diff --git a/src/Orchard/Mvc/Html/HtmlHelperExtensions.cs b/src/Orchard/Mvc/Html/HtmlHelperExtensions.cs index f6b7e3f08..c8644baf9 100644 --- a/src/Orchard/Mvc/Html/HtmlHelperExtensions.cs +++ b/src/Orchard/Mvc/Html/HtmlHelperExtensions.cs @@ -52,7 +52,7 @@ namespace Orchard.Mvc.Html { } public static MvcHtmlString SelectOption(this HtmlHelper html, T currentValue, T optionValue, string text, object htmlAttributes) { - return SelectOption(html, optionValue, object.Equals(optionValue, currentValue), text, new RouteValueDictionary(htmlAttributes)); + return SelectOption(html, optionValue, object.Equals(optionValue, currentValue), text, HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes)); } public static MvcHtmlString SelectOption(this HtmlHelper html, T currentValue, T optionValue, string text, RouteValueDictionary htmlAttributes) { @@ -64,7 +64,7 @@ namespace Orchard.Mvc.Html { } public static MvcHtmlString SelectOption(this HtmlHelper html, object optionValue, bool selected, string text, object htmlAttributes) { - return SelectOption(html, optionValue, selected, text, new RouteValueDictionary(htmlAttributes)); + return SelectOption(html, optionValue, selected, text, HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes)); } public static MvcHtmlString SelectOption(this HtmlHelper html, object optionValue, bool selected, string text, RouteValueDictionary htmlAttributes) { @@ -203,7 +203,7 @@ namespace Orchard.Mvc.Html { #region Image public static MvcHtmlString Image(this HtmlHelper htmlHelper, string src, string alt, object htmlAttributes) { - return htmlHelper.Image(src, alt, new RouteValueDictionary(htmlAttributes)); + return htmlHelper.Image(src, alt, HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes)); } public static MvcHtmlString Image(this HtmlHelper htmlHelper, string src, string alt, IDictionary htmlAttributes) { @@ -238,7 +238,7 @@ namespace Orchard.Mvc.Html { } public static IHtmlString Link(this HtmlHelper htmlHelper, string linkContents, string href, object htmlAttributes) { - return htmlHelper.Link(linkContents, href, new RouteValueDictionary(htmlAttributes)); + return htmlHelper.Link(linkContents, href, HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes)); } public static IHtmlString Link(this HtmlHelper htmlHelper, string linkContents, string href, IDictionary htmlAttributes) { @@ -264,7 +264,7 @@ namespace Orchard.Mvc.Html { } public static IHtmlString LinkOrDefault(this HtmlHelper htmlHelper, string linkContents, string href, object htmlAttributes) { - return htmlHelper.LinkOrDefault(linkContents, href, new RouteValueDictionary(htmlAttributes)); + return htmlHelper.LinkOrDefault(linkContents, href, HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes)); } public static IHtmlString LinkOrDefault(this HtmlHelper htmlHelper, string linkContents, string href, IDictionary htmlAttributes) { @@ -290,7 +290,7 @@ namespace Orchard.Mvc.Html { } public static IHtmlString Hint(this HtmlHelper htmlHelper, LocalizedString text, object htmlAttributes) { - return Hint(htmlHelper, text, htmlAttributes != null ? new RouteValueDictionary(htmlAttributes) : null); + return Hint(htmlHelper, text, htmlAttributes != null ? HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes) : null); } public static IHtmlString Hint(this HtmlHelper htmlHelper, LocalizedString text, IDictionary htmlAttributes) { @@ -392,7 +392,7 @@ namespace Orchard.Mvc.Html { } public static IHtmlString AntiForgeryTokenValueOrchardLink(this HtmlHelper htmlHelper, string linkContents, string href, object htmlAttributes) { - return htmlHelper.AntiForgeryTokenValueOrchardLink(linkContents, href, new RouteValueDictionary(htmlAttributes)); + return htmlHelper.AntiForgeryTokenValueOrchardLink(linkContents, href, HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes)); } public static IHtmlString AntiForgeryTokenValueOrchardLink(this HtmlHelper htmlHelper, string linkContents, string href, IDictionary htmlAttributes) {