I just want to get it out there—once I grokked Spring Data REST, I was sad it took me this long to find it. Hook up Java Persistence Annotation (JPA) and Hibernate to your Spring Boot app, add the Spring Data REST (SDR) dependency, and then a few lines of annotated Java code, and BOOM you have a REST API. As an added bonus, it does the right thing and actually makes your REST API automatically accessible to a Hypermedia as the Engine of Application State (HATEOAS) client.
Having used it a bit more, I need to temper this enthusiasm. SDR enforces HATEOAS by default and you have to put in extra work to turn it off. I feel like HATEOAS is great in theory, and not practical in practice, especially when other humans are programming against your REST API.
To be more precise, when I talk about REST in this blog post I am referring to Level 2 maturity in Richardson’s model of API maturity. In my work, and in talking with other developers, the Fielding level of REST, while interesting, is not really practical in a lot of their use cases. There are few clients, especially in the major web and mobile frameworks, working with a Level 3 API. The benefit from implementing the full REST mantra provides minimal value compared to the extra effort needed to implement it.
This is a long way of saying, be ready for a HATEOAS API, and know that you will have to put in a little work to get it return object IDs.
One of the fun parts about my job is getting to explore new technology. For most of my developer career I was in the Java EE camp (though not EJBs) versus the Spring camp. I think this has mostly to do with historical reasons. At the time when I decided I was not team Spring (July 2004), Spring was an up and comer and, if memory serves, involved a lot of XML. Well, one of the many benefits of coming to VMware is getting to learn Spring from Spring people.
My internal advocate team is working on an internal project and I am leading up the API/backend pieces. I have already used Spring Boot with JPA/Hibernate and liked it, but I remember seeing this Spring Data REST thing and was curious about what it could do for me. I couldn’t quite find a “getting started” guide that put all the pieces together for me, and so, the rest of this blog post is the getting started I wanted.
I assume:
Also, there is a GitHub repository where all the code is available for this blog post. I think that covers about all you need—so away we go!
First step is to create a database in PostgreSQL. For the purposes of this post, we will call it mydatabase
. Start up your PostgreSQL instance (on standard ports).
Next, you’ll need to set up your Spring Boot application. You can do this using the Spring Initializr website. Select the following options:
Once you’ve generated the project, extract the contents and open it in your preferred IDE. I am an IntelliJ fan.
Go to your application.properties
file in src/main/java/resources and add the following configuration for your PostgreSQL database:
spring.datasource.url=jdbc:postgresql://localhost:5432/mydatabase
spring.datasource.username=postgres
spring.datasource.password=your_password_here
spring.jpa.hibernate.ddl-auto=create-drop
Make sure to replace your_password_here
with the password for your PostgreSQL database. In this case and for convenience sake, we are using the postgres
user, which is the root user for the database. Note: DO NOT DO THIS IN PRODUCTION.
Time for some coding. We are going to work with two tables in our database and entity classes—Person and Department. There is a one to many relationship between Department and People.
First, let’s create the Department
entity:
package com.vmware.tanzu.datarestblog.dataobjects;
import jakarta.persistence.Entity;
import jakarta.persistence.GeneratedValue;
import jakarta.persistence.GenerationType;
import jakarta.persistence.Id;
import jakarta.persistence.OneToMany;
import lombok.Data;
import lombok.NoArgsConstructor;
import java.util.List;
@Entity
@Data
@NoArgsConstructor
public class Department {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
private String name;
@OneToMany(mappedBy = "department")
private List<Person> people;
}
This class defines a Department
entity with id
, name
, and people
attributes. The people
attribute is a list of Person
entities, and the @OneToMany
annotation is used to specify the relationship between Department
and Person
. The mappedBy
attribute is used to specify the name of the attribute on the Person
entity that maps back to Department
.
Next, let’s create the Person
entity with a reference to Department
:
import jakarta.persistence.Entity;
import jakarta.persistence.GeneratedValue;
import jakarta.persistence.GenerationType;
import jakarta.persistence.Id;
import jakarta.persistence.ManyToOne;
import lombok.Data;
import lombok.NoArgsConstructor;
@Entity
@Data
@NoArgsConstructor
public class Person {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
private String firstName;
private String lastName;
@ManyToOne
private Department department;
}
This class includes a department
attribute that references a Department
entity, and the @ManyToOne
annotation is used to specify the relationship between Person
and Department
.
Create a PersonRepository
interface:
import org.springframework.data.repository.PagingAndSortingRepository;
import org.springframework.data.rest.core.annotation.RepositoryRestResource;
import com.vmware.tanzu.datarestblog.dataobjects.Person;
@RepositoryRestResource(collectionResourceRel = "people", path = "people")
public interface PersonRepository extends PagingAndSortingRepository<Person, Long> {
}
This interface extends the PagingAndSortingRepository
interface provided by Spring Data JPA and specifies the Person
entity type and the type of the entity’s primary key (Long
). The @RepositoryRestResource
annotation is used to specify the endpoint for accessing Person
resources.
Adding this interface, automagically, creates a REST endpoint at http://your.host.info/people
when your application is started. Because we extended PagingAndSortingRepository
, that endpoint has the ability to return subsets of the data as a page. You can also sort the data before returning it by specifying the sort parameter in the request URL. But a limitation of this inheritance is that the class we extended only contains methods for findAll()
. We will fix that limitation in later code.
Now let’s make an interface for the department table. Since the table will remain small we don’t have to make the results pageable or sortable. We will just extend CrudRepository
to get all the normal CRUD operations on the department table. The JavaDoc for CrudRepository
shows that this class supports the standard CRUD method calls, but not paging and sorting.
Create a DepartmentRepository
Interface:
import com.vmware.tanzu.datarestblog.dataobjects.Department;
import org.springframework.data.repository.CrudRepository;
import org.springframework.data.rest.core.annotation.RepositoryRestResource;
@RepositoryRestResource(collectionResourceRel = "departments", path = "departments")
public interface DepartmentRepository extends CrudRepository<Department, Long> {
}
And that’s it! We are now done with coding. That is all the code needed to create a fully functioning HATEOAS REST interface!
I don’t blame you if you don’t believe me—I was stunned it was this easy. Let’s go ahead and take a look at the REST API we just created.
Start your application, and navigate to http://localhost:8080
in your web browser. You should see JSON that looks like:
{
"_links": {
"people": {
"href": "http://localhost:8080/people{?page,size,sort}",
"templated": true
},
"departments": {
"href": "http://localhost:8080/departments"
},
"profile": {
"href": "http://localhost:8080/profile"
}
}
}
This JSON is a machine (and human) readable representation of the entities available, their URL, and if the URL accepts any parameters.
If you click on the link for the people endpoint, you should see an empty JSON array ([]
), but that’s because there is no data in our database. I have also included a SQL file in the base of the GitHub repository for this blog post that you can use to populate the database. You can also just use your favorite tool and add any data you want.
Once you have added some people to the database, a GET request to the people endpoint should return all the data in the person
table in the database. Along with the values for each person, it returns the URL to get that person’s data and the URLs to get the information from child tables. As explained above, if you click the URL for an individual person such as http://localhost:8080/people/1
, you will receive an error. Again, this is because the PagingAndSortingRepository
does not contain any methods to find a person by ID.
These same capabilities apply to the departments endpoint but you will notice, its URL does not accept parameters. This means it does not allow for paging or sorting.
Let’s fix the PersonRepository
to include a method for finding a Person
entity by ID. Paste the following method inside the PersonRepository
class. The class should now look like the following:
public interface PersonRepository extends PagingAndSortingRepository<Person, Long> {
Person findById(Long id);
}
That’s it! We just added a method to our class that will return a Person
object when given an ID. If you redeploy your app and go back to your rest endpoint, nothing has changed on the top level. However, now the URLs for the individual person in the people response will actually work.
Now suppose you want to have an endpoint that returns a custom result set (the result is not one of our entity objects) based on a query in Java Persistence Query Language (JPQL) or in SQL. Never fear, that is just about as simple as well. All we need to know is an annotation and two lines of code.
Here’s an example of how you can define a custom method that returns a list of Person
entities with a given lastName
and firstName
Steve.
Just add the following two lines to below the findById
method:
@Query("SELECT p FROM Person p WHERE p.lastName = :lastName AND p.firstName = ‘Steve’")
List<Person> findAllStevesByLastName(@Param("lastName") String lastName);
In this example, we defined a custom method named findAllStevesByLastName
that takes a String
parameter named lastName
. We used the @Query
annotation to specify a JPQL query that selects Person
entities with the given lastName
. We also used the @Param
annotation to bind the lastName
parameter to the query. Both the method name and the @Param
name will be part of the resulting URL.
Now, when you access the /people
endpoint of your REST API, you will have a new endpoint available at /people/search/findAllStevesByLastName?lastName={lastName}
that returns a list of Person
entities with the given lastName
. Note that the name of the custom method (findAllStevesByLastName
) is used in the endpoint URL, and the lastName
parameter is passed as a query parameter.
Here is an example URL that should work if you imported the data from the SQL file.
To find me:
http://localhost:8080/people/search/findAllStevesByLastName?lastName=Pousty
To get an empty result set:
http://localhost:8080/people/search/findAllStevesByLastName?lastName=Junod
Note: The URL and parameters are case sensitive.
When I started on the internal project, I was guessing I was going to be writing a bunch of JQL and REST controller code. Having seen how easy and powerful Spring Data REST is for creating REST endpoints, it will actually be much less often that I actually hand write all those endpoints.
We only scratched the surface of what is possible with Spring Data REST. We didn’t cover topics like creating a listener for an API call using events or excerpting data to avoid multiple GETs. I encourage you to explore the resources on the product page.
I hope this blog can get you started on trying it out. Let me know if you found it as eye opening as I did. You can also ask me “have you been under a rock for years?” Either way, I look forward to seeing what you build with Spring Data REST and PostgreSQL—you are using PostgreSQL aren’t you?!?