Building RESTFul API With Spring Data
What is a RESTful API?
RESTful APIs are a cornerstone of modern web development, allowing systems to communicate over HTTP using principles of Representational State Transfer (REST). RESTful APIs are designed to be scalable, stateless, and follow the HTTP protocol standards, making them highly effective for a wide range of applications.
In this blog, we'll explore the levels of RESTful APIs as defined by the Richardson Maturity Model, a framework that describes the stages of RESTful API maturity.
The Richardson Maturity Model
The Richardson Maturity Model breaks down RESTful APIs into four distinct levels, each building upon the previous one. These levels highlight the increasing adherence to REST principles.
Level 0: The Swamp of POX
At this level, APIs act as simple endpoints that accept requests and return responses. These APIs are typically not RESTful—they often use HTTP as a transport protocol rather than leveraging its full capabilities.
Characteristics:
- No standard structure for requests or responses.
- Often uses XML or JSON but lacks meaningful use of HTTP methods.
- Example:
POST /api/resource HTTP/1.1
Host: example.com
Content-Type: text/xml
<data>
<value>Example</value>
</data>
This is more like an RPC (Remote Procedure Call) using HTTP as a vehicle.
Level 1: Resources
At Level 1, APIs start organizing data into distinct resources. Each resource is identified by a unique URI (Uniform Resource Identifier).
Characteristics:
- Resource-based design.
- Unique URIs for different entities.
- Still lacks proper use of HTTP methods.
Example:
GET /api/users/12345 HTTP/1.1
Host: example.com
In this example, the URI /api/users/12345
identifies a specific user resource.
Level 2: HTTP Verbs
At Level 2, APIs begin using HTTP methods like GET
, POST
, PUT
, and DELETE
to indicate the desired operation on resources.
Characteristics:
- Full utilization of HTTP methods.
- Improved clarity of intent.
- Status codes convey the results of operations (e.g.,
200 OK
,404 Not Found
).
Example:
POST /api/users HTTP/1.1
Host: example.com
Content-Type: application/json
{
"name": "Jane Doe",
"email": "jane.doe@example.com"
}
This example demonstrates creating a new user resource using a POST
request.
Level 3: Hypermedia Controls (HATEOAS)
At the highest level, APIs provide Hypermedia as the Engine of Application State (HATEOAS). This means responses include links to related actions or resources, allowing clients to navigate the API dynamically.
Characteristics:
- Responses include hypermedia links.
- Self-documenting API through navigable relationships.
- Closest adherence to REST principles.
Example:
{
"id": 12345,
"name": "Jane Doe",
"links": [
{ "rel": "self", "href": "/api/users/12345" },
{ "rel": "update", "href": "/api/users/12345" },
{ "rel": "delete", "href": "/api/users/12345" }
]
}
This response includes links that guide clients to other actions, such as updating or deleting the user resource.
Build a product query api using Spring Data
Let's condiser the following business requirement we want to address:
Currently, local grocery stores must either visit large distribution companies or contact them individually to inquire about pricing, stock availability, and delivery schedules when placing orders. This process is time-consuming and inefficient for grocery managers. To address this challenge, we propose developing a platform that enables managers to easily check prices, view stock levels, place orders, and track deliveries in one streamlined solution.
We aim to develop a RESTful API to provide access to product data, enabling consumers like microservices, mobile apps, and web applications to retrieve and display the information. This will allow end users, such as grocery store managers, to view, read, and stay informed about the available products on our platform.
What is Spring Data
Spring Data is a part of the broader Spring Framework ecosystem that simplifies data access in Java applications. It provides a set of powerful tools and abstractions that make it easier to interact with various data sources, such as relational databases, NoSQL databases, and more. Spring Data eliminates a lot of the boilerplate code typically involved in working with databases, enabling developers to focus on the business logic rather than the intricacies of data access.
In our case we will use PostgresSQL database, so let's first create a docker container for postgres and create also our database "grocery-marketplace":
docker run -d \
--name postgres-db \
-e POSTGRES_USER=postgres \
-e POSTGRES_PASSWORD=postgres \
-e POSTGRES_DB=grocery-marketplace \
-p 5432:5432 \
postgres:latest
The Grocery Marketplace Product Query API Project
Our project is a maven with the following main dependencies:
Here is the informative list of the dependencies from the provided XML:
- spring-boot-starter-data-jpa (Spring Boot, JPA support)
- spring-boot-starter-data-rest (Spring Boot, Data REST support)
- flyway-core (Flyway, Database migrations)
- flyway-database-postgresql (Flyway, PostgreSQL database support)
- kotlin-reflect (Kotlin, Reflection support)
- kotlin-stdlib (Kotlin, Standard library)
- postgresql (PostgreSQL JDBC driver, runtime scope)
An operational application that covers our use case is shared through Github at the end of this post.
To start your own Spring Boot application, use Spring Initializr, an online tool that helps you quickly bootstrap your Spring project. You can specify the programming language, build tool, and dependencies.
Database Schema
There are several strategies you can use to create and manage tables in a PostgreSQL database, with Flyway being one of the most popular tools for handling database migrations in a structured way. Let's see how to intstruments Flyway to create our tables and inserts sample data.
Flyway is a versioned migration tool that allows you to manage your database schema in a controlled and repeatable way. It works by applying SQL-based migration scripts to your database.
- Create SQL Migration Scripts
Flyway applies migrations sequentially based on versioned SQL scripts. These scripts should follow a naming convention like V1__Initial_schema.sql, V1__create_product_table.sql, etc.
Below a snippet of our V1__create_product_table.sql
-- Create tables for all related entities
CREATE TABLE category (
id INT PRIMARY KEY,
name VARCHAR(100) NOT NULL
);
-- Sample Data Insertion for related tables
-- Insert categories
INSERT INTO category (id, name) VALUES (1, 'Electronics');
INSERT INTO category (id, name) VALUES (2, 'Furniture');
INSERT INTO category (id, name) VALUES (3, 'Clothing');
- Flyway Configuration
You can configure Flyway in your application.yml (for Spring Boot) to point to your PostgreSQL database.
flyway:
enabled: true
locations: classpath:db/migration
baseline-on-migrate: true
clean-disabled: falsek
Running Migrations: When you start your Spring Boot application, Flyway will automatically detect and run any pending migrations in the db/migration directory, creating or modifying tables in the PostgreSQL database.
JPA Entities
In JPA (Java Persistence API), an entity is a lightweight, persistent domain object that is mapped to a table in a relational database. It represents a record in the database and can be used to interact with that record through the application.
Below we define our first JPA Entity Category, which defines the category of product such as Electronics, Clothing, etc ...
@Entity
data class Category(
@Id
@GeneratedValue(strategy = GenerationType.SEQUENCE)
val id: Int? = null,
val name: String
)
Note that we didn't use the @Table
annotation since the entity and table names are case-sensitive and match. If your table name differs, use @Table(name = "tablename")
.
Repositories
a Spring Data Repository is an interface that allows you to interact with a data store (e.g., a relational database like MySQL, or a NoSQL database like MongoDB) without needing to write the implementation of the basic CRUD (Create, Read, Update, Delete) operations yourself. In our case we will use the Spring Data JPA Repository which is a specific type of repository provided by the Spring Data project that simplifies the implementation of the Data Access Layer (DAL) for JPA-based (Java Persistence API) applications. Below a snippet of our CategoryRepository:
@Repository
interface CategoryRepository : JpaRepository<Category, Int>
As we may notice all we need to do is to implement an the generic interface JpaRepository that takes as first type an entity and a second type the identifier of the entity.
Configurations
By default, Spring Data REST provides basic CRUD operations for the entities it exposes, including: GET, POST, PUT, and DELETE. In order to restrict operations to readonly one by implementing the following Spring interface:
@Configuration
class SpringDataRestConfiguration : RepositoryRestConfigurer {
companion object {
const val BASE_PATH = "/grocery/marketplace/query/api/v1"
}
override fun configureRepositoryRestConfiguration(
config: RepositoryRestConfiguration,
cors: CorsRegistry,
) {
val exposureConfiguration = config.setBasePath(BASE_PATH).exposureConfiguration
exposureConfiguration.withItemExposure { metadata: ResourceMetadata?, httpMethods: ConfigurableHttpMethods ->
httpMethods.disable(HttpMethod.PUT)
.disable(HttpMethod.PATCH).disable(HttpMethod.POST).disable(HttpMethod.DELETE)
}
.withCollectionExposure { metadata: ResourceMetadata?, httpMethods: ConfigurableHttpMethods ->
httpMethods.disable(HttpMethod.PUT)
.disable(HttpMethod.PATCH).disable(HttpMethod.POST).disable(HttpMethod.DELETE)
}
}
}
- First we set the default base path of our query api to: "/grocery/marketplace/query/api/v1"
- Second we restricted the allowed operations to only GET
Application in Action
When starting our Spring Boot Application, the Flyway will create the tables defined in our migration files and insert the data
You may want to connect to the created postgres server and view databases, you can create a pgadmin docker container as follow
docker run -p 5050:80 \
-e PGADMIN_DEFAULT_EMAIL=admin@example.com \
-e PGADMIN_DEFAULT_PASSWORD=admin \
--name pgadmin \
-d dpage/pgadmin4
Now, let's access our query api and explore the result:
GET /grocery/marketplace/query/api/v1
Host: http://localhost:8080
{
"_links" : {
// a subset of the full response, other entities cutted to make this example shorter
"products" : {
"href" : "http://localhost:8080/grocery/marketplace/query/api/v1/products{?page,size,sort*}",
"templated" : true
},
"categories" : {
"href" : "http://localhost:8080/grocery/marketplace/query/api/v1/categories{?page,size,sort*}",
"templated" : true
},
"profile" : {
"href" : "http://localhost:8080/grocery/marketplace/query/api/v1/profile"
}
}
}
Above we have a Spring Data REST response uses HAL format, which provides links to related resources.
- "products" and "categories": These are URLs for fetching product and category data, with templated parameters (
page
,size
,sort
) that allow for pagination and sorting. - "profile": A link to the API's profile, which provides metadata about the API.
These links allow clients to navigate the API easily and interact with related data without hardcoding URLs.
Find the full source code of this project at the following Github Repository: https://github.com/algodema/microservices-labs/tree/main/grocery-marketplace-product-queryapi
Conclusion
Understanding the levels of RESTful APIs helps developers create systems that are more robust, scalable, and easier to use. While many APIs remain at Levels 1 or 2, aiming for Level 3 ensures a fully RESTful experience that adheres to the principles of the web.
Which level does your API align with? Share your thoughts in the comments below!