Hello
After migration I have an exception (ValueManagerContext.Storage is null) in my blazor application with using ValueManager in custom middleware.
(done by https://supportcenter.devexpress.com/ticket/details/t960767
C#public class Startup
{
//...
public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
//...
app.UseXaf();
app.UseMiddleware<HttpContextMiddleware>(env);
//...
}
}
and the middleware
C#public class HttpContextMiddleware
{
private readonly RequestDelegate next;
private readonly IWebHostEnvironment env;
public HttpContextMiddleware(RequestDelegate next, IWebHostEnvironment env)
{
this.next = next;
this.env = env;
}
public async Task InvokeAsync(HttpContext httpContext)
{
var v = ValueManager.GetValueManager<AppSetupHelper>("AppSetupHelper").Value; //Exception here
await this.next(httpContext);
}
}
Hello Andrey,
Since our Value Manager is scoped and middlewares and Razor pages have different scopes, we changed the ValueManager's behavior in this case to avoid its unexpected behavior. Please describe the task you want to accomplish using HttpContext so that we can look for an alternative solution.
You can review the article to see my scenario.
Hello Andrey,
Thank you for the update. In this ticket, you wrote:
Please consider using standard recommendations for Razor components to implement this: Get current URL in a Blazor component.
To get the NavigationManager in an XAF Controller, use the BlazorApplication.ServiceProvider.GetRequiredService method: Blazor - How to provide a file download or open an external hyper link using an XAF Action.
If these capabilities are insufficient for your goal, please describe what particular task you need to implement using these paths.
Thanks for the answer, but I think I can't use it in controller and need to explain my scenario in more details:
I use subdomain name to get correct db by name (multitenant) for xaf app. I check it in Startup.Configure method to return error page if db is not found.
I also use AppHelper singleton stored in ValueManager to handle this checking and other business logic. For example to get stored connection string with subdomain for DownloadController or set report params to get correct url to use in report scripts.
I also use this helper in my ASP as well version so I'd like to use it in my module project for both ASP and Blazor apps.
So I need somehow to:
In previous version I initialized the appHelper in value manager in middleware and then immediately checked db (and return error page if not found) and can then use it in common businesslogic, but now I'm not sure how can I achieve this?
This is my middleware method:
public async Task InvokeAsync(HttpContext httpContext) { var host = $"{httpContext.Request.Scheme}://{httpContext.Request.Host}"; AppSetupHelper.Instance.Init( () => { return host; }, () => { return null; }, () => env.ContentRootPath); await this.next(httpContext); }
and appHelper (same init method is using in global.asax of my ASP app)
#region Singleton public static AppSetupHelper Instance { get { if (ValueManager.GetValueManager<AppSetupHelper>("AppSetupHelper").Value == null) { ValueManager.GetValueManager<AppSetupHelper>("AppSetupHelper").Value = new AppSetupHelper(); } return ValueManager.GetValueManager<AppSetupHelper>("AppSetupHelper").Value; } } private AppSetupHelper() { } #endregion Singleton private bool isInitialized; public void Init( GetRequestUrlDelegate getHost, GetRequestParamsDelegate getRequestParams, GetPhysicalApplicationPathDelegate getPhysicalApplicationPath, bool force = false) { if (!isInitialized || force) { this.getHost = getHost; this.getRequestParams = getRequestParams; this.getPhysicalApplicationPath = getPhysicalApplicationPath; isInitialized = true; } }
and part of configure method with db checking (AppSetupHelper.Instance.GetConnectionString is using subdomain name from requested url)
private Dictionary<string, bool> checkedHosts = new Dictionary<string, bool>(); public void Configure(IApplicationBuilder app, IWebHostEnvironment env) { //... app.UseMiddleware<HttpContextMiddleware>(env); app.Use(async (context, next) => { if (!checkedHosts.ContainsKey(AppSetupHelper.Instance.Host)) { checkedHosts.Add(AppSetupHelper.Instance.Host, AppSetupHelper.Instance.IsDbExists(AppSetupHelper.Instance.GetConnectionString(Configuration.GetConnectionString("ConnectionString")))); } // Check db exists if (checkedHosts[AppSetupHelper.Instance.Host]) { await next.Invoke(); } else { await context.Response.WriteAsync(@"<!DOCTYPE html> <html xmlns='http://www.w3.org/1999/xhtml'> <head> <meta http-equiv='Content-Type' content='text/html; charset=utf-8' /> </head> <body style='background-color: #d3e4f6'> <div style='margin:50px auto; width: 500px; padding: 15px; border: 2px solid #dadada; background-color: #fff'> Requested domain doesn't exist. </div> </body> </html>"); } });
Sample of report params init from module.cs (it is not only one of using place)
private void ReportsDataSourceHelper_BeforeShowPreview(object sender, BeforeShowPreviewEventArgs e) { if (e.Report.DisplayName.StartsWith(UpdaterHelper.EliminationChartsReportName)) { e.Report.Parameters.Add(new DevExpress.XtraReports.Parameters.Parameter() { Name = nameof(AppSetupHelper.Instance.PhysicalApplicationPath), Value = AppSetupHelper.Instance.PhysicalApplicationPath, Type = typeof(string), Visible = false }); e.Report.Parameters.Add(new DevExpress.XtraReports.Parameters.Parameter() { Name = "Url", Value = AppSetupHelper.Instance.Host, Type = typeof(string), Visible = false }); } }
Could you help me with this please?