/api-gateway-sdet

API Gateway from SDET Perspective

API Gateway from SDET Perspective

10 API GW features a Test Automation Engineer should know

NOTE: this file has been published as a post on Medium.

IT employers require knowledge of API Gateway (GW), its components, monolithic or microservices architecture. An SDET should know about the role of the API GW, in terms of the cloud, local architecture setup, etc. Take an e-commerce app for example.

Feature # 1

  • Authentication
  • Authorization
  • SSL Certification

For the Cart, just to add something, I need some Authentication. Apply the Authentication concept logic to the cart API.

When I want to add something to my product, or delete a product, I need an Admin Role for my app. A regular customer cannot delete add/delete a product. Only an Admin, with an admin role and access, can add/delete a product, someone like a Product Category Manager. For this purpose, I add Authorization.

Authentication:

  • for a regular customer
  • add to cart or get some cart value
  • can do with simple authentication
  • user has to be logged in

Authorization:

  • require Admin Access

For /home and /products, I don't need any Access. Without username/password, I can hit the app and access it.

Authentication and Authorization are not related to the app features or functionality, they are a separate thing all together. I move the Authentication/Authorization logic inside my API GW. It means that when a Request comes from the client/user side, firstly the user gets authorized before, eg, accessing a cart, and only then the cart API can be accessed.

The Client sends a Request to the API GW, where Authentication is defined. Authentication notifies the Client that they can now access the product API. A client would state their Authentication role and proceed with accessing the product API, adding or deleting a product, etc.

Authentication Role is defined at the API GW level, not at the business/app level. I've disctinstly segregated my Business Logic.

Whatever request I send, I want it to be sent via a Secure HTTP -> HTTPS. For secure communication, I need SSL Certification (secure connection certificate) at the API GW level. Can renew SSL Certification every 30-60 days at the API GW level. No point putting the Certification at the app level.

Requests from a Client come through the SSL Certification at the API GW level. When the certificate is valid, the client can access the app. The advantage of SSL Certification is that the API client can only access a specific entry point with the help of whichever URL they want to access. Business Logic is segregated. For any bad actor (hacker), I actually make the app safe by defining the SSL Certification at the API GW level.

Feature # 2

  • Routing

Let's say I deal with a Monolithic architecture. All the APIs, business logic, etc. are available under one machine/system/component. I convert my Monolithic architecture into Microservices, tiny services available for each feature. 4 segregated Microservices are available.

A client makes a request (via HTTPS) directly to the API GW only. When the client wants to retrive the product info, I can define that in Routing. A client asks to give them (requests) the Product API, so that they can access the Product. So the Routing defines that if the client requests Product, the request is only sent to the Product APIs. Then the response is sent back to the client.

The same thing happens, when a client requests the Cart APIs. When the client requests for the Cart, Routing decides that the request gets sent to the Cart API. Routing gets the response, forwards it back with the cart info. And it spins up in a form of a web [browser] app.

Feature # 3

  • Adaptor

Suppose the app is getting popular and profitable. A big crowd uses it. I offer a couple of extra features during a holiday season: trends, deals, discounts, etc. When Client sends many requests, it can cause a delay. The more requests, the more delay.

I can improve this case by adding another component called Adaptor at the API GW level. My developers always say thay they add the Adaptor at the API GW level. Adaptor decides how many requests to handle.

Improved Diagram:

Adaptor takes the Request from the Client/Device. Client sends Request to the Adaptor via one single network call. It's faster than before, when Client sent 4 different requests. Now Client sends 1 single request over the network to the API GW. The Gateway makes a number of calls, consolidates the response, then gives it back to the Client side in the form of a Response.

Feature # 4

Let the Adaptor decide how many calls to make and to which Microservices. Not from the Client side. Client send an n-number of requests (10, 20 ... 100), the API GW decides how to send them. Client just clicks on Homepage, Trends, Deals, etc.

I can improve my design/diagram. I have the Static Html Homepage [Microservice], which is not a part of the Business Logic. It's a simple homepage, static content. I can decommission the Static Content Html from the Microservices Web App site and define/place it at the aPI GW level. No need to call the actial app server.

Feature #5

  • Caching

Client sends Request via API GW and gets the response back form the 'Trending Products' Miscroservice - the 50 products that are available between 10-11 am. But what if the Client send another Request at 10:10, then another at 10:15 am and 10:20 am, and so forth (multiple times until 11 am). Every time the Client sends the Request, they hit this particular Microservice and a Response back. It's unnecessary, because every time the product count remains the same. The next list of available trending product becomes available after one hour. Once this hour expires at 11 am, I get the next list with the next 50 products available.

What to do? Set Caching at the API GW level. The 1st time the Request is sent to this particular Microservice, the Response is sent back, and the 50 products are stored in the Cache, eg, at 10 am. After 10 minutes, at 10:10 am, I send another request to the API GW. The Gateway has the intelligence to understand that the Client requests a list of the current Trending Products. It does not call the Microservice any longer. There's no point in calling the same Microservice again and again (with the same request), while the product output remains the same. It's better to retrieve the Response from the Cache only.

This is the biggest advantage of Caching. The same Response is given back the Client up until 11 am. I can define/configure a threshold timeout of 1 hour at the API GW level. After 1 hour, the Cache with 50 products gets deleted/detroyed. At 11 am the Request is sent again to the API GW. Then again this particular Microservice gets called, retrieving the next 50 Trending Products. And again the new 50 products get stored in the Cache for the next 1 hour until 12 pm.

The Cache is always handled at the API GW level, not at the Business Logic/WebApp/Microservice level.

Feature # 6

  • Videos

Example: In my web app I provide different videos. There are some video APIs, meaning fast streaming APIs and slow streaming APIs.

Client # 1 sends a Request from a Mobile device. What to do? Define Routing at the device level. When Request comes via Mobile, I need a low-quality video. The Low Quality video Microservice responds back and provides the Client with a low-quality video on their phone (mobile device).

Client # 2 wants an API and sends a Request from a Desktop. The Router determines that the request comes from the desktop device, and the Desktop route is taken. The Router defines that desktop request should go to (call) the HD Quality content [MS], where high-quality videos are available. IT responds back and routes over to the Desktop Client. Then the Desktop [browser] Client is able to view high-quality videos.

Router defines which route to take - Mobile or Desktop? I configure this at the API GW level.

Feature # 7

  • Load Balancer

Client sends a Request to API GW, to the Load Balancer. My Microservice (MS) is all about the /product service. I want to access the Product feature, send a Request. This feature is very popular on my app most of the time. The Request is sent to the Load Balancer. I have multiple copies of this specific MS. The majority of my crowd comes from this particular service.

What to do? Define a Load Balancer. It sends the Request to each copy of the MS and gets the Response back. One MS does not handle (take care of) the entire load. The load is divided among multiple copies od the MSs through the Load Balancer. I can achieve this concept in a Round Robin fashion. Round robin request is sent to the MS. Per the request, a specific MS is called, and the /product can be called again and again - 1st copy, then 2nd, 3rd and so forth.

Let's say, I made some changes to this particular MS. I create a (new) copy of this /product MS at deployment as the New Deploy. But I don't want to grant 100% access to the new deployment. With the Load Balancer, I can route the traffic. 5% is dedicated to the New Deploy, the new service with the new feature I've just added. With this approach, I can achieve Canary Release. 5-10% goes to the Canary Release, the rest 90-95% goes to the previous one.

The MAANG companies do not release 100% of their traffic to the new feature or service they've created. Through the Load Balancer, 95% of the traffic goes to the 'old' one. 5% goes to the new MS. Once I verify there are no bugs in the new feature, I start increasing traffic through the Load Balancer on the API GW level.

*Can also conduct A/B Testing.

Round Robin DNS

Canary Release

Parallel Change Pattern

Postel's Law a.k.a. Robustness Principle

Feature # 8

  • Protocol Adaptor

Another small feature is the Protocol Adaptor for HTTP conversion. Define Protocol Adaptor at the API GW level. MSs get Requests in the form of HTTP1 only. In my app, all the backend services can be accessed via HTTP1 protocol. When the Request is sent from Client 1 in the form of HTTP2, it arrives at the Protocol Adaptor at the API GW level, which converts all the old protocols into the new ones.

Feature # 9

  • Monitoring

Also, API GW provides Monitoring for stats:

  • dashboard/logs/charts
  • number of requests, types of requests I get from different clients, distinction between mobile and desktop app requests

Feature # 10

  • Billing

Let's say, my app got slashdotted. Some external apps now want to use my APIs. Eg, a /payment MS, which is stable and popular on the market. Other companies/web apps want to use this particular /payment API/MS.

I go to a cloud of my choice. On their marketplace I define Billing at the API GW level. Eg, there's a small company, which doesn't own any GW APIs. It simply sends Requests to Billing, and the API GW forwards the calls to the /payment GW APIs and back. The Billing adaptor set at the API GW level charges the Client.

Monetization Example: My /payment API got famous. My Clients' payments happen through my APIs. And whenever a client sends a Request, they have to pay $5.

I host the Billing service at the API GW level, in the form of the Marketplace. Different cloud vendors are available. Can determine how many calls are made by the Client, and on the basis of that generate an Invoice in Billing. I can expose my APIs and generate revenue.