Dagger CI workflow using Java SDK

Dagger CI workflow with Java SDK
In this blog have details how we can create Dagger module and functions using Dagger Java SDK on a simple Java Spring boot project. With Dagger CLI we can run the CI process in our local machine. With the Dagger CLI new modules can be created or use the modules built by community from DaggerVerse. In this example, we initialize the Dagger module with functions to build environment on a container, build jar, test the build and publish the build. With the Dagger CLI we can run the functions in local machine. There is an option to run the Dagger module in Github actions or Jenkins.
What is Dagger?
Dagger is an opensource code that enables to define CI/CD workflows as code. Dagger SDK is available in different languages like Go, Python, Typescript, etc. to develop code. In local the Dagger CLI uses the Dagger engine running in Docker to run the functions. For Dagger architecture refer the dagger.io documentation.
Key terminologies
Module
- Module is collection of Dagger functions.Functions
- Functions is fundamental unit of computing written in regular code in preferred language using available Dagger SDK.
Prerequisites:
- Docker desktop
- Dagger CLI installed. Note, when installing in Windows use Powershell 7+. Dagger CLI will use Docker to run the dagger engine.
Initialize dagger in the project
- We have a simple SpringBoot Java project and first initialize the dagger by executing below command. This will create
dagger.json
file updates dagger engine version and few other metadata.
dagger init
Dagger module creation
Parent module
We can create modules using Dagger CLI, and implement our functions to perform CI/CD operations. To Dagger module in the project we use Java SDK. By executing below command we will create a dagger module. This will create a .dagger
folder and simple java class as it uses the maven code generation. Since the name of the java project was sample-app
, the java code generated under .dagger/src/main/java/io/dagger/modules/sample-app/SampleApp.java
dagger develop --sdk=java
Note, we will be creating a module under the .dagger
module, called backend
. This will be helpful to manage if the same repo has both backend and frontend code and CI process builds them differently. We could have created the functions directly in .dagger/src/main/java/io/dagger/modules/sample-app/SampleApp.java
class in .dagger
folder but we will look how to create modules under the .dagger
and re-use the function from child module in parent module.
Backend module (child module)
Lets create the backend module, the process to create other modules under the parent module is the same.
Create a backend
folder under the .dagger
directory, and issue below command to initialize the dagger module from backend
directory.
cd .dagger/backend
dagger init --name=backend
dagger develop --sdk=java --source .
The project structure will look like below after creating the backend modules. We could see that the dagger.json
at the root of the SpringBoot project folder structure and the modules created under the the .dagger
folder. Note, when we use Dagger CLI to run the functions, the maven code generator will create target folders under the .dagger
and backend
which is not included below.
sample-app/
├── .dagger
│ ├── .gitattributes
│ ├── .gitignore
│ ├── backend
│ │ ├── .gitattributes
│ │ ├── .gitignore
│ │ ├── LICENSE
│ │ ├── dagger.json
│ │ ├── pom.xml
│ │ └── src
│ │ └── main
│ │ └── java
│ │ └── io
│ │ └── dagger
│ │ └── modules
│ │ └── backend
│ │ ├── Backend.java
│ │ └── package-info.java
│ ├── pom.xml
│ └── src
│ └── main
│ └── java
│ └── io
│ └── dagger
│ └── modules
│ └── sampleapp
│ ├── SampleApp.java
│ └── package-info.java
├── dagger.json
├── pom.xml
└── src
├── main
│ ├── java
│ │ └── com
│ │ └── demo
│ │ └── sample_app
│ │ ├── HealthController.java
│ │ └── SampleAppApplication.java
│ └── resources
│ ├── application.properties
│ ├── static
│ └── templates
└── test
└── java
└── com
└── demo
└── sample_app
└── SampleAppApplicationTests.java
Till now we have created two modules, parent module under the .dagger
folder. Another module under backend
. Since Dagger CLI generated the module folder structure now we can create functions using the Dagger library. The functions below are self explanatory since it is similar to Dockerfile definition.
Below is the backend module with simple functions will helps to setup environment on the container, build jar artifact, test, run and publish.
With the functions added, Dagger CLI can list the functions defined in this module. The function arguments name will be used as parameters when we call the function using Dagger CLI. Next we will see how to use Dagger CLI to list the functions.
package io.dagger.modules.backend;
import io.dagger.client.CacheVolume;
import io.dagger.client.Container;
import io.dagger.client.DaggerQueryException;
import io.dagger.client.Directory;
import io.dagger.client.File;
import io.dagger.client.Service;
import io.dagger.module.AbstractModule;
import io.dagger.module.annotation.Function;
import io.dagger.module.annotation.Object;
import java.util.List;
import java.util.concurrent.ExecutionException;
/** Backend main object */
@Object
public class Backend extends AbstractModule {
/** Create environment for build **/
@Function
public Container buildEnv(Directory source)
throws InterruptedException, ExecutionException, DaggerQueryException {
CacheVolume mavenCache = dag.cacheVolume("maven-cache");
return dag.container()
.from("maven:3.9.9-amazoncorretto-21")
.withMountedCache("root/.m2",mavenCache)
.withMountedDirectory("/app/backend",source
.withoutDirectory(".idea")
.withoutDirectory(".dagger"))
.withWorkdir("/app/backend");
}
/** Build the jar artifacts*/
@Function
public File build(Directory source)
throws InterruptedException, ExecutionException, DaggerQueryException {
return buildEnv(source)
.withExec(List.of("mvn","-DskipTests","clean","install"))
.file("target/sample-app-0.0.1-SNAPSHOT.jar");
}
/** Run test */
@Function
public String test(Directory source)
throws InterruptedException, ExecutionException, DaggerQueryException {
return buildEnv(source)
.withExec(List.of("mvn","test"))
.stdout();
}
/** Publish the artifact */
@Function
public String publish(Directory source)
throws InterruptedException, ExecutionException, DaggerQueryException {
return dag.container()
.from("openjdk:21-jdk-slim") // we can use the buildEnv function as well
.withFile("/app/sample-app.jar",build(source)) //build function is called which will create the jar file and return it
.withExec(List.of("chmod","777","/app/sample-app.jar"))
.withEntrypoint(List.of("java","-jar","/app/sample-app.jar"))
.withExposedPort(8080)
// publishing to ttl.sh repo
.publish("ttl.sh/simple-dagger-app-%d".formatted((int) (Math.random() * 10000000)))
;
}
/** Runs as service*/
@Function
public Service run(Directory source)
throws InterruptedException, ExecutionException, DaggerQueryException {
return dag.container()
.from("openjdk:21-jdk-slim")
.withFile("/app/sample-app.jar",build(source))
.withExec(List.of("chmod","777","/app/sample-app.jar"))
.withEntrypoint(List.of("java","-jar","/app/sample-app.jar"))
.withExposedPort(8080)
.asService()
.withHostname("localhost");
}
- Below is the Dagger CLI command that lists the backend module functions. The function name
buildEnv
is displayed asbuild-env
by Dagger CLI.
dagger -m .dagger/backend functions
- Output will look like below
dagger -m .dagger/backend functions
✔ connect 0.7s
✔ load module 14.9s
Name Description
build Build jar artifacts
build-env Create environment for build
publish Publish the artifact
run Runs as service
test Run test
To view the arguments info of the functions we can call the function and pass the --help
like below
dagger -m .dagger/backend call build --help
- Output of the build function displayed by Dagger CLI. Note, the java comments above the function method will be used as function description by Dagger CLI. The method parameter name will be used as argument name.
✔ connect 0.6s
✔ load module 4.6s
✔ parsing command line arguments 0.0s
Build jar artifacts
USAGE
dagger call build [arguments] <function>
FUNCTIONS
contents Retrieves the contents of the file.
digest Return the file's digest. The format of the digest is not guaranteed to be stable between releases of Dagger. It is guaranteed
to be stable between invocations of the same Dagger engine.
export Writes the file to a file path on the host.
name Retrieves the name of the file.
size Retrieves the size of the file, in bytes.
sync Force evaluation in the engine.
with-name Retrieves this file with its name set to the given name.
with-timestamps Retrieves this file with its created/modified timestamps set to the given time.
ARGUMENTS
--source Directory [required]
Use "dagger call build [command] --help" for more information about a command.
Invoke build-env function from the backend module
Now lets call the buildEnv
function from backend
module using Dagger CLI. Since buildEnv
function returns a Container we can use terminal
option so Dagger CLI will run the command and will exec into the container to working directory path with the copied source code.
dagger -m .dagger/backend call build-env --source . terminal
Install and reuse the backend module functions in parent module
Till now we created function in the backend module and executed using Dagger CLI with -m
option. During development for IDE to use the backend
module from .dagger
parent module we need to install the backend module.
To install the backend
module issue below command from the root directiry of Java project root in this case sample-app
. This command will update the dagger.json
at the sample-app
project level. Along with install we need to issue another command perform code gen with the installed module.
dagger install .dagger/backend
After installing the backend
module with above command, from the root directory of java project sample-app
issue execute below command which will run code generate and update the parent module. Now in IDE we will be able to use the function from the backend
module.
dagger develop
The backend
functions can now be used in .dagger/src/main/java/io/dagger/modules/sample-app/SampleApp.java
class to build backend, if we have created frontend functions we can reuse those function to build both codes.
- The
.dagger/src/main/java/io/dagger/modules/sample-app/SampleApp.java
includes function and reuse the backend module functions.
package io.dagger.modules.sampleapp;
import io.dagger.client.Container;
import io.dagger.client.DaggerQueryException;
import io.dagger.client.Directory;
import io.dagger.client.Service;
import io.dagger.module.AbstractModule;
import io.dagger.module.annotation.Function;
import io.dagger.module.annotation.Object;
import java.util.List;
import java.util.concurrent.ExecutionException;
/** SampleApp main object */
@Object
public class SampleApp extends AbstractModule {
/** Publish image to repo */
@Function
public String publish(Directory source)
throws InterruptedException, ExecutionException, DaggerQueryException {
return dag.backend().publish(source);
}
/** Run the app as service */
@Function
public Service run(Directory source){
return dag.backend().run(source);
}
}
Since the parent module is updated, by issuing below command we could see those functions listed like in the output of the command below.
dagger functions
- Output of above command with list of function from the main module looks like below
dagger functions
✔ connect 0.7s
✔ load module 13.9s
Name Description
publish publish image of the apps
run Run the application as service
Call the run function from main module using Dagger CLI
Since the main module is updated now, we can use Dagger CLI just to call the functions instead of referring the modules with -m
option.
dagger call run --source . up --ports 8081:8080
Since the run command returns Service, the --ports 8081:8080
option in Dagger CLI will create a network tunneling and the application can be accessed using http://localhost:8081/api/check
from another terminal. Below is the output of the accessing the endpoint of application running in dagger engine.
$ curl localhost:8081/api/check && echo
[{"check": "ok"},{"time": "2025-03-09 17:31:53"}]
Call the publish function from main module using Dagger CLI
dagger call publish --source .
Using custom dagger engine
In enterprise private repository will be used and behind proxy in this case we might require customization of the dagger images with enterprise certs, security, etc. Dagger CLI can be configured to use the custom dagger engine image from docker by configuring the image name in environment variable _EXPERIMENTAL_DAGGER_RUNNER_HOST
.
Refer Dagger.io documentation for more details.
Source code
The code can be found in github repo. Note, the .dagger
folder is renamed to dagger since it is treated as hidden folder in git. This source currently can't be cloned and used directly, just for reference only.
Subscribe to my newsletter
Read articles from Thirumurthi S directly inside your inbox. Subscribe to the newsletter, and don't miss out.
Written by
