sequenceDiagram participant IHM participant AthleteController participant AthleteService participant CrudRepository IHM->>AthleteController: Request findAllAthletes() AthleteController->>AthleteService: Call findAll() AthleteService->>CrudRepository: Call findAll() CrudRepository->>CrudRepository: Execute findAll() CrudRepository-->>AthleteService: Return list of athletes AthleteService-->>AthleteController: Return list of athletes AthleteController-->>IHM: Return allAthletes
Java - TP4
Ideas of Questions:
- Use Stream
- Check if age >=0
- Use an Enum for gender
Objectives
The aim is to develop a Spring Boot application:
- using layered programming
- creating new functionalities
- coding some API endpoints
The principle will be the same as what was done in IT project 2A.
1 Before you start
When you’ve finished, don’t forget to delete your service.
After each part, create a commit and push your code to the remote repository.
1.1 Launch service VSCode Java
We’re developing an API for public Internet access, which requires us to expose our application through a specific port (9000).
-
- It looks like :
https://user-<sspcloud_username>-<6digits_number>-user<optional: -9000>.user.lab.sspcloud.fr/
- It looks like :
I you meet any issue you can launch it manually:
- In menu Service catalog, Launch VSCode-python
- Service tab ➡️ Use a custom image:
odysseu/onyxia-vscode-java:j21
- Networking detail tab ➡️ Enable a custom service port:
9000
- Optional: Save this configuration
- Launch
The most important is to enable port 9000.
I you meet any issue with the docker image you can:
- Update it, using a recent source image
- Run manually Java, Maven and VSCode extensions after having launched the service
- Create the file below in folder work
- Add execution rights:
chmod +x installjava.sh
- Run it:
./installjava.sh
installjava.sh
# installjava.sh
# chmod +x installjava.sh
export JAVA_VERSION=21
export JAVA_HOME="/usr/lib/jvm/java-$JAVA_VERSION-openjdk-amd64"
export PATH="${JAVA_HOME}/bin:${PATH}"
sudo apt-get update
sudo apt-get install ca-certificates-java libbz2-dev openjdk-"$JAVA_VERSION"-jdk openjdk-${JAVA_VERSION}-jre-headless -y --no-install-recommends
sudo apt-get install maven -y
java --version
mvn --version
code-server --install-extension adamraichu.zip-viewer
code-server --install-extension redhat.java
code-server --install-extension vscjava.vscode-java-pack
code-server --install-extension bierner.markdown-mermaid
1.2 Update your forked repository
As you did in TP2, you need to update your fork of the ENSAI-2A-Java-TP repository.
1.3 Clone and Open Folder
In VSCode :
tp4 must be the root folder in your explorer.
1.4 Compile and run
Use this command to run your application:
mvn spring-boot:run
CTRL + C
to stop
After launching the Spring Boot application, it will be deployed to something like:
https://user-<sspcloud_username>-<6digits_number>-user<optional: -9000>.user.lab.sspcloud.fr/
If you meet that error: Web server failed to start. Port 9000 was already in use.
ps -aux | grep java
- kill processes (ask for help!)
Sometimes mvn clean
could be usefull.
2 Exercice
You’ll be working on an application that manages registrations for running races.
Features already implemented:
- List of all Athletes
- Add an Athlete
- Delete an Athlete
To make it easier to use, you’ll be using:
- Thymeleaf to provide an interface
- An embedded, in-memory H2 database
However, bear in mind that in modern applications, Java is not used for GUIs. For example, the standard stack at INSEE is:
- database: PostgreSQL
- backend: Java (API that exposes endpoints)
- frontend: React JS (uses API data for display)
2.1 Project structure
If you expand all the folders, you’ll notice that there are a lot of files.
Let’s focus on a few key points:
src/main/java/fr/ensai/running/
: the java classes of your Layered Spring Boot application- Layers: controller, service, repository, model…
Application.java
: the Main class to run your application
src/main/resources/
templates/
: html pagesapplication.yml
: Spring Boot properties of your applicationdata.sql
: Used to create and pop your database
pom.xml
: Maven config
/config
: To setup your app (security…)LoggerAspect.java
: Used to log before and after each method call
2.2 For students in IT sections
Let’s use a PostgreSQL database insted of an embedded H2:
-
- search on internet for the necessary configuration
-
- If all goes well, CloudBeaver will automatically connect to your database
- Otherwise you need to create a new connection entering Host, Database, Username and Password
2.3 Workflow example
- You want to display all Athletes
- You click on button Athletes on the navigation bar
- This takes you to the
/athletes
endpoint (check the navbar.html file if necessary)
- This takes you to the
- In the Controller, it match with method
findAllAthletes()
- This method calls method
findAll()
from the service- This method calls method
findAll()
from the repository- But if you check class AthleteRepository.java, there is no method findAll()
- In fact, it’s a native method of CrudRepository, so it is not necessary to implement it
- This method retrieves all athletes from the database
- This method calls method
- Once the data has been retrieved, the method returns the string ‘allAthletes’
- Which means that you will be redirected to the allAthletes.html page.
- This method calls method
2.4 API Rest
Before playing with the GUI, you’re going to create a few endpoints for your API.
In file ApiRestController.java, add the following endpoints:
If you haven’t any http client installed (Insomnia, Bruno, Postman…) on your machine you can use these bash commands to test your endpoints:
curl -X POST https://<your_url>.user.lab.sspcloud.fr/api/athlete \
-H "Content-Type: application/json" \
-d '{
"firstName": "Floria",
"lastName": "Guei",
"age": 34,
"gender": "Female"
}'
curl -X DELETE https://<your_url>.user.lab.sspcloud.fr/api/athlete/1
The ApiRestController class is annotated with @RestController, and is therefore used to create endpoints for a REST API.
Next, we’re going to use simple Controllers to interact with the GUI.
If you prefer, you can continue to develop REST API endpoints without using the GUI.
2.5 Create competitions
The goal is to display all competitions.
In your database there is a table called competition
(loaded at start-up via the data.sql file).
CREATE TABLE competition (
PRIMARY KEY,
id_competition SERIAL VARCHAR(255),
designation VARCHAR(255),
city DATE,
event_date FLOAT,
distance INTEGER
max_athletes );
-
- To store dates, use
LocalDate
from java.time.LocalDate
- To store dates, use
-
- list all competitions
- find a competition from its id
- delete a competition
-
/competition
: display all competition (allCompetition.html)/competition/delete/{id}
: to delete a competition using its id
2.6 Binding Athletes and Competitions
We have Athletes.
We have competitions.
Now we want to register athletes for competitions.
----------------------------------------------
-- Registration
----------------------------------------------
DROP TABLE IF EXISTS registration;
CREATE TABLE registration (
PRIMARY KEY,
id_registration SERIAL REFERENCES athlete(id_athlete),
id_athlete BIGINT REFERENCES competition(id_competition),
id_competition BIGINT DATE
registration_date
);
INSERT INTO registration (id_registration, id_athlete, id_competition, registration_date) VALUES
77770, 99990, 88880, '2024-03-01'),
(77771, 99991, 88881, '2024-03-02'),
(77772, 99992, 88882, '2024-03-03'),
(77773, 99993, 88883, '2024-03-04'),
(77774, 99994, 88884, '2024-03-05'),
(77775, 99995, 88880, '2024-03-06'),
(77776, 99996, 88881, '2024-03-07'),
(77777, 99997, 88882, '2024-03-08'),
(77778, 99998, 88883, '2024-03-09'),
(77779, 99999, 88884, '2024-03-10'),
(77780, 99990, 88881, '2024-03-11'),
(77781, 99991, 88882, '2024-03-12'),
(77782, 99992, 88883, '2024-03-13'),
(77783, 99993, 88884, '2024-03-14'),
(77784, 99994, 88880, '2024-03-15'),
(77785, 99995, 88882, '2024-03-16'),
(77786, 99996, 88883, '2024-03-17'),
(77787, 99997, 88884, '2024-03-18'),
(77788, 99998, 88880, '2024-03-19'),
(77789, 99999, 88881, '2024-03-20'),
(77790, 99990, 88882, '2024-03-21'),
(77791, 99991, 88883, '2024-03-22'),
(77792, 99992, 88884, '2024-03-23'),
(77793, 99993, 88880, '2024-03-24'),
(77794, 99994, 88881, '2024-03-25'),
(77795, 99995, 88883, '2024-03-26'),
(77796, 99996, 88884, '2024-03-27'),
(77797, 99997, 88880, '2024-03-28'),
(77798, 99998, 88881, '2024-03-29'),
(77799, 99999, 88882, '2024-03-30'); (
You can use Query creation from method names
SELECT r.athlete.id
FROM Registration r
WHERE r.competition.id = :idCompetition
Now let’s add some methods in CompetitionService
:
Try to delete a Competition. What’s wrong?
-
- to allow you to force deletion
2.7 Register an Athlete
Difficulty : ⭐ ⭐ ⭐
On the screen listing the athletes in a competition:
In Java classes:
-
- to create the Registration
- call the suitable repository method
If that was too easy, have fun creating other features.