找不到类型Microsoft的Blazor默认构造函数.AspNetCore.组件.批准AuthorizeRouteVi



我不明白为什么我总是得到这个异常。我试图保护Blazor WebAssembly

blazor.webassembly.js:1 WASM:System。MissingMethodException:默认找不到类型的构造函数微软AspNetCore。组件。批准AuthorizeRouteView

app.razor

<Router AppAssembly="@typeof(Program).Assembly">
    <Found Context="routeData">
        <AuthorizeRouteView RouteData="@routeData" DefaultLayout="@typeof(MainLayout)" />
    </Found>
    <NotFound>
        <CascadingAuthenticationState>
            <LayoutView Layout="@typeof(MainLayout)">
                <p>Sorry, there's nothing at this address.</p>
            </LayoutView>
        </CascadingAuthenticationState>
    </NotFound>
</Router>

这是客户端程序.cs

    public class Program
    {
        public static async Task Main(string[] args)
        {
            var builder = WebAssemblyHostBuilder.CreateDefault(args);
            builder.RootComponents.Add<App>("app");
            builder.Services.AddScoped<AuthenticationStateProvider, ApiAuthenticationStateProvider>();
            builder.Services.AddAuthorizationCore(options => { });            
            await builder.Build().RunAsync();
        }
    }

自定义身份验证状态提供程序

public class ApiAuthenticationStateProvider : AuthenticationStateProvider
{
    private readonly HttpClient _httpClient;
    private readonly ILocalStorageService _localStorage;
    public ApiAuthenticationStateProvider(HttpClient httpClient, ILocalStorageService localStorage)
    {
        _httpClient = httpClient;
        _localStorage = localStorage;
    }
    public override async Task<AuthenticationState> GetAuthenticationStateAsync()
    {
        var savedToken = await _localStorage.GetItemAsync<string>("authToken");
        if (string.IsNullOrWhiteSpace(savedToken))
        {
            return new AuthenticationState(new ClaimsPrincipal(new ClaimsIdentity()));
        }
        _httpClient.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("bearer", savedToken);
        return new AuthenticationState(new ClaimsPrincipal(new ClaimsIdentity(ParseClaimsFromJwt(savedToken), "jwt")));
    }
    public void MarkUserAsAuthenticated(string email)
    {
        ClaimsPrincipal authenticatedUser = new ClaimsPrincipal(new ClaimsIdentity(new[] { new Claim(ClaimTypes.Name, email) }, "apiauth"));
        var authState = Task.FromResult(new AuthenticationState(authenticatedUser));
        NotifyAuthenticationStateChanged(authState);
    }
    public void MarkUserAsLoggedOut()
    {
        var anonymousUser = new ClaimsPrincipal(new ClaimsIdentity());
        var authState = Task.FromResult(new AuthenticationState(anonymousUser));
        NotifyAuthenticationStateChanged(authState);
    }
    private IEnumerable<Claim> ParseClaimsFromJwt(string jwt)
    {
        var claims = new List<Claim>();
        var payload = jwt.Split('.')[1];
        var jsonBytes = ParseBase64WithoutPadding(payload);
        var keyValuePairs = JsonSerializer.Deserialize<Dictionary<string, object>>(jsonBytes);
        keyValuePairs.TryGetValue(ClaimTypes.Role, out object roles);
        if (roles != null)
        {
            if (roles.ToString().Trim().StartsWith("["))
            {
                var parsedRoles = JsonSerializer.Deserialize<string[]>(roles.ToString());
                foreach (var parsedRole in parsedRoles)
                {
                    claims.Add(new Claim(ClaimTypes.Role, parsedRole));
                }
            }
            else
            {
                claims.Add(new Claim(ClaimTypes.Role, roles.ToString()));
            }
            keyValuePairs.Remove(ClaimTypes.Role);
        }
        claims.AddRange(keyValuePairs.Select(kvp => new Claim(kvp.Key, kvp.Value.ToString())));
        return claims;
    }
    private byte[] ParseBase64WithoutPadding(string base64)
    {
        switch (base64.Length % 4)
        {
            case 2: base64 += "=="; break;
            case 3: base64 += "="; break;
        }
        return Convert.FromBase64String(base64);
    }
}

}

服务器启动.cs

  public class Startup
    {
        public IConfiguration Configuration { get; }
        // This method gets called by the runtime. Use this method to add services to the container.
        // For more information on how to configure your application, visit https://go.microsoft.com/fwlink/?LinkID=398940
        public Startup(IConfiguration configuration)
        {
            Configuration = configuration;
        }
        public void ConfigureServices(IServiceCollection services)
        {
            services.AddAuthentication(JwtBearerDefaults.AuthenticationScheme)
                .AddJwtBearer(options =>
                {
                    options.TokenValidationParameters = new TokenValidationParameters
                    {
                        ValidateIssuer = true,
                        ValidateAudience = true,
                        ValidateLifetime = true,
                        ValidateIssuerSigningKey = true,
                        ValidIssuer = Configuration["JwtIssuer"],
                        ValidAudience = Configuration["JwtAudience"],
                        IssuerSigningKey = new SymmetricSecurityKey(Encoding.UTF8.GetBytes(Configuration["JwtSecurityKey"]))
                    };
                });
            services.AddBlazoredLocalStorage();
            services.AddAuthorizationCore();
            services.AddScoped<AuthenticationStateProvider, ApiAuthenticationStateProvider>();
            services.AddScoped<IAuthService, AuthService>();
            services.AddMvc();
            services.AddResponseCompression(opts =>
            {
                opts.MimeTypes = ResponseCompressionDefaults.MimeTypes.Concat(
                    new[] { "application/octet-stream" });
            });

            services.AddDbContext<ApplicationDbContext>(options =>
                options.UseSqlServer(Configuration.GetConnectionString("DefaultConnection")));
            services.AddDefaultIdentity<IdentityUser>()
                .AddEntityFrameworkStores<ApplicationDbContext>();
        }
        // This method gets called by the runtime. Use this method to configure the HTTP request pipeline.
        public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
        {
            app.UseResponseCompression();
            if (env.IsDevelopment())
            {
                app.UseDeveloperExceptionPage();
                app.UseBlazorDebugging();
            }
            app.UseStaticFiles();
            app.UseClientSideBlazorFiles<Client.Program>();
            app.UseRouting();
            app.UseAuthentication();
            app.UseAuthorization();
            app.UseEndpoints(endpoints =>
            {
                endpoints.MapDefaultControllerRoute();
                endpoints.MapFallbackToClientSideBlazor<Client.Program>("index.html");
            });
        }
    }

在App.razor文件的顶部添加@using Microsoft.AspNetCore.Components.Authorization

向程序添加选项和授权服务。主(客户端(:

builder.Services.AddOptions();
builder.Services.AddAuthorizationCore();

注:以下(客户端(:

 services.AddBlazoredLocalStorage();
 services.AddAuthorizationCore();
 services.AddScoped<AuthenticationStateProvider, 
                          ApiAuthenticationStateProvider>();
        services.AddScoped<IAuthService, AuthService>();

属于客户端,而不是服务器,尽管最终它的配置相同。。。

注:以下(客户端(:

services.AddScoped<AuthenticationStateProvider, 
                          ApiAuthenticationStateProvider>();

应该是(客户端(:

services.AddScoped<ApiAuthenticationStateProvider>();
services.AddScoped<AuthenticationStateProvider>(provider => provider.GetRequiredService<ApiAuthenticationStateProvider>());

您应该注意启动类中的订单

更新:

禁用链接器,正如我在下面的评论中所建议的,似乎是有效的。然而,这应该是一个临时解决方案。以下是禁用链接器的方式:<BlazorLinkOnBuild>false</BlazorLinkOnBuild>

通过禁用链接器,可以防止删除未使用或未引用的代码,这可能是导致错误的原因。。。通过启用链接器,您可以实现这一点。然而,这个代码:

builder.Services.AddOptions();
builder.Services.AddAuthorizationCore();

本应阻止链接器剥离未引用的逻辑。更好的是,在Program类中添加一个自定义AuthenticationStateProvider实现:

 builder.Services.AddScoped<AuthenticationStateProvider, 
                          ApiAuthenticationStateProvider>(); 

本应解决此问题。你是那样做的吗?

同样,禁用链接器是一个临时解决方案。

我遇到了这个问题。当我从NuGet安装Microsoft.AspNetCore.Components.Authorization时,它就消失了。@using Microsoft.AspNetCore.Components.Authorization指令在visual Studio中没有提供包缺少的任何视觉线索

添加

builder.Services.AddApiAuthorization();

并安装

Microsoft。AspNetCore。组件。WebAssembly。身份验证

我的主要方法:

    public static async Task Main(string[] args)
    {
        var builder = WebAssemblyHostBuilder.CreateDefault(args);
        builder.RootComponents.Add<App>("app");
        builder.Services.AddOptions();
        builder.Services.AddAuthorizationCore();
        builder.Services.AddScoped<CustomStateProvider>();
        builder.Services.AddScoped<AuthenticationStateProvider>(s => s.GetRequiredService<CustomStateProvider>());
        builder.Services.AddTransient(sp => new HttpClient { BaseAddress = new Uri(builder.HostEnvironment.BaseAddress) });
        builder.Services.AddApiAuthorization();
        await builder.Build().RunAsync();
    }
禁用链接器似乎是目前唯一可行的解决方案。

我添加了以下代码:

builder.Services.AddOptions();
builder.Services.AddAuthorizationCore();

并添加了自定义AuthenticationStateProvider,但没有成功。

github上的问题似乎已通过此解决方法解决:https://github.com/dotnet/aspnetcore/issues/22298

相关内容

  • 没有找到相关文章

最新更新