REST API Overview
APIs (Application Programming Interfaces) enable communication and data exchange between systems. Among the various types of APIs, REST (Representational State Transfer) has emerged as a popular architectural style for building web services. This article will delve into the fundamental concepts of REST API, its principles, its benefits in modern web development, and how to design a RESTful API.
What is an API?
An API is an interface with defined functionalities that a software program presents to other programs and, in the case of web APIs, to the rest of the world via the Internet.
APIs are the building blocks that allow interoperability between businesses on the web. Companies implement APIs to expose internal business processes and data to new customers and partners. APIs are how food data containing information about allergens are shared with hundreds of restaurant apps specializing in their presentation to final customers.
We must create programs (API Servers) that serve data and other programs (API Clients) that consume/manipulate that data, as shown in the following figure.
Every company can implement Servers’ and Clients’ APIs using different programming languages, caches, proxies, and security mechanisms. They can choose monolithic or microservices architectures deployed in several application servers or cloud providers.
But how can all these heterogeneous components communicate with each other? The only answer is that they use a common language: the same semantics as the HTTP protocol. In this context, semantic means the interpretation we give every HTTP method.
We create a kind of API program called Web Service to support this interoperable machine-to-machine interaction over a network.
To implement efficiently API Servers or API Clients, you must understand the basic concepts of a REST API.
What is REST API?
REST API is an architectural style that defines a set of guidelines for creating web services. It stands for Representational State Transfer and emphasizes a stateless, client-server communication model. REST APIs use HTTP (Hypertext Transfer Protocol) as the primary protocol for data transmission and rely on standard operations such as GET, POST, PUT, and DELETE to perform actions on resources.
The Hypertext Transfer Protocol (HTTP) is a stateless application-level protocol for distributed, collaborative, hypertext information systems. Stateless in this context means a receiver must not retain the session state from previous requests.
Resources
In REST API, a resource is the information or data exposed through the web service. Resources can be entities such as users, suppliers, articles, or any other data entity that can be uniquely identified. Each resource on the Web is typically represented by a URL (Uniform Resource Locator) or endpoint defining a globally unique address. Giving something a URL turns it into a resource.
In a business context, the following endpoint retrieves a representation of an array of article data under the following URL.
1
2
GET /api/v1/articles HTTP/1.1
Host: codersite.dev
Representations
We have defined an article as an HTTP resource but cannot transmit a physical article over the Internet. But as an information resource, we can send It over the Internet. The information must be helpful for the client. That’s a representation, a machine-readable description of the current state of a resource.
1
2
3
4
5
6
7
8
HTTP/1.1 200 OK
Content-Type: application/json
{
articleID : 12121,
articleName : "Potatoes"
articlePrice : 12,3
articleDeliveryDate : 20230910
}
The server sends (because of GET requests from the client) a representation describing the state of a resource. The client sends (through a POST, PUT, or PATCH request) a representation describing the state it would like the resource to have. That’s representational state transfer.
Uniform Interface
A fundamental principle of REST API is the uniform interface, which establishes a standardized way of interacting with resources. It encompasses several constraints, including:
-
Identification of resources: Resources should be uniquely identified by URIs.
-
Stateless communication: Each request from the client to the server should contain all the necessary information to process the request without relying on the server’s previous state.
-
Manipulation of resources through representations: Resources can be accessed and modified through representations, such as JSON or XML.
-
Self-descriptive messages: Requests and responses should include sufficient metadata to describe the meaning and format of the data.
-
Hypermedia as the engine of application state (HATEOAS): The API should provide links to related resources, allowing clients to navigate the application’s state.
HTTP Methods
API clients can interact with APIs by sending HTTP methods to perform actions on resources. The four commonly used HTTP methods are:
GET
Retrieves a representation of a resource or a collection of resources.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
GET /api/v1/suppliers HTTP/1.1
Host: codersite.dev
HTTP/1.1 200 OK
Content-Type: application/json
{
"suppliers" : {
"supplier" : [
{
"supplierId": 1881,
"supplierName": "ACME"
"supplierCurrency": "EUR"
},
{
"supplierId": 132,
"supplierName": "ROYAL"
"supplierCurrency": "EUR"
},
.
.
]
},
"resultCountTotal" : 980
}
Even when a GET method is defined as a safe HTTP method, your application MUST ensure that it never changes the resource state.
The success response code to a GET request is 200 (OK).
POST
Creates a new resource.
When a client sends a POST request, it sends a representation of the resource it wants to create that corresponds with the semantics defined for the endpoint.
1
2
3
4
5
6
7
8
9
POST /api/v1/buyers HTTP/1.1
Content-Type: application/json
Host: codersite.dev
{
"buyerName" : "XYZ Restaurant",
"buyerContact" : "Mr Bond",
"buyerAddress" : "Berliner strasse 12",
}
If you include an extra attribute in the body request not defined in the endpoint representation, you probably receive an Error from the server.
The success response code to a POST request is 201 (Created).
PUT
Updates an existing resource or creates it if it doesn’t exist.
The client usually takes the representation from a GET request, modifies it, and sends it back as the body of a PUT request.
1
2
3
4
5
6
7
8
9
PUT /api/v1/buyers/{buyerId} HTTP/1.1
Content-Type: application/json
Host: codersite.dev
{
"buyerName" : "XYZ Restaurant",
"buyerContact" : "Herr Olaf",
"buyerAddress" : "Berliner strasse 12",
}
The success response code to a PUT request is 200 (OK).
DELETE
Removes a resource from the server.
1
2
3
DELETE /api/v1/buyers/{buyerId} HTTP/1.1
Content-Type: application/json
Host: codersite.dev
The success response code to a DELETE request is 204 (No Content).
Common HTTP Method Properties
Safe Methods
A Request method is safe when it does not change the state of a resource on the API Server.
The GET, HEAD, OPTIONS, and TRACE methods are defined to be safe.
Idempotent Methods
A request method is idempotent when multiple identical requests have the same effect on the API Server.
The PUT, and DELETE methods are defined to be idempotent.
Request and Response
REST API requests and responses are typically formatted in JSON (JavaScript Object Notation) or XML (eXtensible Markup Language). Requests consist of an HTTP method, headers, and, optionally, a request body containing data. Responses include an HTTP status code indicating the outcome of the request, along with the response body containing the requested resource or an error message.
Common error HTTP status codes include:
-
400 Bad Request. This means that client-request input is not well-formed. The response body will include an error providing further information.
-
401 Unauthorized. Missing or incorrect authentication credentials.
-
403 Forbidden. This means the user is authenticated, but it’s not allowed to access a resource.
-
404 Not Found. The requested resource could not be found but may be available in the future.
-
405 Method Not Allowed. Indicates that the method received in the request-line is known by the origin server but not supported by the target resource.
-
406 Not Acceptable. Returned when an invalid format is specified in the request
-
429 Too Many Requests. Indicates the user has sent too many requests in a given amount of time (“rate limiting”).
-
500 Internal Server Error. Something is broken.
Building a RESTful Web Service
Business requirement: A company wants to implement an API to allow Restaurants (represented as a buyer) to place orders at suppliers to get food articles delivered.
The following figure shows an Aggregation/Composition diagram that describes a class that references one or more objects of other classes. This allows you to model a has-a association between objects.
We use API endpoints and HTTP methods to define and implement the specific functional requirements.
We should use the nouns representing the entity with the endpoint we’re retrieving or manipulating as the pathname
1
2
GET /buyers
A list of accessible buyers is given
It is highly recommended to use the SwaggerHub editor to create a consistent API design compliant with the OpenAPI Specifications. Create a Free Account.
While you are designing, SwaggerHub can generate documentation automatically, making it easy for both API consumers and internal users to learn and test your APIs. You can create a Client SDK for different programming languages from the Editor and generate the object model and API controllers on your server side.
The following figure shows an example of how SwaggerHub generates the documentation.
We will use the Spring portfolio to build a RESTful service.
Separation of REST controllers and business logic
Following the “separation of concerns” principle, we delegated the responsibility for managing all HTTP requests and responses to a REST Controller (BuyersApiController) and managing all business logic, mappings, and the database connection to a Service class (BuyersService).
@RestController annotation tells that this Class describes endpoints that should be made available over the web to handle all HTTP requests.
We inject the BuyersService dependency class into the BuyersApiController class using its constructor method.
1
2
3
4
5
6
7
8
9
10
11
12
@RestController
@Validated
public class BuyersApiController implements BuyersApi {
private final BuyersService buyersService;
@Autowired
public BuyersApiController(
BuyersService buyersService) {
this.buyersService = buyersService;
}
}
The following listing shows common operations in a REST Controller.
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
@Override
@RequestMapping(value = "/api/v1/buyers", method = RequestMethod.POST)
public ResponseEntity<Buyer> addBuyer(Buyer body) throws Exception {
Buyer newBuyer = buyersService.addBuyer(body);
return new ResponseEntity<Buyer>(newBuyer, HttpStatus.CREATED);
}
@Override
@RequestMapping(value = "/api/v1/buyers/{buyerId}", method = RequestMethod.DELETE)
public ResponseEntity<Void> deleteBuyer(Integer buyerId) throws Exception {
buyersService.deleteBuyer(buyerId);
return new ResponseEntity<Void>(HttpStatus.NO_CONTENT);
}
@Override
@RequestMapping(value = "/api/v1/buyers/{buyerId}", method = RequestMethod.GET)
public ResponseEntity<Buyer> getBuyerById(Integer buyerId) throws Exception {
Buyer buyer = buyersService.getBuyerById(buyerId);
if (buyer == null)
throw new ResourceNotFoundException("The buyerId does not exist");
return new ResponseEntity<Buyer>(buyer, HttpStatus.OK);
}
@Override
@RequestMapping(value = "/api/v1/buyers/{buyerId}", method = RequestMethod.PUT)
public ResponseEntity<Buyer> updateBuyer(Integer buyerId, Buyer body) throws Exception {
Buyer updatedBuyer = buyersService.updateBuyer(buyerId, body);
return new ResponseEntity<Buyer>(updatedBuyer, HttpStatus.OK);
}
@Override
@RequestMapping(value = "/api/v1/buyers", method = RequestMethod.GET)
public ResponseEntity<List<Buyer>> listBuyers(
String buyerCompanyName,
String sortBy,
String sortOrder,
Integer offset,
Integer limit) throws Exception {
ListBuyersResponse response = buyersService.listBuyers(
buyerCompanyName, sortBy, sortOrder, offset, limit);
HttpHeaders responseHeaders = new HttpHeaders();
responseHeaders.set("X-TotalResultCount", String.valueOf(response.getTotalResultCount()));
return new ResponseEntity<List<Buyer>>(response.getListBuyers(), responseHeaders, HttpStatus.OK);
}
Include Loggers
Redirect the requests to the server logs. If you must fix any error your clients reported, you can always reproduce the original request and debug it in your backends.
1
2
3
4
5
6
7
8
9
10
@Override
@RequestMapping(value = "/api/v1/buyers/{buyerId}", method = RequestMethod.DELETE)
public ResponseEntity<Void> deleteBuyer(Integer buyerId) throws Exception {
logger.info(request.getMethod() + UtilitiesService.REQUESTED_PARAMETERS + utilitiesService.getRequestURLWithQueryParam(request));
buyersService.deleteBuyer(buyerId);
return new ResponseEntity<Void>(HttpStatus.NO_CONTENT);
}
We include a new dependency in your Api Controller:
1
2
3
4
5
6
7
8
9
10
11
12
public class UtilitiesServiceImpl implements UtilitiesService {
@Override
public String getRequestURLWithQueryParam(HttpServletRequest request) {
StringBuffer requestURL = request.getRequestURL();
if (request.getQueryString() != null) {
requestURL.append('?').append(request.getQueryString());
}
return requestURL.toString();
}
}
Even you can create statistics of how many requests by endpoint arrive per minute by implementing a hot-warm architecture in Elasticsearch.
1
[12/4/23 15:48:27:888 CET] 00000121 SystemOut INFO 19484 BuyersApiController : DELETE_REQUESTED_PARAMETERS: https://yourapidomain/api/v1/buyers/12345
When the API Strategy is not aligned with the IT infrastructure
In most business scenarios, companies decide to externalize part of their current functionalities by developing APIs to obtain new potential clients.
For example, one requirement is to retrieve an assortment by a unique ID. You, as an API designer, create the following endpoint:
It seems logical to retrieve a unique resource by using a unique ID parameter. But your IT department says you need additional parameters to retrieve an assortment entity because that’s how your business model works.
Then you add the following required parameters (queries):
If we know and follow the relationships between our business entities, we can redesign the endpoint to a final version:
It is up to you to give a better developer experience when your API documentation is understandable and easy to navigate.
Designing an Endpoint from aggregation/composition relationships
Once you build a class design of your business model, think about in entities and relationships to create functional endpoints.
For example, from the following figure we can interpret that the Buyer entity includes one or more instances of the BuyingList entity.
Then, we can connect them by using the following relationship:
1
GET /api/v2/buyers/{buyerId}/buyingLists
Additionally, the BuyingList entity includes one or more instances of the Article entity.
To navigate until the Article entity we can build the following endpoint:
1
GET /api/v2/buyers/{buyerId}/buyingLists/{buyingListId}/articles
Benefits of REST API
REST APIs offer several advantages that contribute to their widespread adoption in modern web development:
-
Scalability: REST API’s stateless nature allows for horizontal scalability, where multiple servers can handle requests independently, improving performance and accommodating a growing user base.
-
Interoperability: REST APIs leverage standardized HTTP methods and formats, enabling communication between different systems regardless of their underlying technologies.
-
Simplicity and ease of use: REST APIs have a straightforward design, making them easy to understand, implement, and consume. Developers can quickly grasp the concepts and start building applications around the exposed resources.
-
Flexibility: REST APIs support various data formats and can be used with different client applications, including web browsers, mobile devices, and third-party services.
Conclusion
REST API concepts provide a foundation for building scalable, interoperable, and easily consumable web services. Understanding the principles of REST, such as resources, uniform interface, HTTP methods, and request/response formats, is crucial for designing and consuming RESTful APIs effectively.