angular - 在 HttpInterceptor 中对 NGRX 可观察对象的 RXJS 链接/处理进行排序
问题描述
我在派生自 HttpInterceptor (Angular 8) 基类的类中将 JWT 访问令牌附加到 HTTP 请求的标头时遇到问题。
我已将问题缩小到在 Http 拦截器中执行操作的顺序。在令牌从 NGRX 存储(异步)返回之前,请求似乎已发送给调用者。
我不清楚如何确保仅在从 NGRX 商店收到令牌后才返回请求。
export class ServerInterceptor implements HttpInterceptor, OnDestroy {
private state$: Observable<any>;
private unsubscribeAll: Subject<any>;
constructor(
private store: Store<AuthenticationState>
) {
this.unsubscribeAll = new Subject();
this.state$ = this.store.select(getAccessToken);
}
ngOnDestroy(): void {
this.unsubscribeAll.next();
this.unsubscribeAll.complete();
}
intercept(request: HttpRequest<any>, next: HttpHandler): Observable<HttpEvent<any>> {
const output = next.handle(request).pipe(
takeUntil(this.unsubscribeAll),
withLatestFrom(
this.store.select(selectAuthenticationState), (_action, state) => {
console.log(state.accessToken);
if (state.accessToken) {
request = request.clone({
setHeaders: {
Authorization: `Bearer ${state.accessToken}`
}
});
}
console.log('1');
return request;
}
),
switchMap((req) => {
console.log('2');
return next.handle(req);
}),
);
console.log('3');
return output;
}
}
控制台的输出按3、1、2的顺序处理拦截器,应该是1、2、3处理。
解决方案
您可以考虑将select
运算符与您的选择器getAccessToken
和“链”可观察对象一起使用。如果您使用或运算符
,则无需订阅。Observable 会在一个值后自动完成。first()
take(1)
您的HttpInterceptor
可能代码如下所示:
export class ServerInterceptor implements HttpInterceptor {
constructor(
private store: Store<AuthenticationState>
) {}
intercept(request: HttpRequest<any>, next: HttpHandler): Observable<HttpEvent<any>> {
return this.store.pipe(
select(getAccessToken),
first(),
mergeMap(accessToken => {
const authReq = !!accessToken ? request.clone({
headers: request.headers.set('Authorization', `Bearer ${accessToken}`)
}) : request;
return next.handle(authReq);
})
);
}
}
关于这个主题的一些资源:
- 使用 NgRx Store Auth Token 和 Angular HttpInterceptor by Antony Derham
- WASi 使用NGRX将 JWT 令牌添加到 Angular HTTP 请求
推荐阅读
- python - 在Django中过滤时过滤数据/忽略空值
- moodle - moodlecloud 3.9.2 上缺少安装插件选项
- android - Resource Owner Password Credentials Grant 的使用有效时
- python - 将字符串转换为 unicode 视图
- python - 我如何在 Python 中使用 GCP 枚举?
- angular - 如何在角度的 p-editor 上使用 ql-image 添加验证?
- python - 如何从列表中的 X 文件创建 X 列表,并立即将 X 值分配给创建列表中的 dict 键。[Python]
- android - android导航将参数传递给片段构造函数
- javascript - VueJS 3 路由器在开发和生产中的行为不同
- java - ChronicleMap 是改进的并发 HashMap 吗?