Обновить токен в Angular 4 для нескольких вызовов API
Я пытаюсь реализовать концепцию обновления токенов в своем веб-приложении.
При обновлении страницы я вызываю 4 API, а когда срок действия маркера доступа истекает, я вызываю сервер, чтобы получить новый токен доступа, основанный на маркере обновления.
Таким образом, в моем случае я могу получить новый токен доступа, но снова не могу инициировать 4 вызова API до тех пор, пока я не обновлю страницу вручную или не перезагружу страницу из службы. Но я не хочу перезагружать страницу и хочу, чтобы вызовы API выполнялись без ведома конечного пользователя.
Дайте некоторое предложение или идею сделать это.
2 ответа
Вы можете использовать Angular HttpInterceptor для решения вашей проблемы. смотрите фрагмент ниже.
@Injectable()
export class KgRequestInterceptorService implements HttpInterceptor {
authenticationService: MyAuthenticationService;
snackbarService: KgSnackbarService
constructor(private injector: Injector) { }
addBearerAndHeaders(req: HttpRequest<any>, token: string, overwrite?: boolean): HttpRequest<any> {
reqHeaders = reqHeaders.set("Authorization", 'Bearer ' + token);
return req.clone({ headers: reqHeaders });
}
intercept(req: HttpRequest<any>, next: HttpHandler): Observable<HttpSentEvent | HttpHeaderResponse | HttpProgressEvent | HttpResponse<any> | HttpUserEvent<any>> {
this.authenticationService = this.authenticationService ? this.authenticationService : this.injector.get<MyAuthenticationService>(MyAuthenticationService);
return next.handle(this.addBearerAndHeaders(req, this.authenticationService.accessToken)).pipe(
catchError((error, cought) => {
if (error instanceof HttpErrorResponse) {
switch ((<HttpErrorResponse>error).status) {
case 400:
return this.handle400Error(error);
case 401:
return this.handle401Error(req, next);
case 403:
return this.handle403Error(error);
default:
return _throw(error);
}
} else {
return _throw(error);
}
})
)
}
handle401Error(req: HttpRequest<any>, next: HttpHandler) {
if (!this.isRefreshingToken) {
this.tokenSubject.next(null);
this.isRefreshingToken = true;
console.log("isRefreshingToken", this.isRefreshingToken);
// Reset here so that the following requests wait until the token
// comes back from the refreshToken call.
this.authenticationService = this.authenticationService ? this.authenticationService : this.injector.get<KgAuthenticationService>(KgAuthenticationService);
this._location = this._location ? this._location : this.injector.get<Location>(Location);
return this.authenticationService.renewToken().pipe(
switchMap((newToken: string) => {
if (newToken) {
console.log("newToken Recieved:", newToken);
this.tokenSubject.next(newToken);
this.authenticationService.storeRenewedToken(newToken);
return next.handle(this.addBearerAndHeaders(req, newToken, true));
}
// If we don't get a new token, we are in trouble so logout.
//return this.logout();
}),
catchError(error => {
// If there is an exception calling 'refreshToken', bad news so logout.
//return this.logout();
}),
finalize(() => {
this.isRefreshingToken = false;
console.log("isRefreshingToken", this.isRefreshingToken);
})
);
} else {
return this.tokenSubject.pipe(
filter(token => token != null),
take(1),
switchMap(token => {
console.log("newtoken:", token.substr(token.length - 20, token.length - 1))
return next.handle(this.addBearerAndHeaders(req, token, true));
})
)
}
}
handle400Error(error) {
if (error && error.status === 400 && error.error && error.error.error === 'invalid_grant') {
// If we get a 400 and the error message is 'invalid_grant', the token is no longer valid so logout.
return this.logoutUser();
}
return _throw(error);
}
handle403Error(error) {
if (error.status === 403) { }
return _throw(error);
}
}
Хорошая статья об этом в https://www.intertech.com/Blog/angular-4-tutorial-handling-refresh-token-with-new-httpinterceptor/
В этом сценарии лучше использовать switchMap, когда у вас есть один Observable, и вам нужно получить что-то из другого запроса и вернуть другой Observable, вы можете использовать SwitchMap: https://blog.angular-university.io/rxjs-switchmap-operator/
ngOnInit() {
this._moviesDataService.getShowtimes()
.switchMap(res => {
const id = Object.keys(res[0].showtimes)[0]; // assuming you have one element in your array and you want the first id from showtimes
return this.getMovies(id); // assuming, you have a separate method that returns the movies
})
.subscribe(res => this.results = res)
}