In this lab we will automate our build process using [GitHub Actions] (https://github.com/features/actions). CI is Continuous Integration, a software engineering method where we ensure our local development software versions are merged into the mainline code several times a day. There are several CI approaches available, but GitHub actions is easy to plug into our software production pipeline.
After this lab you will be able to:
-
Pull your project to return to your previous development state.
-
Add continuous integration to your project.
-
Integrate Docker into CI build step.
-
Define the Gitflow Workflow.
-
Link Docker containers.
-
Package an application JAR with IntelliJ.
-
Create a release on GitHub.
At the end of the last lab we had a working application that we could deploy to Docker. Everything was done using three files:
-
A pom.xml Maven build file, which we have not explored further yet.
-
An App.java code file that contains our current code which is just a Hello World example.
-
A Dockerfile that specifies how to run our application in a separate Docker container.
We have three other files in our repository:
-
A .gitignore file to tell Git which files and folders to ignore for versioning.
-
A README.md file for our project.
-
A LICENSE file defining the licensing terms for our project.
Everything is in our GitHub repository. We can pull this back in IntelliJ to start from where we left off. If your code is still on the machine you are using you can ignore this step.
If you are using your own machine IntelliJ will open the last project that was opened. If not then you will need to clone the project from your GitHub Repository
IntelliJ Start Window
The button to click on is Get from VCS, then select Git:
IntelliJ Import from Git
If you have not saved login credentials you may be asked to provide these The simplest method is to click on Login with GitHub and add your details:
Note that GitHub now prefers a token for authentication rather than a password.
Go to your GitHub account. From the menu at the top right select settings
On the left hand side select Developer Settings
On the next screen select Personal access tokens then Generate New Token
Tick the workflow option which will allow us access using our CI environment and at the bottom of the page select generate token
The Token that is generated is used instead of your password. Make sure you take a note of this as once you navigate away from the page you will no longer be able to retrieve it.
Next time you try to push code to your repository a GitHub login window will appear. Use the token instead of your password. Sometimes I find that this does not work the first time and I am prompted again from the IntelliJ terminal window. Entering the details again seems to fix the issue.
Now we need to check that everything works correctly. Perform the following steps:
-
Build the project (Build then Build Project).
-
Run the project locally (open App.java and click the green triangle next to public class App and select Run App.main()).
-
Install Docker and the IntelliJ Docker plugin if needed (see the last lab).
-
Run the project via Docker (open Dockerfile and click the green triangles at the top of the file and select Run on Docker).
Hopefully everything has worked and we are back to the point we left off at last week. Remember these steps. You will need to repeat them every time you pull back your project to a new local system.
We can now set-up GitHub Actions. On your GitHub repository select the actions tab at the top then select set up workflow yourself
This will create a file in your repository named
.github/workflows/main.yml
Replace the default text with the following
name: A workflow for my Hello World App
on: push
jobs:
build:
name: Hello world action
runs-on: ubuntu-20.04
steps:
- name: Checkout
uses: actions/checkout@v2
- name: Set up JDK 11
uses: actions/setup-java@v2
with:
java-version: '11'
distribution: 'adopt'
- name: Compile with Maven
run: mvn clean package
- name: Build Docker Image
run: docker build -t semimage .
- name: Run image
run: docker run --name semcontainer -d semimage
- name: view logs
run: docker logs semcontainer
To sync our local version do a pull from IntelliJ
To check if our CI workflow is working make some changes and push them to GitHub
- Add some text to your
Readme.md
file. - Add the updates to the commit.
- Create a commit. Use a sensible message.
- Push the commit.
Now we can go to GitHub see if our build was successful.
You can click on the action to show more details of the build.
The stages shown above duplicate the stages in our workflow file that we defined above main.yml
You might have seen build status badges on GitHub before like this one:
You can add the badge for your build status to your README.md
file as well. To do this, add the following text to your Readme.md
file:
![GitHub Workflow Status (branch)](https://img.shields.io/github/actions/workflow/status/<repository owner>/<repository>/<action file name>.yml?branch=<branch name>)
You will need to replace the parameter <repository owner>
, <repository>
, <action file name>
and <branch name>
with the correct values.
- replace
<repository owner>
with the GitHub username of the repository you are working on. For these lab exercises this will be your own GitHub username, for the coursework, it will be the GitHub username of the group member owning your group's repository. - replace
<respository>
with the repository's name. - replace
<action file name>
with the name of the GitHub actions file that contains the build job you want to get the batch for. - replace
<branch name>
with the name of the branch you want to build batch for.
The batch for the master
branch for the repository sem
on my account asteyven
will be as follows.
![GitHub Workflow Status (branch)](https://img.shields.io/github/actions/workflow/status/asteyven/sem/main.yml?branch=master)
Now go through our Git update steps:
- Add files to commit.
- Create commit.
- Push to GitHub.
Now if you go to your GitHub page you should see the following:
And now we have our project automatically building on pushes to GitHub, and the current build status
You can add various badges to your project. Shields.io is one such site that provides badges. We are going to add two to our README.md
: one for our license and one for our release. The license badge takes the URL:
[![LICENSE](https://img.shields.io/github/license/<github-username>/sem.svg?style=flat-square)](https://github.com/<github-username>/sem/blob/master/LICENSE)
Just replace <github-username>
with your GitHub username. The release badge is:
[![Releases](https://img.shields.io/github/release/<github-username>/sem/all.svg?style=flat-square)](https://github.com/<github-username>/sem/releases)
And then update your GitHub repository:
- Add files to commit.
- Create commit.
- Push to GitHub.
If you go to your repository's dashboard in GitHub you should see your new badges.
Our next step is to set-up our workflow. This is our approach to managing separate features and collaborators in our project. Gitflow is one such workflow that works well with Git. Gitflow is examine further in Lecture 02. To work with Gitflow we manage several branches:
- master which is the main Git branch. This is created automatically when a Git repository is created. Only main releases are tracked in this branch.
- develop is the integration branch. Features are merged into this branch as they are completed. It is a feature integration branch.
- feature branches are where new features are worked on before integration with with
develop
. - release is where releases are made. A release is normally a collection of features, or a set point in time. Note that a release must be a working version. The
release
branch comes from a version ofdevelop
.master
is a version ofrelease
. - hotfix branches are maintenance ones based on
master
. We are fixing a production version of the code, so rather than working fromdevelop
we work frommaster
.
The first step in setting up Gitflow is the creation of a develop
branch in our project. We can do this in IntelliJ. Select VCS, Git then Branches... to open the branches window:
Select New Branch and call the branch develop. Make sure the Checkout branch checkbox is ticked.
The develop
branch only exists on the local system. To add it to GitHub we have to perform a push. Do this now. From IntelliJ, VCS, Git then Push. Click Push and the branch will be added to GitHub. You can confirm this on GitHub by opening the branches drop-down, refreshing the page if you are currently on it:
GitHub will have automatically added this branch to its build. As develop
is a key branch of our project we will add this build status to the README.md
file.
This time I will use Shields.io to create a build badge. Go to https://shields.io/badges. In the menu on the left, select Build and then GitHub Actions Workflow Status. Fill in the UserName, RepositoryName, Workflow name (taken from the main.yml file) and branch name (the branch name field shows up after clicking Show optional parameters).
Update the README.md
as below (keep the other badges):
# Software Engineering Methods
![GitHub Workflow Status (branch)](https://img.shields.io/github/actions/workflow/status/<username>/<repository>/<action file name>.yml?branch=<branch>)
And add this to GitHub:
- Add files to commit.
- Create commit.
- Push commit to GitHub.
And if you go to the dashboard for the repository on GitHub you will see that nothing has changed. That is because we have pushed to our develop
branch, not the master
branch. You can see the updates by switching to the develop
branch on GitHub using the branches drop-down from earlier:
The formatting of the above screenshot is achieved with the following markdown:
# Software Engineering Methods
* Master Build Status ![GitHub Workflow Status (branch)](https://img.shields.io/github/actions/workflow/status/<username>/<repository>/<action file name>.yml?branch=<master branch>)
* Develop Branch Status ![GitHub Workflow Status (branch)](https://img.shields.io/github/actions/workflow/status/<username>/<repository>/<action file name>.yml?branch=<develop branch>)
* License [![LICENSE](https://img.shields.io/github/license/<username>/<repository>.svg?style=flat-square)](https://github.com/<username>/<repository>/blob/master/LICENSE)
* Release [![Releases](https://img.shields.io/github/release/<username>/<repository>/all.svg?style=flat-square)](https://github.com/<username>/<repository>/releases)
To end this lab we will add a new feature to our application - database support via MongoDB. We will perform the following steps:
- Start a MongoDB server via Docker.
- Start a new feature in our project.
- Add MongoDB support to our application.
- Link our container with the MongoDB container.
- Test that everything works.
- Merge the feature into our
develop
branch. - Create a release.
- Add a version.
This may seem like a lot of steps, but individually they are simple. What could be seen as the hardest part - setting up a database and connecting to it - is simple in our build pipeline.
Our first step is to start a new MongoDB container. Let us do this via IntelliJ rather than the command line.
Open the Services panel at the bottom of IntelliJ and make sure Images is highlighted. The button on the top-left allows us to pull images for Docker. Click this button to open the Images Console window:
Type mongo press ctrl + Enter to start. The latest version of MongoDB will now be pulled as an image. It will appear in the Docker panel of IntelliJ:
With mongo:latest
selected, click the plus sign to Create Container. This will open the following window:
Add the run options from the Modify Options Link
MongoDB is a server application which listens on port 27017. We could just open that port, but just in case MongoDB is already running locally we will switch ports. We looked at this in the last lab. In the Run Options text box add -p 27000:27017 as shown in the image. Then click Run. IntelliJ will start the container and it will be waiting for you to use.
We are going to add a new feature to our application. To do this, we need to create a new branch as we did for develop
. The steps you need to undertake are:
- Create a new branch called
feature/mongo-intergration
(Git then New Branch - Push the branch to GitHub.
That is it. We are now working on a feature branch, which we created from our develop
branch since it was the one we had checked out.
We will use Maven to manage the import of MongoDB functionality into our application. This is done by adding dependencies to our pom.xml
file. IntelliJ will recognise these dependencies and pull in the relevant libraries and functionality. The new code we want is:
<dependencies>
<dependency>
<groupId>org.mongodb</groupId>
<artifactId>mongodb-driver</artifactId>
<version>3.6.4</version>
</dependency>
</dependencies>
We add this to the pom.xml
file as follows:
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.napier.sem</groupId>
<artifactId>seMethods</artifactId>
<version>0.1.0.1</version>
<dependencies>
<dependency>
<groupId>org.mongodb</groupId>
<artifactId>mongodb-driver</artifactId>
<version>3.6.4</version>
</dependency>
</dependencies>
</project>
If IntelliJ does not automatically download the dependencies click on the refresh button on the maven panel:
IntelliJ will manage everything for us. Initially the test org.mongodb
and mongodb-driver
will be red, but once the import is complete it will turn black. Let us to a commit.
- Add files to the commit.
- Create the commit.
- Push the commit to GitHub.
Now we can test that we can talk to the MongoDB server. We will update App.java
to the following:
package com.napier.sem;
import com.mongodb.MongoClient;
import com.mongodb.client.MongoDatabase;
import com.mongodb.client.MongoCollection;
import org.bson.Document;
public class App
{
public static void main(String[] args)
{
// Connect to MongoDB on local system - we're using port 27000
MongoClient mongoClient = new MongoClient("localhost", 27000);
// Get a database - will create when we use it
MongoDatabase database = mongoClient.getDatabase("mydb");
// Get a collection from the database
MongoCollection<Document> collection = database.getCollection("test");
// Create a document to store
Document doc = new Document("name", "Kevin Sim")
.append("class", "Software Engineering Methods")
.append("year", "2021")
.append("result", new Document("CW", 95).append("EX", 85));
// Add document to collection
collection.insertOne(doc);
// Check document in collection
Document myDoc = collection.find().first();
System.out.println(myDoc.toJson());
}
}
Now all we have to do is run the application normally (i.e. not as a Docker container). Select Run then Run and select App as the configuration. Your application should launch, connect to the MongoDB server running in the Docker container and perform some basic operations as shown. The console output will look something like the following:
"C:\Program Files\Java\jdk1.8.0_181\bin\java.exe" "-javaagent:C:\Program Files\JetBrains\IntelliJ IDEA 2020.3.2\lib\idea_rt.jar=51700:C:\Program Files\JetBrains\IntelliJ IDEA 2020.3.2\bin" -Dfile.encoding=UTF-8 -classpath "C:\Program Files\Java\jdk1.8.0_181\jre\lib\charsets.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\deploy.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\access-bridge-64.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\cldrdata.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\dnsns.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\jaccess.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\jfxrt.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\localedata.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\nashorn.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\sunec.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\sunjce_provider.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\sunmscapi.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\sunpkcs11.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\ext\zipfs.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\javaws.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\jce.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\jfr.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\jfxswt.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\jsse.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\management-agent.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\plugin.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\resources.jar;C:\Program Files\Java\jdk1.8.0_181\jre\lib\rt.jar;C:\Users\KevL\Desktop\SEM_Demo\target\classes;C:\Users\KevL\.m2\repository\org\mongodb\mongodb-driver\3.6.4\mongodb-driver-3.6.4.jar;C:\Users\KevL\.m2\repository\org\mongodb\bson\3.6.4\bson-3.6.4.jar;C:\Users\KevL\.m2\repository\org\mongodb\mongodb-driver-core\3.6.4\mongodb-driver-core-3.6.4.jar" com.napier.sem.App
Feb 02, 2021 11:41:28 AM com.mongodb.diagnostics.logging.JULLogger log
INFO: Cluster created with settings {hosts=[localhost:27000], mode=SINGLE, requiredClusterType=UNKNOWN, serverSelectionTimeout='30000 ms', maxWaitQueueSize=500}
Feb 02, 2021 11:41:28 AM com.mongodb.diagnostics.logging.JULLogger log
INFO: Cluster description not yet available. Waiting for 30000 ms before timing out
Feb 02, 2021 11:41:28 AM com.mongodb.diagnostics.logging.JULLogger log
INFO: Opened connection [connectionId{localValue:1, serverValue:1}] to localhost:27000
Feb 02, 2021 11:41:28 AM com.mongodb.diagnostics.logging.JULLogger log
INFO: Monitor thread successfully connected to server with description ServerDescription{address=localhost:27000, type=STANDALONE, state=CONNECTED, ok=true, version=ServerVersion{versionList=[4, 4, 3]}, minWireVersion=0, maxWireVersion=9, maxDocumentSize=16777216, logicalSessionTimeoutMinutes=30, roundTripTimeNanos=4243600}
Feb 02, 2021 11:41:28 AM com.mongodb.diagnostics.logging.JULLogger log
INFO: Opened connection [connectionId{localValue:2, serverValue:2}] to localhost:27000
{ "_id" : { "$oid" : "60193a68517ac5004c914c07" }, "name" : "Kevin Sim", "class" : "Software Engineering Methods", "year" : "2021", "result" : { "CW" : 95, "EX" : 85 } }
Process finished with exit code 0
If you look at the logs of the MongoDB container in IntelliJ you will see the something similar to the following partial log:
{"t":{"$date":"2021-02-02T11:41:28.619+00:00"},"s":"I", "c":"NETWORK", "id":22943, "ctx":"listener","msg":"Connection accepted","attr":{"remote":"172.17.0.1:58692","connectionId":2,"connectionCount":2}}
{"t":{"$date":"2021-02-02T11:41:28.619+00:00"},"s":"I", "c":"NETWORK", "id":51800, "ctx":"conn2","msg":"client metadata","attr":{"remote":"172.17.0.1:58692","client":"conn2","doc":{"driver":{"name":"mongo-java-driver","version":"3.6.4"},"os":{"type":"Windows","name":"Windows 10","architecture":"amd64","version":"10.0"},"platform":"Java/Oracle Corporation/1.8.0_181-b13"}}}
{"t":{"$date":"2021-02-02T11:41:28.639+00:00"},"s":"I", "c":"STORAGE", "id":20320, "ctx":"conn2","msg":"createCollection","attr":{"namespace":"mydb.test","uuidDisposition":"generated","uuid":{"uuid":{"$uuid":"b862f932-c8aa-4b30-904b-0c3a1e6a1dd2"}},"options":{}}}
{"t":{"$date":"2021-02-02T11:41:28.658+00:00"},"s":"I", "c":"INDEX", "id":20345, "ctx":"conn2","msg":"Index build: done building","attr":{"buildUUID":null,"namespace":"mydb.test","index":"_id_","commitTimestamp":{"$timestamp":{"t":0,"i":0}}}}
{"t":{"$date":"2021-02-02T11:41:29.035+00:00"},"s":"I", "c":"NETWORK", "id":22944, "ctx":"conn2","msg":"Connection ended","attr":{"remote":"172.17.0.1:58692","connectionId":2,"connectionCount":0}}
{"t":{"$date":"2021-02-02T11:41:29.035+00:00"},"s":"I", "c":"NETWORK", "id":22944, "ctx":"conn1","msg":"Connection ended","attr":{"remote":"172.17.0.1:58688","connectionId":1,"connectionCount":1}}
A good time to push this update to GitHub.
- Add files to commit.
- Create commit.
- Push to GitHub.
Now we need to modify our application so that it runs in our Docker containers.
Linking containers requires container discovery. There are a few ways to do this, but we will use the simplest. Docker networking and container discovery is an entire subject in itself, and outside the scope of this module.
We are going to undertake the following steps:
- Create a self-contained JAR for our project - this will include any external libraries.
- Add a network bridge to docker.
- Update our code files, Dockerfile, and MongoDB instance.
- Update GitHub Actions build file.
So far we have not been doing good practice. For Java, JAR (Java ARchive) files should be deployed and not individual code files as we have been doing. The advantage of a JAR file is it can contain library dependencies, such as the MongoDB one we have added. Maven can build this for us automatically.
First we must update our pom.xml
file. Add the following below the dependencies
section:
<properties>
<maven.compiler.source>10</maven.compiler.source>
<maven.compiler.target>10</maven.compiler.target>
</properties>
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-assembly-plugin</artifactId>
<version>3.3.0</version>
<configuration>
<archive>
<manifest>
<mainClass>com.napier.sem.App</mainClass>
</manifest>
</archive>
<descriptorRefs>
<descriptorRef>jar-with-dependencies</descriptorRef>
</descriptorRefs>
</configuration>
<executions>
<execution>
<id>make-assembly</id>
<phase>package</phase>
<goals>
<goal>single</goal>
</goals>
</execution>
</executions>
</plugin>
</plugins>
</build>
We have added two new sections:
properties
- here we are telling Maven to produce Java 8 code (1.8).build
- there is quite a bit going on here. You can happily reuse the code though:- We are defining how Maven assembles the JAR file.
- We are telling Maven which class to run when the JAR is executed (
mainClass
). - We are telling Maven to build the
jar-with-dependencies
- in other words pull in the MongoDB code.
First rebuild your project so that everything is up to date: Build then Build Project. We can now ask Maven to package up our application. In IntelliJ open the Maven Panel on the right hand side:
Open the Lifecycle collapsed menu, and select package and click the green triangle in the Maven panel to start the package process. This will take a few seconds as your code and the external JAR libraries are combined into a single JAR. You will see this in the target folder in the Project Structure as seMethods-0.1.0.1-jar-with-dependencies.jar
. So we have successfully built our project into a single JAR for deployment. Time to push to GitHub.
- Add files to commit.
- Create commit.
- Push to GitHub.
In Docker, containers can discover each other by name if they are on the same Docker network bridge which is not the default one. Therefore, we need to create a new bridge for our applications to talk on. This is actually quite easy. Run the following command from the command line:
docker network create --driver bridge se-methods
This will have created a new network called se-methods
. We can use this network to connect our main application to our MongoDB server. Let us do this now.
First we need to stop our current MongoDB server. In IntelliJ you should be able to see this in the Services panel under Containers. To stop it, select the container and click the red stop button on the left:
Once stopped, right-click on the container, and select Delete Container and then Yes in the prompt.
We need to create a new MongoDB server that uses our network infrastructure, and we also want to define the name of the server. We do this by creating a new container from mongo:latest
in IntelliJ using the following parameters:
Click Run and the container will start. Next we need to update our main application so it can talk to this MongoDB server. The only line that needs updating is the MongoClient
creation one:
// Connect to MongoDB
MongoClient mongoClient = new MongoClient("mongo-dbserver");
We are now explicitly connecting to the server called mongo-dbserver
, which is the name we gave to our MongoDB container. To test this, we need to update our Dockerfile:
FROM openjdk:latest
COPY ./target/seMethods-0.1.0.1-jar-with-dependencies.jar /tmp
WORKDIR /tmp
ENTRYPOINT ["java", "-jar", "seMethods-0.1.0.1-jar-with-dependencies.jar"]
We have changed what we are copying to the JAR file that has been created. We are also changing our entry point to execute this JAR. We need to first update our jar file, rebuild the docker image and restart the container.
- Delete the target directory
- Run Maven Package to recreate the jar file
- Build the Docker Image (You can do this by clicking the green triangles in the Dockerfile and selecting Build Image for Dockerfile. )
- The image will be created and added to the list of images in IntelliJ's Docker Panel, near the bottom with an
sha256
name. If you have more than one of these because of previous builds, delete all thesha256
images and rebuild to have only one.
To test our new image, select Create Container with it selected and use the following properties:
Click Run and our container will start our application which will connect to the MongoDB server and exit. If successful you will see output on the container logs
Time for an update to GitHub.
- Add files to commit.
- Create commit.
- Push to GitHub.
Now to put this into our GitHub Actions file:
name: A workflow for my Hello World App
on: push
jobs:
build:
name: Hello world action
runs-on: ubuntu-20.04
steps:
- name: Checkout
uses: actions/checkout@v2
- name: Set up JDK 11
uses: actions/setup-java@v2
with:
java-version: '11'
distribution: 'adopt'
- name: Setup network
run: |
docker network create --driver bridge se-methods
docker pull mongo
docker run -d --name mongo-dbserver --network se-methods mongo
- name: Build with Maven
run: mvn package
- name: Build
run: docker build -t se_methods .
- name: Run image
run: docker run --network se-methods --name semcontainer se_methods
- name: view logs
run: docker logs semcontainer
We have a few more Docker commands but these are just the ones we added via IntelliJ or otherwise. Now push this to GitHub:
- Add files to commit.
- Create commit.
- Push to GitHub.
Check with GitHub Actions and ensure that not only is the project building but that it successfully runs in Docker. You will need to open some of the code folds to verify.
It's time to merge our feature back into the develop
branch. Before doing this, we need to check that no changes have occurred in develop
. Although we know there hasn't been, we need to get into the habit of managing the workflow.
To merge
any changes in the develop
branch onto our feature branch, select Git, then Branches.... This will open up the Branches Popup:
Select the orign/develop
branch (this is the one in GitHub) and then select Merge into Current. You might spot a little pop-up at the bottom stating Already Up-to-date which means we can proceed and merge our feature back into develop
.
Now we just need to switch back to the develop
branch. Open the Branches Popup again, and select the develop
branch in Local Branches and select Checkout:
And now merge
the feature/mongo-integration
branch in the Local Branches into the current branch as before. This will complete the feature. We just need to push
this change into GitHub. It is just a push
as all the feature changes have been added to develop
.
Now we need to create a release. First, create a new release branch. Follow the instructions as before.
We are going to call this release 0.1.0.2
(0.1-alpha-2
). You will need to change the following files to reflect this:
pom.xml
- theversion
tag.Dockerfile
- theCOPY
andENTRYPOINT
values need updated with the new JAR file name.
Once you've done that, test that everything still works locally. This involves:
- Rebuilding the project.
- Telling Maven to package the project.
- Building the Docker image.
- Running the Docker image.
If everything goes well, push the changes to GitHub:
- Add files to commit.
- Create commit.
- Push to GitHub.
Now we need to merge
this release back into master
. The steps you need to take are:
- Checkout
master
. - Merge
release
ontomaster
. - Push to GitHub.
Git commits can also be tagged. To do this in IntelliJ, select Git then New Tag to open the Tag window:
Use the name provided, and click Create Tag. Now we just need to push
the tag to GitHub. Select push, but this time ensure the Push Tags checkbox is ticked as indicated:
Click Push and your tag will be added to GitHub.
Now to create a release on GitHub. Go to the GitHub page for your project and select the Releases tab to open the following window:
Click Create new release to start entering the release details:
The details we want are below:
Also make sure the checkbox This is a pre-release is ticked. Then click Publish release. Your release details will then be presented:
And if you go back to the main GitHub repository page you will find that your badges have been updated:
All you need to do now is merge the release
branch back into develop
:
- Checkout
develop
. - Merge
release
intodevelop
. - Push
develop
.
And we are done. We have done a lot, but still not much code. We have built our development pipeline, defined our workflow, and integrated a database along the way. Not bad work.
Before stepping away from your machine there are some things you should do:
- Stop any running Docker containers.
- Delete any unneeded containers.
- Delete any unneeded images.
- Ensure any changes have been pushed.
With that done, you can happily walk away from the machine.
This is our current workflow. This is an important set of steps so document them:
- Pull the latest
develop
branch. - Start a new feature branch.
- Once feature is finished, create JAR file.
- Update and test Docker configuration with GitHub Actions.
- Update feature branch with
develop
to ensure feature is up-to-date. - Check feature branch still works.
- Merge feature branch into
develop
. - Repeat 2-7 until release is ready.
- Merge
develop
branch intorelease
and create release. - Merge
release
intomaster
anddevelop
.