In this lab we will build further automated tests for our project, focusing on integration testing. We will modify our App to allow local debugging to save time constantly building docker containers and add Integration testing to out GitHub Actions pipeline.
- Modify our App to allow local connections to the database for faster debugging.
- Create integration tests.
- Use code coverage from version control.
We have some tidying up in our existing project to make our life easier. We will modify the pom.xml
file to use a more up-to-date versions of our mysql dependencies that allows local connections.
If your mysql driver uses an older version, update the dependencies section of your pom.xml
to use version 8.0.28 of the mysql driver:
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>8.0.28</version>
</dependency>
We will also modify our App.java
file to use the latest mysql connector and to allow the hostname and delay for the database to be set programmatically. This will allow us to debug our code locally much more quickly than having to build docker containers each time a change is made.
Below is our updated connect
method. The updated lines are:
- The method definition adds
location
anddelay
parameters. Class.forName
which uses the most up-to-date MySQL driver.Driver.getConnection
uses thelocation
, and also updates the parameters to useallowPublicKeyRetrieval=true
as we are using a more up-to-date version of MySQL.
public void connect(String location, int delay) {
try {
// Load Database driver
Class.forName("com.mysql.cj.jdbc.Driver");
} catch (ClassNotFoundException e) {
System.out.println("Could not load SQL driver");
System.exit(-1);
}
int retries = 10;
boolean shouldWait = false;
for (int i = 0; i < retries; ++i) {
System.out.println("Connecting to database...");
try {
if (shouldWait) {
// Wait a bit for db to start
Thread.sleep(delay);
}
// Connect to database
con = DriverManager.getConnection("jdbc:mysql://" + location
+ "/employees?allowPublicKeyRetrieval=true&useSSL=false",
"root", "example");
System.out.println("Successfully connected");
break;
} catch (SQLException sqle) {
System.out.println("Failed to connect to database attempt " + i);
System.out.println(sqle.getMessage());
// Let's wait before attempting to reconnect
shouldWait = true;
} catch (InterruptedException ie) {
System.out.println("Thread interrupted? Should not happen.");
}
}
}
Modify the main method to use command line parameters, if supplied, or to default to localhost.
public static void main(String[] args) {
// Create new Application and connect to database
App a = new App();
if (args.length < 1) {
a.connect("localhost:33060", 10000);
} else {
a.connect(args[0], Integer.parseInt(args[1]));
}
Department dept = a.getDepartment("Development");
ArrayList<Employee> employees = a.getSalariesByDepartment(dept);
// Print salary report
a.printSalaries(employees);
// Disconnect from database
a.disconnect();
}
At the start of the project we built a JAR file with the version number and jar-with-dependencies
added to the name. This has slowly become problematic with the number of files where our version number is provided. Therefore, we will update the pom.xml
file to produce a JAR file called seMethods
.
The section we have to update is in the <build><plugins>
section for the maven-assembly-plugin
. The updated version is below:
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-assembly-plugin</artifactId>
<version>3.3.0</version>
<configuration>
<finalName>seMethods</finalName>
<archive>
<manifest>
<mainClass>com.napier.sem.App</mainClass>
</manifest>
</archive>
<descriptorRefs>
<descriptorRef>jar-with-dependencies</descriptorRef>
</descriptorRefs>
<appendAssemblyId>false</appendAssemblyId>
</configuration>
<executions>
<execution>
<id>make-assembly</id>
<phase>package</phase>
<goals>
<goal>single</goal>
</goals>
</execution>
</executions>
</plugin>
We have set <finalName>
and stated we do not want the ID attached (<appendAssemblyId>
is set to false
). Our Maven build will now produce a file called seMethods.jar
.
Now we need to update the Dockerfile
for the application to use the new JAR file name, and to provide the correct location for the database. The updated Dockerfile
is below.
FROM openjdk:latest
COPY ./target/seMethods.jar /tmp
WORKDIR /tmp
ENTRYPOINT ["java", "-jar", "seMethods.jar", "db:3306", "10000"]
Make sure you have exposed the docker database port
Your docker-compose.yml
file should be:
version: '3'
services:
# Application Dockerfile is in same folder which is .
app:
build: .
# db is is db folder
db:
build: db/.
command: --default-authentication-plugin=mysql_native_password
restart: always
ports:
- "33060:3306"
The last line tells docker to make port 3306 available on our local machine on port 33060
Delete your old containers and images and restart the database from the docker compose file using the arrow at line 8 below.
You can leave the database running to save having to rebuild each time you want to connect.
When the database is running and ready for connections the log from the docker container should show ready for connections
:
You should now be able to run the App locally without having to package and build docker images
Just start the App directly using the arrow next to the main method.
The new version of App should work locally either running directly or by starting from docker-compose and on GitHub Actions without any modification.
Test that all three scenarios are working
- Locally as described above by running the App directly
- Locally in docker using docker-compose to start the App (remember to delete target directory, old containers and images, repackage with maven, recreate and start App image using docker-compose )
- Remotely on GitHub Actions
We want to separate our tests into different files as we will have different types of tests. Unit tests and integration tests are different, and we want to manage them as such.
From the Continuous Integration lecture we defined the following steps in a basic build script:
- Clean.
- Compile source code.
- Integrate database.
- Run tests.
- Run inspections.
- Deploy software.
We will update our GitHub Actions workflow to separate the different stages
name: A workflow for my Hello World App
on: push
jobs:
UnitTests:
name: Unit Tests
runs-on: ubuntu-20.04
steps:
- name: Checkout
uses: actions/checkout@v2
with:
submodules: recursive
- name: Set up JDK 11
uses: actions/setup-java@v2
with:
java-version: '11'
distribution: 'adopt'
- name: Unit Tests
run: mvn -Dtest=com.napier.sem.AppTest test
build:
name: Build and Start Using docker-compose
runs-on: ubuntu-20.04
steps:
- name: Checkout
uses: actions/checkout@v2
with:
submodules: recursive
- name: Set up JDK 11
uses: actions/setup-java@v2
with:
java-version: '11'
distribution: 'adopt'
- name: Package and Run docker compose
run: |
mvn package -DskipTests
docker-compose up --abort-on-container-exit
We have separated our tasks into different jobs. This allows us to define build stages for more control. It also allows the different jobs to run concurrently on GitHub Actions speeding up the process. Commit and push this change. If successful then you should see the different stages on GitHub Actions as shown below.
Remember that Unit Tests are check the integrity of the smallest parts of our program. These do not need a connection to the database. We are now ready to write integration tests. These are tests against our database, and therefore we need to be connected. Our tests will be written in the same manner as unit tests, but require a bit more configuration.
Add a new Java file to the test folder called AppIntegrationTest.java
. The code for the file is below.
package com.napier.sem;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.TestInstance;
import java.util.ArrayList;
import static org.junit.jupiter.api.Assertions.*;
public class AppIntegrationTest
{
static App app;
@BeforeAll
static void init()
{
app = new App();
app.connect("localhost:33060", 30000);
}
@Test
void testGetEmployee()
{
Employee emp = app.getEmployee(255530);
assertEquals(emp.emp_no, 255530);
assertEquals(emp.first_name, "Ronghao");
assertEquals(emp.last_name, "Garigliano");
}
}
We are testing the getEmployee
method to see if it returns a correct answer. A random entry to the database has been used. With this in place, we can update our GitHub Actions workflow.
All we need to do now is update the GitHub Actions main.yml
file to run our integration tests:
name: A workflow for my Hello World App
on:
push:
branches:
- master
- develop
jobs:
UnitTests:
name: Unit Tests
runs-on: ubuntu-20.04
steps:
- name: Checkout
uses: actions/checkout@v2
with:
submodules: recursive
- name: Set up JDK 11
uses: actions/setup-java@v2
with:
java-version: '11'
distribution: 'adopt'
- name: Unit Tests
run: mvn -Dtest=com.napier.sem.AppTest test
IntegrationTests:
name: Integration Tests
runs-on: ubuntu-20.04
steps:
- name: Checkout
uses: actions/checkout@v2
with:
submodules: recursive
- name: Set up JDK 11
uses: actions/setup-java@v2
with:
java-version: '11'
distribution: 'adopt'
- name: Integration Tests
run: |
docker build -t database ./db
docker run --name employees -dp 33060:3306 database
mvn -Dtest=com.napier.sem.AppIntegrationTest test
docker stop employees
docker rm employees
docker image rm database
build:
name: Build and Start Using docker-compose
runs-on: ubuntu-20.04
steps:
- name: Checkout
uses: actions/checkout@v2
with:
submodules: recursive
- name: Set up JDK 11
uses: actions/setup-java@v2
with:
java-version: '11'
distribution: 'adopt'
- name: Package and Run docker compose
run: |
mvn package -DskipTests
docker-compose up --abort-on-container-exit
We have added an integration tests
stage to invoke mvn test
on the AppIntegrationTest
file. Commit and push your changes and check that everything still works on GitHub Actions
Now your task is to write integration tests to ensure that your application works correctly in all cases. You have the template in AppIntegrationTest.java
. Add similar tests that test all the pathways and conditions through your code.
Last week we looked at using IntelliJ to provide a code coverage report. This is good for working at a single workstation, but our aim is to make information global. Therefore, we will use an online code coverage tool to generate reports.
We are going to use a service called Codecov. First, you need to go to their website and signup via your GitHub account. The process from then should be fairly straightforward, but if you have any problems ask.
We need Maven to generate reports for us. There are different plugins that can do this for us, and we will use the jacoco one. Add the following to the plugins
section of the pom.xml
file:
<plugin>
<groupId>org.jacoco</groupId>
<artifactId>jacoco-maven-plugin</artifactId>
<version>0.8.2</version>
<executions>
<execution>
<goals>
<goal>prepare-agent</goal>
</goals>
</execution>
<execution>
<id>report</id>
<phase>test</phase>
<goals>
<goal>report</goal>
</goals>
</execution>
</executions>
</plugin>
That is all we have to do in our Maven file. Everything else is automated.
We will add another action to the Integration Test stage of our GitHub Actions main.yml
file to upload the coverage reports created during the maven test stage to codecov.
name: A workflow for my Hello World App
on:
push:
branches:
- master
- develop
jobs:
UnitTests:
name: Unit Tests
runs-on: ubuntu-20.04
steps:
- name: Checkout
uses: actions/checkout@v2
with:
submodules: recursive
- name: Set up JDK 11
uses: actions/setup-java@v2
with:
java-version: '11'
distribution: 'adopt'
- name: Unit Tests
run: mvn -Dtest=com.napier.sem.AppTest test
IntegrationTests:
name: Integration Tests
runs-on: ubuntu-20.04
steps:
- name: Checkout
uses: actions/checkout@v2
with:
submodules: recursive
- name: Set up JDK 11
uses: actions/setup-java@v2
with:
java-version: '11'
distribution: 'adopt'
- name: Integration Tests and CodeCov
run: |
docker build -t database ./db
docker run --name employees -dp 33060:3306 database
mvn -Dtest=com.napier.sem.AppIntegrationTest test
docker stop employees
docker rm employees
docker image rm database
- name: CodeCov
uses: codecov/[email protected]
with:
token: ${{ secrets.CODECOV_TOKEN }} # now required for public repos too
directory: ./target/site/jacoco
flags: Integration Tests # optional
verbose: true # optional (default = false)
slug: <GitHub username>/<repository name>
build:
name: Build and Start Using docker-compose
runs-on: ubuntu-20.04
steps:
- name: Checkout
uses: actions/checkout@v2
with:
submodules: recursive
- name: Set up JDK 11
uses: actions/setup-java@v2
with:
java-version: '11'
distribution: 'adopt'
- name: Package and Run docker compose
run: |
mvn package -DskipTests
docker-compose up --abort-on-container-exit
Please note that the following part of this code has to be updated with your own details:
slug: <github-username>/<repo-name>
For example:
slug: kevin-sim/sem_employees
The jacoco maven plugin creates html reports in the ./target/site/jacoco
folder. The new action shown below uploads this folder to https://codecov.io
- name: CodeCov
uses: codecov/[email protected]
with:
token: ${{ secrets.CODECOV_TOKEN }}
directory: ./target/site/jacoco
flags: Integration Tests # optional
verbose: true # optional (default = false)
slug: Lawful24/sem-demo-labs
Just like GitHub, CodeCov also uses personal access tokens, but for each repository instead of each user. Before we can get the code coverage report of our repository, we need to add this token to our repository on GitHub.
GitHub Secrets is a handy tool if you want certain parts of your project to be accessible by the users or services of your choosing. It allows you to assign hidden values to your repository that only the admins of the repository can access and don't get downloaded when someone clones your repository.
For this project, we will only need to add one secret to our repository to verify ourselves when CodeCov tries to access our project during the GitHub action job defined above.
First, go to https://codecov.io/gh/<github-username>/<repo-name>
. This is the CodeCov page of your repository. Make sure to replace the placeholder text with your details, for example, https://codecov.io/gh/kevin-sim/sem_employees
.
You will come across the default page since we have yet to share our code coverage report with CodeCov.
Copy the whole line starting with CODECOV_TOKEN=...
and go to the Settings page of your GitHub repository.
Find "Secrets and variables" and go to "Actions".
Here, we have two options to register a new secret. We need a new "Repository Secret".
Enter "CODECOV_TOKEN
" to the top bar and copy the value of the token into the text area below. Then click on "Add secret".
Now you should have an entry on the previous page in "Repository Secrets".
Now commit and push the changes you have made to your main.yml
file. GitHub Actions should undertake the build process, and once one stage is complete you can view it at https://codecov.io/gh/<github-username>/<repo-name>
. For example, https://codecov.io/gh/kevin-sim/sem_employees
.
Under the Settings area of Codecov you will find the necessary markdown for your code coverage badge:
Add the Markdown to your project readme.md
for both the master
and develop
branches. This will allow you to see your code coverage status from your main GitHub project page.
We will now add a new feature to our application: adding a new employee. We will adopt a Test-Driven Development approach by first writing the test. First, add the empty method to App.java
:
public void addEmployee(Employee emp)
{
}
And add the following code to your AppIntergrationTest.java
file:
@Test
void testAddEmployee()
{
Employee emp = new Employee();
emp.emp_no = 500000;
emp.first_name = "Kevin";
emp.last_name = "Chalmers";
app.addEmployee(emp);
emp = app.getEmployee(500000);
assertEquals(emp.emp_no, 500000);
assertEquals(emp.first_name, "Kevin");
assertEquals(emp.last_name, "Chalmers");
}
Running this test (you should be able to do so straight from IntelliJ if your database is still running in the background) will fail. This is because we haven't written the code yet. Let us do that now. Update addEmployee
to the following:
public void addEmployee(Employee emp)
{
try
{
Statement stmt = con.createStatement();
String strUpdate =
"INSERT INTO employees (emp_no, first_name, last_name, birth_date, gender, hire_date) " +
"VALUES (" + emp.emp_no + ", '" + emp.first_name + "', '" + emp.last_name + "', " +
"'9999-01-01', 'M', '9999-01-01')";
stmt.execute(strUpdate);
}
catch (Exception e)
{
System.out.println(e.getMessage());
System.out.println("Failed to add employee");
}
}
And the test should pass. Commit and push and check with GitHub Actions.
As always, cleanup your system. Stop any running containers, commit everything, and bring your branches up-to-date.