Using Dagger SDKs
Dagger SDKs make it easy to call the Dagger API from your favorite programming language, by developing Dagger Functions or custom applications.
A Dagger SDK provides two components:
- A client library to call the Dagger API from your code
- Tooling to extend the Dagger API with your own Dagger Functions (bundled in a Dagger module)
The Dagger API uses GraphQL as its low-level language-agnostic framework, and can also be accessed using any standard GraphQL client. However, you do not need to know GraphQL to call the Dagger API; the translation to underlying GraphQL API calls is handled internally by the Dagger SDKs.
Official Dagger SDKs are currently available for Go, TypeScript and Python. There are also experimental and community SDKs contributed by the Dagger community.
Dagger SDKs can also be used to build custom applications that use the Dagger API. These applications can be standalone or integrated into existing systems, allowing you to leverage Dagger's capabilities in your own software solutions.
Dagger Functions​
The recommended, and most common way, to interact with the Dagger API is through Dagger Functions. Dagger Functions are just regular code, written in your usual language using a type-safe Dagger SDK.
Dagger Functions are packaged, shared and reused using Dagger modules. A new Dagger module is initialized by calling dagger init. This creates a new dagger.json configuration file in the current working directory, together with sample Dagger Function source code. The configuration file will default the name of the module to the current directory name, unless an alternative is specified with the --name argument.
Once a module is initialized, dagger develop --sdk=... sets up or updates all the resources needed to develop the module locally using a Dagger SDK. By default, the module source code will be stored in the current working directory, unless an alternative is specified with the --source argument.
Here is an example of initializing a Dagger module:
- Go
- Python
- TypeScript
- PHP
- Java
dagger init --name=my-module
dagger develop --sdk=go
dagger init --name=my-module
dagger develop --sdk=python
dagger init --name=my-module
dagger develop --sdk=typescript
dagger init --name=my-module
dagger develop --sdk=php
dagger init --name=my-module
dagger develop --sdk=java
Running dagger develop regenerates the module's code based on dependencies, the current state of the module, and the current Dagger API version. This can result in unexpected results if there are significant changes between the previous and latest installed Dagger API versions. Always refer to the changelog for a complete list of changes (including breaking changes) in each Dagger release before running dagger develop, or use the --compat=skip option to bypass updating the Dagger API version.
The default template from dagger develop creates the following structure:
- Go
- Python
- TypeScript
- PHP
- Java
.
├── LICENSE
├── dagger.gen.go
├── dagger.json
├── go.mod
├── go.sum
├── internal
│ ├── dagger
│ ├── querybuilder
│ └── telemetry
└── main.go
In this structure:
dagger.jsonis the Dagger module configuration file.go.mod/go.summanage the Go module and its dependencies.main.gois where your Dagger module code goes. It contains sample code to help you get started.internalcontains automatically-generated types and helpers needed to configure and run the module:daggercontains definitions for the Dagger API that's tied to the currently running Dagger Engine container.querybuilderhas utilities for building GraphQL queries (used internally by thedaggerpackage).telemetryhas utilities for sending Dagger Engine telemetry.
.
├── LICENSE
├── pyproject.toml
├── uv.lock
├── sdk
├── src
│ └── my_module
│ ├── __init__.py
│ └── main.py
└── dagger.json
In this structure:
dagger.jsonis the Dagger module configuration file.pyproject.tomlmanages the Python project configuration.uv.lockmanages the module's pinned dependencies.src/my_module/is where your Dagger module code goes. It contains sample code to help you get started.sdk/contains the vendored Python SDK client library.
This structure hosts a Python import package, with a name derived from the project name (in pyproject.toml), inside a src directory. This follows a Python convention that requires a project to be installed in order to run its code. This convention prevents accidental usage of development code since the Python interpreter includes the current working directory as the first item on the import path (more information is available in this blog post on Python packaging).
.
├── LICENSE
├── dagger.json
├── package.json
├── sdk
├── src
│ └── index.ts
├── tsconfig.json
└── yarn.lock
In this structure:
dagger.jsonis the Dagger module configuration file.package.jsonmanages the module dependencies.src/is where your Dagger module code goes. It contains sample code to help you get started.sdk/contains the TypeScript SDK.
.
├── LICENSE
├── README.md
├── composer.json
├── composer.lock
├── entrypoint.php
├── sdk
├── src
│ └── MyModule.php
└── vendor
In this structure:
dagger.jsonis the Dagger module configuration file.composer.jsonmanages the module dependencies.src/is where your Dagger module code goes. It contains sample code to help you get started.sdk/contains the PHP SDK.
.
├── LICENSE
├── dagger.json
├── pom.xml
├── src
│  └── main
│  └── java
│  └── io
│  └── dagger
│  └── modules
│  └── mymodule
│  ├── MyModule.java
│  └── package-info.java
└── target
9 directories, 5 files
In this structure:
dagger.jsonis the Dagger module configuration file.pom.xmlmanages the module dependencies.src/main/java/io/dagger/modules/mymodule/is where your Dagger module code goes. It contains sample code to help you get started.targetcontains the generated Java source classes.
While you can use the utilities defined in the automatically-generated code above, you cannot edit these files. Even if you edit them locally, any changes will not be persisted when you run the module.
You can now write Dagger Functions using the selected Dagger SDK. Here is an example, which calls a remote API method and returns the result:
- Go
- Python
- TypeScript
- PHP
- Java
package main
import (
"context"
)
type MyModule struct{}
func (m *MyModule) GetUser(ctx context.Context) (string, error) {
return dag.Container().
From("alpine:latest").
WithExec([]string{"apk", "add", "curl"}).
WithExec([]string{"apk", "add", "jq"}).
WithExec([]string{"sh", "-c", "curl https://randomuser.me/api/ | jq .results[0].name"}).
Stdout(ctx)
}
This Dagger Function includes the context as input and error as return in its signature.
from dagger import dag, function, object_type
@object_type
class MyModule:
@function
async def get_user(self) -> str:
return await (
dag.container()
.from_("alpine:latest")
.with_exec(["apk", "add", "curl"])
.with_exec(["apk", "add", "jq"])
.with_exec(
["sh", "-c", "curl https://randomuser.me/api/ | jq .results[0].name"]
)
.stdout()
)
Dagger Functions are implemented as @dagger.function decorated methods, of a @dagger.object_type decorated class.
It's possible for a module to implement multiple classes (object types), but the first one needs to have a name that matches the module's name, in PascalCase. This object is sometimes referred to as the "main object".
For example, for a module initialized with dagger init --name=my-module,
the main object needs to be named MyModule.
import { dag, object, func } from "@dagger.io/dagger"
@object()
class MyModule {
@func()
async getUser(): Promise<string> {
return await dag
.container()
.from("alpine:latest")
.withExec(["apk", "add", "curl"])
.withExec(["apk", "add", "jq"])
.withExec([
"sh",
"-c",
"curl https://randomuser.me/api/ | jq .results[0].name",
])
.stdout()
}
}
<?php
declare(strict_types=1);
namespace DaggerModule;
use Dagger\Attribute\{DaggerObject, DaggerFunction};
use function Dagger\dag;
#[DaggerObject]
class MyModule
{
#[DaggerFunction]
public function getUser(): string
{
return dag()
->container()
->from('alpine:latest')
->withExec(['apk', 'add', 'curl'])
->withExec(['apk', 'add', 'jq'])
->withExec([
'sh',
'-c',
'curl https://randomuser.me/api/ | jq .results[0].name',
])
->stdout();
}
}
package io.dagger.modules.mymodule;
import static io.dagger.client.Dagger.dag;
import io.dagger.client.Container;
import io.dagger.client.Directory;
import io.dagger.client.exception.DaggerQueryException;
import io.dagger.module.annotation.Function;
import io.dagger.module.annotation.Object;
import java.util.List;
import java.util.concurrent.ExecutionException;
/** MyModule main object */
@Object
public class MyModule {
@Function
public String getUser() throws ExecutionException, DaggerQueryException, InterruptedException {
return dag().container()
.from("alpine:latest")
.withExec(List.of("apk", "add", "curl"))
.withExec(List.of("apk", "add", "jq"))
.withExec(List.of("sh", "-c", "curl https://randomuser.me/api/ | jq .results[0].name"))
.stdout();
}
}
You can try this Dagger Function by copying it into the default template generated by dagger init, but remember that you must update the module name in the code samples above to match the name used when your module was first initialized.
In simple terms, this Dagger Function:
- initializes a new container from an
alpinebase image. - executes the
apk add ...command in the container to add thecurlandjqutilities. - uses the
curlutility to send an HTTP request to the URLhttps://randomuser.me/api/and parses the response usingjq. - retrieves and returns the output stream of the last executed command as a string.
Every Dagger Function has access to the dag client, which is a pre-initialized Dagger API client. This client contains all the core types (like Container, Directory, etc.), as well as bindings to any dependencies your Dagger module has declared.
Here is an example call for this Dagger Function:
- System shell
- Dagger Shell
- Dagger CLI
dagger -c 'get-user'
get-user
dagger call get-user
Here's what you should see:
{
"title": "Mrs",
"first": "Beatrice",
"last": "Lavigne"
}
Dagger Functions execute within containers spawned by the Dagger Engine. This "sandboxing" serves a few important purposes:
- Reproducibility: Executing in a well-defined and well-controlled container ensures that a Dagger Function runs the same way every time it is invoked. It also guards against creating "hidden dependencies" on ambient properties of the execution environment that could change at any moment.
- Caching: A reproducible containerized environment makes it possible to cache the result of Dagger Function execution, which in turn allows Dagger to automatically speed up operations.
- Security: Even when running third-party Dagger Functions sourced from a Git repository, those Dagger Functions will not have default access to your host environment (host files, directories, environment variables, etc.). Access to these host resources can only be granted by explicitly passing them as argument values to the Dagger Function.