Ticket T1043424
Visible to All Users

Xaf valueManager doesn't work in mIddleware

created 3 years ago

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); } }
Show previous comments (1)

    You can review the article to see my scenario.

    Anatol (DevExpress) 3 years ago

      Hello Andrey,

      Thank you for the update. In this ticket, you wrote:

      I need to resolve using http context in my module (to get phisycal path and base url path)

      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:

        • resolve subdomain database checking in configure method (or another place on start session and return error page)
        • set current url (and physicalPath) to AppHelper in ValueManager to use it in module business logic.

        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:

        C#
        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)

        C#
        #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)

        C#
        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)

        C#
        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?

        Answers approved by DevExpress Support

        created 3 years ago

        Hello,

        For example, you can register the AppSetupHelper class as a scoped service - refer to the following article for details: Dependency injection in ASP.NET Core. Initialize it in your HttpContextMiddleware middleware. Pass a request state to the Blazor application as described in the following article: Blazor and shared state. To access the service in a controller, use the BlazorApplication.ServiceProvider.GetRequiredService method. Please refer to the attachment for a sample. I hope you find this information helpful.

        Thanks,
        Ilya P

          Show previous comments (8)
          DevExpress Support Team 3 years ago

            It is still necessary to pass initial data to Blazor, so initialApplicationStorage and initialApplicationState are required.

              Thanks for your help!

              DevExpress Support Team 3 years ago

                You are always welcome. Feel free to contact us any time.

                Disclaimer: The information provided on DevExpress.com and affiliated web properties (including the DevExpress Support Center) is provided "as is" without warranty of any kind. Developer Express Inc disclaims all warranties, either express or implied, including the warranties of merchantability and fitness for a particular purpose. Please refer to the DevExpress.com Website Terms of Use for more information in this regard.

                Confidential Information: Developer Express Inc does not wish to receive, will not act to procure, nor will it solicit, confidential or proprietary materials and information from you through the DevExpress Support Center or its web properties. Any and all materials or information divulged during chats, email communications, online discussions, Support Center tickets, or made available to Developer Express Inc in any manner will be deemed NOT to be confidential by Developer Express Inc. Please refer to the DevExpress.com Website Terms of Use for more information in this regard.