我想要的是:
在我的应用程序中,我想为电子邮件使用模板。不幸的是,我之前在另一个项目中使用的代码不再起作用。
引发的错误:
Could not find an IRouter associated with the ActionContext.
If your application is using endpoint routing then you can get a IUrlHelperFactory with dependency injection
and use it to create a UrlHelper, or use Microsoft.AspNetCore.Routing.LinkGenerator.'
我不知道如何解决此问题,因为我找不到任何注入IUrlHelper
的方法。我不确定为什么甚至需要这样做,因为无论如何它都不存在。
字符串渲染方法:
public async Task<string> RenderToStringAsync(string viewName, object model)
{
var httpContext = new DefaultHttpContext { RequestServices = _serviceProvider };
var actionContext = new ActionContext(httpContext, new RouteData(), new ActionDescriptor());
using (var sw = new StringWriter())
{
var viewResult = FindView(actionContext, viewName);
if (viewResult == null)
{
throw new ArgumentNullException($"{viewName} does not match any available view");
}
var viewDictionary = new ViewDataDictionary(new EmptyModelMetadataProvider(), new ModelStateDictionary())
{
Model = model
};
var viewContext = new ViewContext(
actionContext,
viewResult,
viewDictionary,
new TempDataDictionary(actionContext.HttpContext, _tempDataProvider),
sw,
new HtmlHelperOptions());
await viewResult.RenderAsync(viewContext); // Throws error <<<
return sw.ToString();
}
}
参考方案
找到了解决方案:
我认为,由于创建ActionContext并为其提供空的RouteData对象会给我IRouter异常,因此,下一个最佳解决方案是查看我是否可以仅使用实际请求中的HttpContext。
因此,通过依赖性注入,我添加了_httpContextAccessor并使用了可用的HttpContext对象。
为了完整起见,我将分享最终的实现:
要将视图呈现为HTML:
RenderToString(string,object);
var htmlBody = await Renderer.RenderToString($"/Views/Shared/confirm-email.cshtml", model);
服务:
public class ViewRenderService : IViewRenderService
{
private readonly IRazorViewEngine _razorViewEngine;
private readonly ITempDataProvider _tempDataProvider;
private readonly IHttpContextAccessor _contextAccessor;
public ViewRenderService(IRazorViewEngine razorViewEngine,
ITempDataProvider tempDataProvider,
IHttpContextAccessor contextAccessor)
{
_razorViewEngine = razorViewEngine;
_tempDataProvider = tempDataProvider;
_contextAccessor = contextAccessor;
}
public async Task<string> RenderToString(string viewName, object model)
{
var actionContext = new ActionContext(_contextAccessor.HttpContext, _contextAccessor.HttpContext.GetRouteData(), new ActionDescriptor());
await using var sw = new StringWriter();
var viewResult = FindView(actionContext, viewName);
if (viewResult == null)
{
throw new ArgumentNullException($"{viewName} does not match any available view");
}
var viewDictionary = new ViewDataDictionary(new EmptyModelMetadataProvider(), new ModelStateDictionary())
{
Model = model
};
var viewContext = new ViewContext(
actionContext,
viewResult,
viewDictionary,
new TempDataDictionary(actionContext.HttpContext, _tempDataProvider),
sw,
new HtmlHelperOptions()
);
await viewResult.RenderAsync(viewContext);
return sw.ToString();
}
private IView FindView(ActionContext actionContext, string viewName)
{
var getViewResult = _razorViewEngine.GetView(executingFilePath: null, viewPath: viewName, isMainPage: true);
if (getViewResult.Success)
{
return getViewResult.View;
}
var findViewResult = _razorViewEngine.FindView(actionContext, viewName, isMainPage: true);
if (findViewResult.Success)
{
return findViewResult.View;
}
var searchedLocations = getViewResult.SearchedLocations.Concat(findViewResult.SearchedLocations);
var errorMessage = string.Join(
Environment.NewLine,
new[] { $"Unable to find view '{viewName}'. The following locations were searched:" }.Concat(searchedLocations));
throw new InvalidOperationException(errorMessage);
}
}
现在,它可以完美运行了。
ASP.NET Core Singleton实例与瞬态实例的性能 - c#在ASP.NET Core依赖注入中,我只是想知道注册Singleton实例是否会比注册Transient实例更好地提高性能?在我看来,对于Singleton实例,创建新对象和相关对象只需花费一次时间。对于Transient实例,此成本将针对每个服务请求重复。因此Singleton似乎更好。但是,在Singleton上使用Transient时,我们可以获得多…
如何在ASP.NET Core Web应用程序中增加JSON反序列化MaxDepth限制 - c#我们正在将ASP.NET Core 2.1与.NET Framework 4.6.2结合使用。我们有一个客户需要向我们的Web应用程序发送一个很大程度上嵌套的json结构。当他们进行此调用时,我们将输出以下日志并返回错误: 读取器的MaxDepth超过了32。路径“ super.long.path.to property”,第1行,位置42111。”我浏览了…
ASP.NET Core-在Singleton注入上存储库依赖项注入失败 - c#我正在使用SoapCore为我的ASP.NET Core MVC应用程序创建Web服务。我正在使用Entity Framework Core和简单的存储库模式来获取我的数据库数据。我通过Startup.cs中的.AddSingleton()注入存储库类:services.AddSingleton<IImportRepository, ImportRep…
为什么Swashbuckle.aspnet.core.swagger无法识别 - c#我已经通过nuget package manager安装了Swashbuckle.AspNetCore.Swagger,并在顶部包含了Swashbuckle.AspNetCore.Swagger,但是在Info{ Title}方法上却出现了错误。任何人都可以建议如何解决此问题。 services.AddSwaggerGen(c => { c.Swagg…
.NET Core 3 / EF.Core 3:QueryModelGenerator,RelationalQueryModelVisitor等发生了什么 - c#我想知道发生了什么事:QueryModelGenerator,RelationalQueryModelVisitor和queryCompilationContext.CreateQuerymodelVisitor()似乎都消失了。我有以下代码,但无法尝试将其从2.2转换为.NET Core 3public static string ToSql<TEn…