admin管理员组文章数量:1345113
I'm developing a Blazor Server
application that calls an external API. After successful login, I store the JWT token in ProtectedSessionStorage
. While I can retrieve the token fine in Razor components, I get a JavaScript interop exception when trying to access it from a custom DelegatingHandler
.
Error Details:
System.InvalidOperationException: 'JavaScript interop calls cannot be issued at this time. This is because the component is being statically rendered. When prerendering is enabled, JavaScript interop calls can only be performed during the OnAfterRenderAsync lifecycle method.'
Service Registration:
builder.Services.AddScoped<ProtectedSessionStorage>();
builder.Services.AddScoped<TokenService>();
builder.Services.AddScoped<AuthTokenHandler>();
builder.Services.AddHttpClient("AuthenticatedClient", client =>
{
client.BaseAddress = new Uri(BaseUrl);
}).AddHttpMessageHandler<AuthTokenHandler>();
TokenService.cs
public class TokenService(ProtectedSessionStorage sessionStorage)
{
readonly string Key = "AccessToken";
public async Task SetAsync(string value)
{
try
{
await sessionStorage.SetAsync(Key, value);
}
catch { }
}
public async Task<string> GetAsync()
{
try
{
string result = string.Empty;
var tokenResult = await sessionStorage.GetAsync<string>(Key);
if (tokenResult.Success)
{
result = tokenResult.Value ?? string.Empty;
}
return result;
}
catch
{
return string.Empty;
}
}
}
AuthTokenHandler.cs
public class AuthTokenHandler(TokenService tokenService) : DelegatingHandler
{
protected override async Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken)
{
var token = await tokenService.GetAsync();//exception occurs in this line
if (!string.IsNullOrEmpty(token))
{
request.Headers.Authorization = new AuthenticationHeaderValue("Bearer", token);
}
return await base.SendAsync(request, cancellationToken);
}
}
Razor Component
string token = await tokenService.GetAsync(); //this line can retrieve token
var httpClient = HttpClientFactory.CreateClient("AuthenticatedClient");
var response = await httpClient.GetAsync(RelativeUrl);
if (response.IsSuccessStatusCode)
{
message = "Authorized";
var result = await response.Content.ReadAsStringAsync();
}
else if (response.StatusCode == System.Net.HttpStatusCode.Unauthorized)
{
message = "Unauthorized";
}
I'm developing a Blazor Server
application that calls an external API. After successful login, I store the JWT token in ProtectedSessionStorage
. While I can retrieve the token fine in Razor components, I get a JavaScript interop exception when trying to access it from a custom DelegatingHandler
.
Error Details:
System.InvalidOperationException: 'JavaScript interop calls cannot be issued at this time. This is because the component is being statically rendered. When prerendering is enabled, JavaScript interop calls can only be performed during the OnAfterRenderAsync lifecycle method.'
Service Registration:
builder.Services.AddScoped<ProtectedSessionStorage>();
builder.Services.AddScoped<TokenService>();
builder.Services.AddScoped<AuthTokenHandler>();
builder.Services.AddHttpClient("AuthenticatedClient", client =>
{
client.BaseAddress = new Uri(BaseUrl);
}).AddHttpMessageHandler<AuthTokenHandler>();
TokenService.cs
public class TokenService(ProtectedSessionStorage sessionStorage)
{
readonly string Key = "AccessToken";
public async Task SetAsync(string value)
{
try
{
await sessionStorage.SetAsync(Key, value);
}
catch { }
}
public async Task<string> GetAsync()
{
try
{
string result = string.Empty;
var tokenResult = await sessionStorage.GetAsync<string>(Key);
if (tokenResult.Success)
{
result = tokenResult.Value ?? string.Empty;
}
return result;
}
catch
{
return string.Empty;
}
}
}
AuthTokenHandler.cs
public class AuthTokenHandler(TokenService tokenService) : DelegatingHandler
{
protected override async Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken)
{
var token = await tokenService.GetAsync();//exception occurs in this line
if (!string.IsNullOrEmpty(token))
{
request.Headers.Authorization = new AuthenticationHeaderValue("Bearer", token);
}
return await base.SendAsync(request, cancellationToken);
}
}
Razor Component
string token = await tokenService.GetAsync(); //this line can retrieve token
var httpClient = HttpClientFactory.CreateClient("AuthenticatedClient");
var response = await httpClient.GetAsync(RelativeUrl);
if (response.IsSuccessStatusCode)
{
message = "Authorized";
var result = await response.Content.ReadAsStringAsync();
}
else if (response.StatusCode == System.Net.HttpStatusCode.Unauthorized)
{
message = "Unauthorized";
}
Share
Improve this question
asked yesterday
Anuj KarkiAnuj Karki
6176 silver badges29 bronze badges
1 Answer
Reset to default 0You could try create a CustomHttpClientFactory instead of using DelegatingHandler to pass token parameter when creating httpclient.
public class CustomHttpClientFactory(HttpClient client)
{
public HttpClient CreateClient(string? token)
{
if (!string.IsNullOrEmpty(token))
{
client.DefaultRequestHeaders.Authorization = new AuthenticationHeaderValue("Bearer", token);
}
//client.BaseAddress = new Uri(BaseUrl);
return client;
}
}
builder.Services.AddScoped<CustomHttpClientFactory>();
builder.Services.AddHttpClient<CustomHttpClientFactory>();
Then use directly
@inject CustomHttpClientFactory HttpClientFactory
...
string token = await tokenService.GetAsync();
var httpClient = HttpClientFactory.CreateClient(token);
本文标签: cBlazor Server Unable to access ProtectedSessionStorage value from HttpMessageHandlerStack Overflow
版权声明:本文标题:c# - Blazor Server: Unable to access ProtectedSessionStorage value from HttpMessageHandler - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743779125a2537517.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论