Я пишу плагин для nopcommerce beta 4.3 и пытаюсь переписать общий c маршрут (Category / Slug). Приложение было перемещено в. net core 3.x, и поэтому маршрутизация теперь находится в промежуточном программном обеспечении и обрабатывается иначе, чем раньше. Я попытался перезаписать Nop.Web.Infrastructure.GenericUrlRouteProvider и Nop.Web.Framework. Mvc .Routing.SlugRouteTransformer, как показано ниже ...
public partial class GenericUrlRouteProviderExt : IRouteProvider
{
#region Methods
/// <summary>
/// Register routes
/// </summary>
/// <param name="endpointRouteBuilder">Route builder</param>
public void RegisterRoutes(IEndpointRouteBuilder endpointRouteBuilder)
{
var pattern = "{SeName}";
if (DataSettingsManager.DatabaseIsInstalled)
{
var localizationSettings = endpointRouteBuilder.ServiceProvider.GetRequiredService<LocalizationSettings>();
if (localizationSettings.SeoFriendlyUrlsForLanguagesEnabled)
{
var langservice = endpointRouteBuilder.ServiceProvider.GetRequiredService<ILanguageService>();
var languages = langservice.GetAllLanguages().ToList();
pattern = "{language:lang=" + languages.FirstOrDefault().UniqueSeoCode + "}/{SeName}";
}
}
endpointRouteBuilder.MapDynamicControllerRoute<SlugRouteTransformerExt>(pattern);
endpointRouteBuilder.MapControllerRoute("WineCategory", pattern,
new { controller = "Wine", action = "WineCategory" });
}
#endregion
#region Properties
/// <summary>
/// Gets a priority of route provider
/// </summary>
/// <remarks>
/// it should be the last route. we do not set it to -int.MaxValue so it could be overridden (if required)
/// </remarks>
public int Priority => 1000;
/// <summary>
/// Gets the order value of endpoint.
/// </summary>
/// <remarks>
/// The order value provides absolute control over the priority
/// of an endpoint. Endpoints with a lower numeric value of order have higher priority.
/// </remarks>
public int Order => 1;
#endregion
}
и
public class SlugRouteTransformerExt : DynamicRouteValueTransformer
{
private readonly ILanguageService _languageService;
private readonly LocalizationSettings _localizationSettings;
private readonly IUrlRecordService _urlRecordService;
public SlugRouteTransformerExt(ILanguageService languageService,
LocalizationSettings localizationSettings,
IUrlRecordService urlRecordService)
{
_languageService = languageService;
_localizationSettings = localizationSettings;
_urlRecordService = urlRecordService;
}
public override ValueTask<RouteValueDictionary> TransformAsync(HttpContext httpContext, RouteValueDictionary values)
{
if (values == null)
return new ValueTask<RouteValueDictionary>(values);
if (!values.TryGetValue("SeName", out var slugValue) || string.IsNullOrEmpty(slugValue as string))
return new ValueTask<RouteValueDictionary>(values);
var slug = slugValue as string;
//performance optimization, we load a cached verion here. It reduces number of SQL requests for each page load
var urlRecord = _urlRecordService.GetBySlug(slug);
//no URL record found
if (urlRecord == null)
return new ValueTask<RouteValueDictionary>(values);
//virtual directory path
var pathBase = httpContext.Request.PathBase;
//if URL record is not active let's find the latest one
if (!urlRecord.IsActive)
{
var activeSlug = _urlRecordService.GetActiveSlug(urlRecord.EntityId, urlRecord.EntityName, urlRecord.LanguageId);
if (string.IsNullOrEmpty(activeSlug))
return new ValueTask<RouteValueDictionary>(values);
//redirect to active slug if found
values[NopPathRouteDefaults.ControllerFieldKey] = "Common";
values[NopPathRouteDefaults.ActionFieldKey] = "InternalRedirect";
values[NopPathRouteDefaults.UrlFieldKey] = $"{pathBase}/{activeSlug}{httpContext.Request.QueryString}";
values[NopPathRouteDefaults.PermanentRedirectFieldKey] = true;
httpContext.Items["nop.RedirectFromGenericPathRoute"] = true;
return new ValueTask<RouteValueDictionary>(values);
}
//Ensure that the slug is the same for the current language,
//otherwise it can cause some issues when customers choose a new language but a slug stays the same
if (_localizationSettings.SeoFriendlyUrlsForLanguagesEnabled)
{
var urllanguage = values["language"];
if (urllanguage != null && !string.IsNullOrEmpty(urllanguage.ToString()))
{
var language = _languageService.GetAllLanguages().FirstOrDefault(x => x.UniqueSeoCode.ToLowerInvariant() == urllanguage.ToString().ToLowerInvariant());
if (language == null)
language = _languageService.GetAllLanguages().FirstOrDefault();
var slugForCurrentLanguage = _urlRecordService.GetActiveSlug(urlRecord.EntityId, urlRecord.EntityName, language.Id);
if (!string.IsNullOrEmpty(slugForCurrentLanguage) && !slugForCurrentLanguage.Equals(slug, StringComparison.InvariantCultureIgnoreCase))
{
//we should make validation above because some entities does not have SeName for standard (Id = 0) language (e.g. news, blog posts)
//redirect to the page for current language
values[NopPathRouteDefaults.ControllerFieldKey] = "Common";
values[NopPathRouteDefaults.ActionFieldKey] = "InternalRedirect";
values[NopPathRouteDefaults.UrlFieldKey] = $"{pathBase}/{slugForCurrentLanguage}{httpContext.Request.QueryString}";
values[NopPathRouteDefaults.PermanentRedirectFieldKey] = false;
httpContext.Items["nop.RedirectFromGenericPathRoute"] = true;
return new ValueTask<RouteValueDictionary>(values);
}
}
}
//since we are here, all is ok with the slug, so process URL
switch (urlRecord.EntityName.ToLowerInvariant())
{
case "product":
values[NopPathRouteDefaults.ControllerFieldKey] = "Product";
values[NopPathRouteDefaults.ActionFieldKey] = "ProductDetails";
values[NopPathRouteDefaults.ProductIdFieldKey] = urlRecord.EntityId;
values[NopPathRouteDefaults.SeNameFieldKey] = urlRecord.Slug;
break;
case "producttag":
values[NopPathRouteDefaults.ControllerFieldKey] = "Catalog";
values[NopPathRouteDefaults.ActionFieldKey] = "ProductsByTag";
values[NopPathRouteDefaults.ProducttagIdFieldKey] = urlRecord.EntityId;
values[NopPathRouteDefaults.SeNameFieldKey] = urlRecord.Slug;
break;
case "category":
if (!pathBase.ToString().Contains("wine-category"))
{
values[NopPathRouteDefaults.ControllerFieldKey] = "Wine";
values[NopPathRouteDefaults.ActionFieldKey] = "WineCategory";
values[NopPathRouteDefaults.CategoryIdFieldKey] = urlRecord.EntityId;
values[NopPathRouteDefaults.SeNameFieldKey] = urlRecord.Slug;
}
else
{
values[NopPathRouteDefaults.ControllerFieldKey] = "Catalog";
values[NopPathRouteDefaults.ActionFieldKey] = "Category";
values[NopPathRouteDefaults.CategoryIdFieldKey] = urlRecord.EntityId;
values[NopPathRouteDefaults.SeNameFieldKey] = urlRecord.Slug;
}
break;
case "manufacturer":
values[NopPathRouteDefaults.ControllerFieldKey] = "Catalog";
values[NopPathRouteDefaults.ActionFieldKey] = "Manufacturer";
values[NopPathRouteDefaults.ManufacturerIdFieldKey] = urlRecord.EntityId;
values[NopPathRouteDefaults.SeNameFieldKey] = urlRecord.Slug;
break;
case "vendor":
values[NopPathRouteDefaults.ControllerFieldKey] = "Catalog";
values[NopPathRouteDefaults.ActionFieldKey] = "Vendor";
values[NopPathRouteDefaults.VendorIdFieldKey] = urlRecord.EntityId;
values[NopPathRouteDefaults.SeNameFieldKey] = urlRecord.Slug;
break;
case "newsitem":
values[NopPathRouteDefaults.ControllerFieldKey] = "News";
values[NopPathRouteDefaults.ActionFieldKey] = "NewsItem";
values[NopPathRouteDefaults.NewsItemIdFieldKey] = urlRecord.EntityId;
values[NopPathRouteDefaults.SeNameFieldKey] = urlRecord.Slug;
break;
case "blogpost":
values[NopPathRouteDefaults.ControllerFieldKey] = "Blog";
values[NopPathRouteDefaults.ActionFieldKey] = "BlogPost";
values[NopPathRouteDefaults.BlogPostIdFieldKey] = urlRecord.EntityId;
values[NopPathRouteDefaults.SeNameFieldKey] = urlRecord.Slug;
break;
case "topic":
values[NopPathRouteDefaults.ControllerFieldKey] = "Topic";
values[NopPathRouteDefaults.ActionFieldKey] = "TopicDetails";
values[NopPathRouteDefaults.TopicIdFieldKey] = urlRecord.EntityId;
values[NopPathRouteDefaults.SeNameFieldKey] = urlRecord.Slug;
break;
default:
//no record found, thus generate an event this way developers could insert their own types
break;
}
return new ValueTask<RouteValueDictionary>(values);
}
}
но получите сообщение об ошибке: Использование ExpandEndpoint требует, чтобы замененная конечная точка имела уникальный приоритет. Следующие конечные точки были найдены с одинаковым приоритетом ...
Я ищу эффективный способ перехватить указанный маршрут и перенаправить его ...
Я также пытался это сделать в PluginStratup не знаю, к кому подойти ...
public void Configure(IApplicationBuilder application)
{
//our custom middleware
application.Use(async (context, next) =>
{
await next();
});
}
Кто-нибудь изучал это и нашел решение?