How to add multiple, independent HTTP interceptors to an Angular 4 application?
I tried to add them by extending the providers
array with more than one interceptors. But only the last one is actually executed, Interceptor1
is ignored.
@NgModule({ declarations: [ /* ... */ ], imports: [ /* ... */ HttpModule ], providers: [ { provide: Http, useFactory: (xhrBackend: XHRBackend, requestOptions: RequestOptions) => new Interceptor1(xhrBackend, requestOptions), deps: [XHRBackend, RequestOptions], }, { provide: Http, useFactory: (xhrBackend: XHRBackend, requestOptions: RequestOptions) => new Interceptor2(xhrBackend, requestOptions), deps: [XHRBackend, RequestOptions] }, ], bootstrap: [AppComponent] }) export class AppModule {}
I could obviously combine them into a single Interceptor
class and that should work. However, I would like to avoid that as these interceptors have completely different purposes (one for error handling, one to show a loading indicator).
So how can I add multiple interceptors?
So how can I add multiple interceptors? Http doesn't allow to have more than one custom implementation. But as @estus mentioned the Angular team has added a new HttpClient service recently (release 4.3) which supports multiple interceptors concept. You don't need to extend the HttpClient as you do with the old Http .
TL;DR Yes, providing multiple interceptors is fine and they run in order.
The first step is to build the interceptor. To do this, create an injectable class that implements HttpInterceptor. Any interceptor we want to create needs to implement the HttpInterceptor interface. This means that our new class should have a method called intercept with parameters HttpRequest and HttpHandler.
Http
doesn't allow to have more than one custom implementation. But as @estus mentioned the Angular team has added a new HttpClient service recently (release 4.3) which supports multiple interceptors concept. You don't need to extend the HttpClient
as you do with the old Http
. You can provide an implementation for HTTP_INTERCEPTORS
instead which can be an array with the 'multi: true'
option:
import {HTTP_INTERCEPTORS, HttpClientModule} from '@angular/common/http'; ... @NgModule({ ... imports: [ ... , HttpClientModule ], providers: [ ... , { provide: HTTP_INTERCEPTORS, useClass: InterceptorOne, multi: true, }, { provide: HTTP_INTERCEPTORS, useClass: InterceptorTwo, multi: true, } ], ... })
Interceptors:
import {HttpEvent, HttpHandler, HttpInterceptor, HttpRequest} from '@angular/common/http'; ... @Injectable() export class InterceptorOne implements HttpInterceptor { intercept(req: HttpRequest<any>, next: HttpHandler): Observable<HttpEvent<any>> { console.log('InterceptorOne is working'); return next.handle(req); } } @Injectable() export class InterceptorTwo implements HttpInterceptor { intercept(req: HttpRequest<any>, next: HttpHandler): Observable<HttpEvent<any>> { console.log('InterceptorTwo is working'); return next.handle(req); } }
This server call will print both interceptors' log messages:
import {HttpClient} from '@angular/common/http'; ... @Component({ ... }) export class SomeComponent implements OnInit { constructor(private http: HttpClient) {} ngOnInit(): void { this.http.get('http://some_url').subscribe(); } }
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With