Advertisement
  1. Code
  2. JavaScript
  3. Angular

Beginner's Guide to Angular: HTTP

Scroll to top
This post is part of a series called Beginner's Guide to Angular 4.
Beginner's Guide to Angular: Routing

Angular HTTP is all about using the right techniques for building a single-page application that creates HTTP requests and handles HTTP responses in a maintainable and scalable manner. The @angular/common/http package comes with a client, interceptor, and many more features for your Angular application.

Single-page applications often need to scale, and the framework needs to provide the right mix of tools and techniques to support this. Angular comes with many built-in tools to scale gracefully. Today, you'll learn how to set up a basic architecture for HTTP requests in your app, and how to use interceptors to keep your app scalable. 

Big or small, any Angular app will have a flow of HTTP requests. You'll be sending and receiving HTTP requests. How would you apply rules and change the way your request workflow happens? Interceptors are Angular services used to achieve this. As the name suggests, you can intercept any request sent or response received. You will be able to add or modify the values of the request. 

But before you venture into HTTP interceptors, you need to be aware of the basics of HTTP in Angular. 

Skeleton of the Project

Before we get started, I would like to walk you through the overall project structure. You can find the completed project live in this GitHub repository.

1
-src
2
--app
3
----child
4
------child.component.ts
5
----parent
6
------parent.component.ts
7
------parent.service.ts
8
----utility
9
------header.interceptor.ts
10
------response.interceptor.ts
11
------error.interceptor.ts
12
----app-routing.module.ts
13
----app.component.css
14
----app.component.ts
15
----app.module.ts
16
--assets

1. Preparing Your Project for HTTP

HttpClient is a built-in service class in the Angular package: @angular/common/http. When you want to communicate over an HTTP protocol in Angular, you may use fetch or XMLHttpRequest. Either way, the communication happens through HttpClient. This service class comes with many interesting signatures and return types. 

Interesting features of the HttpClient are:

  • perform GET, POST, PUT, and DELETE requests
  • streamline all your error-handling strategies
  • intercept any HTTP request sent or response received 
  • create typed request or response objects
  • introduce RxJS observables

Step 1: Importing HttpClient in app.module.ts

In order to use HttpClient, you must import the HttpClientModule in your app.module.ts file. And this should be a part of the imports array in your NgModule

1
import { NgModule } from '@angular/core';
2
import { BrowserModule } from '@angular/platform-browser';
3
import { HttpClientModule } from '@angular/common/http';
4
5
@NgModule({
6
  imports: [BrowserModule, HttpClientModule],
7
  .
8
  .
9
  providers: []
10
})
11
export class AppModule { }

With the above declaration, you are good to use HttpClient in your components. To access the HttpClient in a component, you need to create a service, and it has to be an Injectable service. For our purpose, we'll create a service which can be injected at root level. 

Step 2: Create an Injectable Service

An Injectable is a decorator which can be used in Angular. It comes with metadata like provideIn. If provideIn is set to root, you will be able to use the service throughout your application. The component should be defined after the service. If the component is defined before a service, a null reference error will be thrown at run-time.

Any injectable service will have the following skeleton. In our sample project, you will find an injectable service in the file parent.service.ts. It provides all the APIs required by parent.component.ts.

1
import { HttpClient } from  '@angular/common/http';
2
import { Injectable } from  '@angular/core';
3
4
@Injectable({
5
    providedIn:  'root'
6
})
7
8
export class HttpService {
9
    constructor(private https: HttpClient) { }
10
}

Now, let's improve the above skeleton and make it fetch data from an API. If you are new to HTTP, we have a great post to help you learn the basics

1
import { HttpClient, HttpHeaders } from  '@angular/common/http';
2
import { Injectable } from  '@angular/core';
3
import {HttpParams} from "@angular/common/http";
4
5
@Injectable({
6
    providedIn:  'root'
7
})
8
9
export class ParentHttpService {
10
11
    private url = 'https://reqres.in/api/users';
12
13
    constructor(private http: HttpClient) { }
14
15
    getUsers() {
16
        return this.http.get(this.url);
17
    }
18
19
    // ...

20
}

Step 3: Consuming the Injectable Services

In our parent.service.ts file, you will find many getter API calls. These APIs are called from parent.component.ts. Let's take a quick look at parent.component.ts, which is used for calling the methods in parent.service.ts.

What we want to achieve here is an injection of the service created for fetching users.

  1. We subscribe to the method in the service.
  2. The moment we run this method, we will be executing the GET request and receiving the response/error object. 

In our example, we have three different buttons, each pointing to a different method in the injectable service. 

1
import { Component } from '@angular/core';
2
import { ParentHttpService } from './parent.service';
3
4
@Component({
5
  selector: 'parent',
6
  template: `

7
    <div>

8
      <h3>Parent Page</h3>

9
      <div>Get All Users <button (click)="getAllUsers()">Get All Users</button></div>

10
      <div>Get users in page 2 <button (click)="getUserInPageTwo()">Get Items on Page 2</button></div>

11
      <div>Get users in page 2 with custom Header <button (click)="getUsersOnAPageWithHeaders()">Get users in page 2 with custom Header</button></div>

12
      <div>Users:</div>

13
      <div>{{users}}</div>

14
    </div>

15
  `,
16
})
17
export class ParentComponent {
18
19
  users : any;
20
21
  constructor(private httpService: ParentHttpService) { }
22
23
  ngOnInit() {}
24
25
  getAllUsers = () => {
26
      this.httpService.getUsers().subscribe(
27
        (response) => { this.users = JSON.stringify(response); },
28
        (error) => { console.log(error); });
29
      }
30
  getUserInPageTwo = () => {
31
    this.httpService.getUsersOnAPage(2).subscribe(
32
      (response) => { this.users = JSON.stringify(response); },
33
      (error) => { console.log(error); });
34
    }
35
  getUsersOnAPageWithHeaders = () => {
36
    this.httpService.getUsersOnAPageWithHeaders(2).subscribe(
37
      (response) => { this.users = JSON.stringify(response); },
38
      (error) => { console.log(error); });
39
    }
40
    
41
}

The above component appears as below. In the screenshot, I have clicked on the Get All Users button, and the results have appeared successfully. 

results pageresults pageresults page

Step 4: Configuring an HttpInterceptor

As mentioned before, you can add or modify the values of any request. In an application, you may have multiple interceptors. This is why it's important for you to register the interceptor as a provider in app.module.ts. If the interceptor is not registered here, it will not be able to intercept the requests we make using the HttpClient service. 

One of the interceptors in our project can be found in utility/header.interceptor.ts. And we have to import it into app.module.ts

1
import { NgModule } from '@angular/core';
2
import { BrowserModule } from '@angular/platform-browser';
3
import { HttpClientModule, HTTP_INTERCEPTORS } from '@angular/common/http';
4
5
import { AppRoutingModule } from './app-routing.module';
6
import { AppComponent } from './app.component';
7
import { ParentHttpService } from './parent/parent.service';
8
import { HeaderInterceptor } from './utility/header.interceptor';
9
10
@NgModule({
11
  declarations: [
12
    AppComponent
13
  ],
14
  imports: [
15
    BrowserModule,
16
    AppRoutingModule,
17
    HttpClientModule
18
  ],
19
  providers: [ { provide: HTTP_INTERCEPTORS, useClass: HeaderInterceptor, multi: true }, ParentHttpService],
20
  bootstrap: [AppComponent]
21
})
22
export class AppModule { }

The syntax for adding interceptors in your @NgModule configuration is:

1
providers: [{ provide: HTTP_INTERCEPTORS, useClass: <name of interceptor>, multi: true }]

Step 5: Building the HttpInterceptor

In order to create an interceptor, the HttpInterceptor interface available in @angular/common/http has to be implemented. Every time your Angular application wants to make a request or receive a response through the HTTP protocol using the HttpClient service, the interceptor's intercept method will be called. 

The intercept method has the following anatomy:

  • input: accepts a reference to the httpRequest object
  • purpose: inspects and modifies the httpRequest object as required
  • output: calls next.handle with the modified httpRequest

Step 6: Intercepting the Header

As we learn about intercepting, let's add a header to all our API requests. Here, we are adding a new header called ANGULAR_TUTS_INTERCEPTOR to our get request. As mentioned, we can have multiple interceptors in the code. Don't forget to name the interceptor correctly, so that you can identify and maintain code better.

1
//app.interceptor.ts
2
import { Injectable } from '@angular/core';
3
import { HttpInterceptor, HttpEvent, HttpResponse, HttpRequest, HttpHandler, HttpErrorResponse } from '@angular/common/http';
4
import { Observable } from 'rxjs';
5
6
@Injectable()
7
export class HeaderInterceptor implements HttpInterceptor {
8
  intercept(httpRequest: HttpRequest<any>, next: HttpHandler): Observable<HttpEvent<any>> { 
9
    console.log("Inside Intercept");     
10
    const ANGULAR_TUTS_INTERCEPTOR = '123456';
11
    return next.handle(httpRequest.clone({ setHeaders: { ANGULAR_TUTS_INTERCEPTOR } }));
12
  }
13
}

Here is a screenshot of the header being included in our GET request. 

the header being included in our GET request. the header being included in our GET request. the header being included in our GET request. 

Step 7: Intercepting the Response Body

Apart from intercepting the header of a request, we are allowed to modify the response body as well. The code for intercepting the response body can be found in response.interceptor.ts. Here, we consume the HttpResponse and append a new parameter called projectCode into the body. When the response is printed in our component, you will see projectCode in the output. 

With the new ResponseInterceptor, our app.module.ts appears as below:

1
providers: [ 
2
    { provide: HTTP_INTERCEPTORS, useClass: ResponseInterceptor, multi: true }, 
3
    { provide: HTTP_INTERCEPTORS, useClass: HeaderInterceptor, multi: true }
4
    ParentHttpService
5
]
1
//response.interceptor.ts

2
import { Injectable } from '@angular/core';
3
import { HttpInterceptor, HttpEvent, HttpResponse, HttpRequest, HttpHandler } from '@angular/common/http';
4
import { Observable } from 'rxjs';
5
import { map } from 'rxjs/operators'
6
7
@Injectable()
8
export class ResponseInterceptor implements HttpInterceptor {
9
  intercept(req: HttpRequest<any>, next: HttpHandler): Observable<HttpEvent<any>> {
10
    return next.handle(req).pipe(
11
      map((event: HttpEvent<any>) => {
12
            if (event instanceof HttpResponse) {
13
                event = event.clone({ body: this.handleResponse(event) })
14
            }
15
            return event;
16
        })
17
    )
18
  }
19
  private handleResponse(event: any): any {
20
      //  override http response body here

21
      const body = { ...event.body, "projectCode": "angular_tuts_http" };
22
      return body;
23
  }  
24
}

In this screenshot, you will be able to see the response interceptor in action. We have highlighted projectCode to showcase the newly added parameter to the response.

response interceptor in action.response interceptor in action.response interceptor in action.

Step 8: Error Handling

Interceptors help us handle errors better. An API doesn't always return the expected result. Sometimes, when the server is down, or if the request does not contain the right body, you are bound to get an error. This is why error handling is extremely important. In our error.interceptor.ts file, we have added some simple error-handling logic. The API request will be repeated four times, until the final error is thrown to the component.

One of the most important aspects to be seen in this piece of code would be the use of RxJS. Later in this tutorial, we will see why RxJS is important. 

We need to improve app.module.ts with the new ErrorInterceptor. The file appears as below.

1
providers: [ 
2
    { provide: HTTP_INTERCEPTORS, useClass: ResponseInterceptor, multi: true }, 
3
    { provide: HTTP_INTERCEPTORS, useClass: HeaderInterceptor, multi: true }, 
4
    { provide: HTTP_INTERCEPTORS, useClass: ErrorInterceptor, multi: true }, 
5
    ParentHttpService
6
]
1
import { HttpInterceptor, HttpRequest, HttpHandler, HttpEvent } from '@angular/common/http';
2
import { Injectable } from '@angular/core';
3
import { of, throwError } from 'rxjs';
4
import { concatMap, delay, retryWhen } from 'rxjs/operators';
5
6
@Injectable()
7
export class ErrorInterceptor implements HttpInterceptor {
8
    retryCount = 3;
9
    retryWaitTime = 1000;
10
    intercept(req: HttpRequest <any>, next: HttpHandler) {
11
        return next.handle(req).pipe(
12
            retryWhen(error =>
13
                error.pipe(
14
                    concatMap((error, count) => {
15
                        if (count <= this.retryCount && error.status == 404) {
16
                            return of(error);
17
                        }
18
                        return throwError(error);
19
                    }),
20
                    delay(this.retryWaitTime)
21
                )
22
            )
23
        );
24
    }
25
}

Here is a screenshot that shows the behavior of error.interceptor.ts. As you can see, the same API request is made multiple times in the network tab. 

behavior of error.interceptor.ts.behavior of error.interceptor.ts.behavior of error.interceptor.ts.

Tips and Tricks for Using @angular/common/http

HttpParams

HTTP Get can have multiple query string parameters. Here is a simple example:

1
https://reqres.in/api/users?page=2

In the above snippet, there is a query string parameter: page = 2. With the @angular/common/http package, you can add query string parameters quite easily. To achieve this, you have to import HttpParams from the package. HttpParams is immutable. All API methods linked to HttpParams will not result in any kind of object mutation. That is why we have to chain the set method call—if you try any other way of setting the HttpParams, it would not work. Instead, you will receive an empty HttpParams object. 

1
//parent.service.ts

2
import {HttpParams} from "@angular/common/http";
3
4
//CHAINING

5
const params = new HttpParams()
6
        .set('page', pageNo)
7
8
return this.http.get(this.url, {params})
9
    
10
//DOES NOT WORK

11
params.set('page', 'pageNo"')

What would you do if the URL consists of existing query string parameters? In this case, you can make use of fromString. fromString is a part of the HttpParams, and here is how you can use it:

1
const params = new HttpParams({
2
  fromString: 'page=2'
3
});

HttpHeaders

Next, let's understand how to read and set HTTP headers. Once again, if you are new to HTTP, you are highly encouraged to read our post about HTTP

HTTP headers make a big difference in any HTTP request or response. Some headers are automatically added to the request, and some can be custom-added into the request. To achieve this, you need to make use of the HttpHeaders class from @angular/common/http. Just like HttpParams, HttpHeaders is also immutable. 

1
const headers = new HttpHeaders()
2
.set("Content-Type", "application/json");
3
const params = new HttpParams()
4
.set('page', pageNo)
5
6
return this.http.get(this.url, {headers, params})

RxJS

The role of RxJS is to ensure that only one HTTP request is made per subscription. Many a time, duplicate requests could be raised in your code. This will affect performance, and can even terminate your application. When you want data to be queried just once from the back-end, you need to make use of RxJS.

Also, when HTTP requests need to be made in parallel, and the results have to be combined, we can use forkJoin from the RxJS library. 

If you want your HTTP requests to be executed in a sequence, and the result of the first request has to be used to construct the second request, you can use switchMap. These are just two of the common use cases for RxJS. 

Another important aspect of RxJS would be its operators. This library offers operators like map and filter, which can be used along with next.handle.

Wrapping Up

We have come to the end of this tutorial on Angular HTTP. Making requests and waiting for responses is an inevitable step in any single-page application. It is important to build a scalable application that is easy to maintain. Leverage the @angular/common/http library along with RxJS to build your client's HTTP workflow. 

Advertisement
Did you find this post useful?
Want a weekly email summary?
Subscribe below and we’ll send you a weekly email summary of all new Code tutorials. Never miss out on learning about the next big thing.
Advertisement
Looking for something to help kick start your next project?
Envato Market has a range of items for sale to help get you started.