当Angular的HttpInterceptor在请求新的JWT令牌时导致无限循环时,通常是因为在拦截器中不正确地处理了JWT令牌的过期和更新逻辑。以下是一个解决方法的代码示例:
首先,创建一个名为 TokenInterceptor
的拦截器类,它会拦截所有的HTTP请求:
import { Injectable } from '@angular/core';
import { HttpInterceptor, HttpRequest, HttpHandler, HttpEvent } from '@angular/common/http';
import { Observable } from 'rxjs';
import { AuthService } from 'path/to/auth.service';
@Injectable()
export class TokenInterceptor implements HttpInterceptor {
constructor(private authService: AuthService) {}
intercept(request: HttpRequest, next: HttpHandler): Observable> {
// 检查是否有JWT令牌
if (this.authService.getToken()) {
// 在请求头中添加JWT令牌
request = request.clone({
setHeaders: {
Authorization: `Bearer ${this.authService.getToken()}`
}
});
}
return next.handle(request);
}
}
然后,创建一个名为 AuthService
的服务类,用于管理JWT令牌的过期和更新逻辑:
import { Injectable } from '@angular/core';
import { BehaviorSubject, Observable } from 'rxjs';
@Injectable()
export class AuthService {
private tokenSubject: BehaviorSubject = new BehaviorSubject(null);
// 模拟从服务器获取JWT令牌
private getTokenFromServer(): Observable {
// 发起HTTP请求获取新的JWT令牌
// 返回Observable,表示异步操作
}
// 获取当前JWT令牌的方法
getToken(): string {
return this.tokenSubject.getValue();
}
// 更新JWT令牌的方法
updateToken(): Observable {
if (!this.getToken()) {
// 如果当前没有JWT令牌,则直接从服务器获取
return this.getTokenFromServer().pipe(
tap((token: string) => {
this.tokenSubject.next(token); // 更新订阅者的值
})
);
} else {
// 如果当前有JWT令牌,则先从服务器获取新的JWT令牌,然后更新订阅者的值
return this.getTokenFromServer().pipe(
tap((token: string) => {
this.tokenSubject.next(token); // 更新订阅者的值
})
);
}
}
}
最后,在Angular的 AppModule
中配置拦截器和服务:
import { NgModule } from '@angular/core';
import { HttpClientModule, HTTP_INTERCEPTORS } from '@angular/common/http';
import { TokenInterceptor } from 'path/to/token-interceptor';
import { AuthService } from 'path/to/auth.service';
@NgModule({
imports: [
HttpClientModule
],
providers: [
AuthService,
{
provide: HTTP_INTERCEPTORS,
useClass: TokenInterceptor,
multi: true
}
]
})
export class AppModule { }
通过以上代码示例,我们在拦截器中检查JWT令牌是否存在,并在请求头中添加JWT令牌。在 AuthService
中,我们通过一个 BehaviorSubject
对象来管理JWT令牌的值,并提供了一个 updateToken
方法来更新JWT令牌。这样,在每次请求新的JWT令牌时,拦截器会正确处理JWT令牌的更新逻辑,避免了无限循环的问题。
上一篇:Angular的HttpInterceptor在嵌套的Observables中不触发。
下一篇:Angular的HTTP客户端接受responseType为'json',但不接受responseType为'text'的情况怎么办?