Skip to main content

TypeScript SDK

The TypeScript SDK lets you write Dagger modules in TypeScript. You define an @object() class with @func() methods, and Dagger exposes those methods as API functions that anyone can call from the CLI, from another module, or over the API. Inside your functions you orchestrate containers, files, services, and secrets through a generated, fully-typed client.

This page assumes you already understand the Dagger module model. If you don't, read these first — this page stays light on platform concepts and links back where relevant:

important

Module development tooling lives in the SDK module, not the dagger CLI: the TypeScript SDK is itself a Dagger modulegithub.com/dagger/typescript-sdk. You install it into your workspace once, then create and maintain TypeScript modules by calling its functions by name.

Create a module

note

Run these commands from inside a Git repository — that's where the new module is created.

Install the TypeScript SDK into your workspace, then scaffold a new module with its init function:

dagger install github.com/dagger/typescript-sdk
dagger call typescript-sdk init --name=my-module

init returns a changeset describing the files to create — which Dagger shows you to review before anything is written to disk.

init arguments (from dagger call typescript-sdk init --help):

ArgumentRequiredDescription
--nameyesThe module name.
--pathnoWhere to create the module. Defaults to the nearest .dagger/modules/<name> visible from the workspace path.
--runtimenoOne of NODE, BUN, DENO. Defaults to Node.
--templatenoMaterialize files from templates/<template>. Empty uses the SDK's default template.
--ignore-generatednoAdd generated SDK paths to .gitignore instead of checking them in.

By default the module is created under the nearest .dagger directory:

<nearest .dagger>/modules/my-module

Pass --path to choose a different location. The target path must not already contain a Dagger module.

File layout

Once created and generated, a Node module looks like this:

my-module/
├── dagger-module.toml # module config
├── package.json # npm/yarn/pnpm package definition
├── tsconfig.json # TypeScript config with the @dagger.io/dagger path alias
├── sdk/ # generated TypeScript bindings (the typed client)
└── src/
└── index.ts # your module code
note

init writes only your source (dagger-module.toml, package.json, tsconfig.json, src/). The generated sdk/ client is produced by the first binding regeneration — committed by default — not by init itself.

dagger-module.toml declares the SDK source:

dagger-module.toml
name = "my-module"
engineVersion = "v1.0.0-beta.3"

[runtime]
source = "typescript"

tsconfig.json wires the @dagger.io/dagger import to the local generated SDK so editors resolve types without a separate install:

tsconfig.json
{
"compilerOptions": {
"target": "ES2022",
"moduleResolution": "Node",
"experimentalDecorators": true,
"strict": true,
"skipLibCheck": true,
"paths": {
"@dagger.io/dagger": ["./sdk/index.ts"],
"@dagger.io/dagger/telemetry": ["./sdk/telemetry.ts"]
}
}
}

package.json declares the package as an ES module:

package.json
{
"type": "module"
}

Define objects and functions

A module is a class decorated with @object(). Methods decorated with @func() become Dagger API functions. The first @object() class is the module's main object; its name in the API is derived from the module name.

src/index.ts
import { dag, Container, Directory, object, func } from "@dagger.io/dagger"

@object()
class MyModule {
/**
* Build and publish a container image, returning the published ref
*/
@func()
async build(src: Directory): Promise<string> {
const builder = dag
.container()
.from("golang:latest")
.withDirectory("/src", src)
.withWorkdir("/src")
.withEnvVariable("CGO_ENABLED", "0")
.withExec(["go", "build", "-o", "myapp"])

const prodImage = dag
.container()
.from("alpine")
.withFile("/bin/myapp", builder.file("/src/myapp"))
.withEntrypoint(["/bin/myapp"])

return await prodImage.publish("ttl.sh/myapp:latest")
}
}

Call it like any other module:

dagger -m ./my-module call build --src=.

Key points:

  • dag is the generated Dagger client. Every core type (Container, Directory, File, Service, Secret, …) is imported from @dagger.io/dagger.
  • The Dagger API is lazy and immutable: each builder method returns a new value and nothing executes until you await a leaf operation (publish, stdout, sync, entries, …). You can return an unresolved Container/Directory and the caller decides when to evaluate it.
  • A @func() may be async and return a Promise<T>, or return a lazy Dagger type directly.

The constructor

A class constructor becomes the module's constructor — its parameters become arguments that callers (or workspace config) can set when they reference the module. Store them as fields so other functions can use them.

import { dag, Directory, object, func } from "@dagger.io/dagger"

@object()
class MyModule {
source: Directory
nodeVersion: string

constructor(source: Directory, nodeVersion = "20") {
this.source = source
this.nodeVersion = nodeVersion
}

@func()
build(): Container {
return dag
.container()
.from(`node:${this.nodeVersion}`)
.withDirectory("/app", this.source)
.withWorkdir("/app")
.withExec(["npm", "ci"])
.withExec(["npm", "run", "build"])
}
}

A field decorated with @func() is exposed as a readable API field; an undecorated field is private internal state.

Workspace inputs

To read the user's project files, give your @object() constructor a Workspace parameter. Dagger auto-populates it from the current workspace — the caller passes nothing, and nothing is uploaded up front. The module then reads project content lazily, on demand, so only the paths you actually touch are loaded (and become part of the cache key).

import { dag, Container, Directory, Workspace, object, func } from "@dagger.io/dagger"

@object()
class MyModule {
source: Directory

// ws is auto-populated from the current workspace; the caller passes nothing.
constructor(ws: Workspace) {
// Read the workspace root as the module's source directory.
this.source = ws.directory("/")
}

@func()
build(): Container {
return dag
.container()
.from("node:20")
.withDirectory("/app", this.source)
.withWorkdir("/app")
.withExec(["npm", "ci"])
.withExec(["npm", "run", "build"])
}
}
note

The @object() constructor declares the Workspace as a plain typed parameter — no decorator — that Dagger fills from the current workspace; the caller passes nothing. The Workspace type is imported from @dagger.io/dagger.

A Workspace exposes lazy accessors for project content:

  • ws.directory(path, opts?) — return a Directory for path. Takes an options object with exclude, include, and gitignore. Use exclude aggressively to keep the cache key tight:

    this.source = ws.directory("/", { exclude: ["node_modules", ".git", "dist"] })
  • ws.file(path) — return a File for path.

    const pkg = ws.file("package.json")
  • ws.findUp(name, opts?) — walk up from a start path (opts.from) within the workspace looking for name; returns the absolute workspace path, or null if not found. The search stops at the workspace root.

    const modRoot = await ws.findUp("package.json")

Path resolution: relative paths (e.g. "src", "package.json") resolve from the workspace cwd; absolute paths (e.g. "/src", "/go.mod") resolve from the workspace root.

Because these accessors are lazy, reading a Directory or File from the workspace does not upload anything until a downstream operation actually evaluates it. Pull only the paths you need, with tight exclude patterns, to minimize cache invalidation.

Arguments and return values

Function parameters become Dagger function arguments. TypeScript types map onto the Dagger type system (see Type System):

TypeScriptDagger APINotes
stringString
numberInt / Float
booleanBoolean
string[], T[]list
Directory, File, Container, Service, Secret, Port, …core typesimported from @dagger.io/dagger
void / Promise<void>Voidcommon for checks and side-effecting functions
a custom @object() classobjectfor returning multiple related values
a TypeScript enumenumrestricts to a set of values

Documenting arguments and functions

Use JSDoc comments. The comment above a method is the function description; the comment above a parameter is the argument description. Both appear in dagger functions and dagger call --help.

@object()
class MyModule {
/**
* Query the GitHub API
*/
@func()
async githubApi(
/**
* GitHub API token
*/
token: Secret,
): Promise<string> {
return await dag
.container()
.from("alpine:3.17")
.withSecretVariable("GITHUB_API_TOKEN", token)
.withExec(["apk", "add", "curl"])
.withExec([
"sh",
"-c",
`curl "https://api.github.com/repos/dagger/dagger/issues" --header "Authorization: Bearer $GITHUB_API_TOKEN"`,
])
.stdout()
}
}

Defaults and optional arguments

A TypeScript default value makes the argument optional:

@func()
greet(name = "world"): string {
return `Hello, ${name}!`
}
dagger call greet            # Hello, world!
dagger call greet --name=Sam # Hello, Sam!

Nullability

An optional parameter (name?: string) maps to a nullable argument that may be omitted. A required parameter without a default must always be provided.

@func()
maybeGreet(name?: string): string {
return name ? `Hello, ${name}!` : "Hello!"
}

Enums

A TypeScript enum restricts an argument to a fixed set of values. Document members with JSDoc.

import { func, object } from "@dagger.io/dagger"

/**
* Severity levels for a scan
*/
export enum Severity {
Low = "LOW",
Medium = "MEDIUM",
High = "HIGH",
Critical = "CRITICAL",
}

@object()
export class Security {
@func()
describe(severity: Severity): string {
return `scanning at severity ${severity}`
}
}

Passing an invalid value produces an error listing the valid choices.

Custom object types

To return multiple related values, define another @object() class and expose its fields with @func():

import { dag, File, object, func } from "@dagger.io/dagger"

@object()
class BuildResult {
@func()
binary: File

@func()
version: string

constructor(binary: File, version: string) {
this.binary = binary
this.version = version
}
}

@object()
class MyModule {
@func()
build(): BuildResult {
const bin = dag
.container()
.from("golang:1.22")
.withExec(["sh", "-c", "echo built > /out/app"])
.file("/out/app")
return new BuildResult(bin, "1.0.0")
}
}

Dagger prefixes custom type names in the schema (e.g. MyModuleBuildResult) to avoid collisions when multiple modules are loaded together.

Interfaces

Interfaces let your module accept arbitrary objects from other modules without depending on their concrete types. Declare a plain TypeScript interface whose members are the functions you need — written as fields with function types that return Promises (every call is remote). Any object that provides matching functions can be passed in:

import { dag, object, func } from "@dagger.io/dagger"

export interface Duck {
quack: () => Promise<string>
}

@object()
export class MyModule {
// Accept any object that satisfies the Duck interface.
@func()
async quackIt(duck: Duck): Promise<string> {
return await duck.quack()
}

// A function can also return an interface value.
@func()
getDuck(): Duck {
return dag.mallard()
}
}

Methods that take arguments are typed the same way — withName: (name: string) => Promise<Duck>. Any object whose functions match the interface, including objects returned by other modules, can be passed where the interface is expected.

Core Dagger types

The generated client exposes the full Dagger API. The patterns below are the ones you'll reach for most.

Containers and files

@func()
test(source: Directory): Container {
return dag
.container()
.from("node:20")
.withDirectory("/app", source)
.withWorkdir("/app")
.withMountedCache("/app/node_modules", dag.cacheVolume("node-modules"))
.withExec(["npm", "ci"])
.withExec(["npm", "test"])
}

withMountedCache + dag.cacheVolume("name") give you a persistent, content-addressed cache keyed by name. Every builder operation is layer-cached automatically by its inputs.

Secrets

Accept secrets as the Secret type, never as a plain string. Dagger scrubs secret values from logs, caches, and crash reports.

@func()
async deploy(token: Secret): Promise<string> {
return await dag
.container()
.from("alpine")
.withSecretVariable("DEPLOY_TOKEN", token)
.withExec(["sh", "-c", "deploy --token=$DEPLOY_TOKEN"])
.stdout()
}

Callers supply secrets through providers:

dagger call deploy --token=env:DEPLOY_TOKEN
dagger call deploy --token=file:./token.txt
dagger call deploy --token=cmd:"gh auth token"
dagger call deploy --token=op://vault/item/field

Services

@func()
integrationTest(source: Directory): Container {
const db = dag
.container()
.from("postgres:16")
.withEnvVariable("POSTGRES_PASSWORD", "test")
.withExposedPort(5432)
.asService()

return dag
.container()
.from("node:20")
.withDirectory("/app", source)
.withWorkdir("/app")
.withServiceBinding("db", db)
.withEnvVariable("DATABASE_URL", "postgres://postgres:test@db:5432/test")
.withExec(["npm", "run", "test:integration"])
}

Services are content-addressed: the same definition always resolves to the same hostname, so there are no port conflicts.

Concurrency

Because the API is lazy, you can start independent pipelines and await them together with Promise.all:

@func()
async ci(source: Directory): Promise<void> {
await Promise.all([
this.lint(source).sync(),
this.test(source).sync(),
])
}

Module dependencies

Dependencies are declared in dagger-module.toml and managed with the SDK module's mod deps functions, which return changesets you review and apply.

# List current dependencies
dagger call typescript-sdk mod deps list

# Add a dependency
dagger call typescript-sdk mod deps add --source=github.com/dagger/dagger/modules/go

# Update one dependency, or all remote dependencies
dagger call typescript-sdk mod deps update --help

# Remove a dependency by name or source
dagger call typescript-sdk mod deps remove --help

mod deps add arguments: --source (required, the module reference) and --name (optional, override the local alias).

A module reference is [proto://]host/repo[/subpath][@version], e.g. github.com/shykes/daggerverse/hello@v0.3.0. Local paths (./path/to/module) are also supported. After adding a dependency, dagger-module.toml gains a dependencies entry:

dagger-module.toml
name = "my-module"
engineVersion = "v1.0.0-beta.3"

[runtime]
source = "typescript"

[[dependencies]]
name = "go"
source = "github.com/dagger/dagger/modules/go"

Once a dependency is added and bindings are regenerated, call it through dag. The dependency's functions are accessible by its name, and arguments are passed as an options object:

import { dag, Directory, object, func } from "@dagger.io/dagger"

@object()
class MyModule {
@func()
example(buildSrc: Directory, buildArgs: string[]): Directory {
return dag.go().build({ source: buildSrc, args: buildArgs }).terminal()
}
}

Regenerate bindings

The sdk/ directory holds the generated typed client. Regenerate it whenever you add or update a dependency, or bump the engine version. Use the SDK module's mod generate, which returns a changeset:

dagger call typescript-sdk mod generate

If the generate skip marker is present in the module (or an ancestor), the changeset is empty — see mod skip-generate.

You can also use the top-level CLI shortcut, which runs the configured SDK's generators for the module in the current workspace:

dagger generate

Commit vs. generate

By default, generated SDK files are checked into version control so that consumers and CI don't need to regenerate before building. This is the recommended setup. If you'd rather keep generated files out of git, pass --ignore-generated to init (or configure generation accordingly), which adds the generated paths to .gitignore. In that mode, anyone building the module must regenerate first.

Either way, treat regeneration as a reviewable step: run mod generate, inspect the changeset, and commit the result alongside the change that required it.

Engine version

dagger-module.toml's engineVersion pins the engine your module targets. Manage it with mod engine:

# Show the required version (without leading "v")
dagger call typescript-sdk mod engine required

# Pin to a specific version
dagger call typescript-sdk mod engine require --help

# Pin to the current engine, or the latest stable release
dagger call typescript-sdk mod engine require-current
dagger call typescript-sdk mod engine require-latest

After changing the engine version, regenerate bindings (mod generate) so the client matches.

Checks, generators, directives, and ignore

A useful, reusable module provides at least one first-class function type: a check (run by dagger check), a generator (run by dagger generate), or a service (started by dagger up). Each is a regular @func() annotated with an additional decorator.

Checks

A check is a function that validates something and passes or fails by exit code. Mark it with @check(). dagger check discovers and runs every check in the module. See Checks.

import { dag, Directory, Workspace, object, func, check } from "@dagger.io/dagger"

@object()
class MyModule {
source: Directory

constructor(ws: Workspace) {
this.source = ws.directory("/")
}

/**
* Lint the code
*/
@func()
@check()
async lint(): Promise<void> {
await dag
.container()
.from("node:20")
.withDirectory("/app", this.source)
.withWorkdir("/app")
.withExec(["npm", "ci"])
.withExec(["npm", "run", "lint"])
.sync()
}

/**
* Run unit tests
*/
@func()
@check()
async test(): Promise<void> {
await dag
.container()
.from("node:20")
.withDirectory("/app", this.source)
.withWorkdir("/app")
.withExec(["npm", "ci"])
.withExec(["npm", "test"])
.sync()
}
}
dagger check

A check passes if it returns without throwing; it fails if any withExec returns a non-zero exit code (which surfaces as a thrown error when you sync).

Generators

A generator produces a changeset — a diff between the current source and generated output. Mark it with @generate() and return a Changeset. dagger generate runs every generator and presents the combined changeset for review.

import { dag, Directory, Workspace, Changeset, object, func, generate } from "@dagger.io/dagger"

@object()
class MyModule {
source: Directory

constructor(ws: Workspace) {
this.source = ws.directory("/")
}

/**
* Generate API client code
*/
@func()
@generate()
generateApi(): Changeset {
return dag
.container()
.from("node:20")
.withDirectory("/app", this.source)
.withWorkdir("/app")
.withExec(["npm", "ci"])
.withExec(["npm", "run", "generate:api"])
.directory("/app")
.changes(this.source)
}
}

.changes(this.source) computes the diff against the original source.

This author-written @generate() function is distinct from mod generate (covered in Regenerate bindings), which regenerates the SDK's own typed client in sdk/. Your @generate() functions produce project-specific output (API clients, config, scaffolding) and are run by dagger generate.

Services

A service is a long-running container — a database, web server, or other dependency. Mark a @func() with @up() and return a Service. dagger up starts every service in the module. See Services.

import { Service, dag, object, func, up } from "@dagger.io/dagger"

@object()
class MyModule {
/**
* A web server service
*/
@func()
@up()
web(): Service {
return dag.container().from("nginx:alpine").withExposedPort(80).asService()
}

/**
* A redis service
*/
@func()
@up()
redis(): Service {
return dag.container().from("redis:alpine").withExposedPort(6379).asService()
}
}
dagger up

The @up() decorator goes below @func(), and the function returns a Service built from a container with .asService().

Cache directive

Control function caching with the @func({ cache }) option:

// Persistent caching for 10 minutes (useful for external data)
@func({ cache: "10m" })
async latestRelease(): Promise<string> { /* ... */ }

// Cache only for the current session
@func({ cache: "session" })
async sessionId(): Promise<string> { /* ... */ }

// Never cache — re-execute every call
@func({ cache: "never" })
async currentTime(): Promise<string> { /* ... */ }

A cache miss still benefits from layer caching for operations inside the function. "never" forces the function to re-run but does not disable layer caching for its operations.

Ignore

When you read a directory from the workspace, filter it with the exclude option on ws.directory(path, { exclude }) (see Workspace inputs). Push file access to the leaves: read only the paths you need, with tight exclude patterns, to minimize cache invalidation.

this.source = ws.directory("/", { exclude: ["node_modules", ".git", "dist"] })

Testing TypeScript modules

The primary way to test a module is to call its functions and run its checks:

# Smoke test — does it build?
dagger -m ./my-module call build --src=.

# Run all checks
dagger check

# Run generators and confirm there's no drift
dagger check --generate

For richer behavioral tests, write a separate test module (in TypeScript or any SDK) that adds your module as a dependency and asserts on its outputs. Because services and containers are content-addressed and reproducible, integration tests against ephemeral databases or HTTP services are deterministic.

In CI

Run dagger check from CI to execute every check the module defines:

.github/workflows/ci.yml
name: ci
on: [push, pull_request]
jobs:
check:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- uses: dagger/dagger-for-github@v6
with:
version: "v1.0.0-beta.3"
verb: check

If your module keeps generated sdk/ files in git, also gate on drift by running dagger check --generate.

IDE and package-manager setup

Runtimes and package managers

init --runtime selects the runtime: NODE (default), BUN, or DENO. For Node, the SDK works with npm, yarn, and pnpm — the package manager is detected from your lockfile (package-lock.json, yarn.lock, or pnpm-lock.yaml). Commit the lockfile so builds are reproducible. Bun and Deno manage dependencies through their own native tooling; the module code and decorators are identical across runtimes — only dependency installation and lockfiles differ.

Editor type resolution

Because tsconfig.json maps @dagger.io/dagger to the local ./sdk/index.ts, most editors resolve Dagger types and offer completion without a separate install:

"experimentalDecorators": true,
"paths": {
"@dagger.io/dagger": ["./sdk/index.ts"],
"@dagger.io/dagger/telemetry": ["./sdk/telemetry.ts"]
}

experimentalDecorators is required for @object(), @func(), @check(), @generate(), and @argument() to work.

Source maps

When working across inter-dependent modules, the SDK attaches source maps to generated code (line comments of the form ./path/to/filename:line), so you can click through to a function's declaration in another module. Most editors support this natively or via a plugin such as the Open file plugin for VS Code.

Packaging and release

A TypeScript module is published simply by pushing it to a git repository. There is no separate publish step. Consumers reference it by its git address and version (a tag, branch, or commit):

# In a consumer module's workspace
dagger install github.com/you/my-module@v1.0.0

This adds the dependency to the consumer's dagger-module.toml. Tag releases (e.g. v1.0.0) so consumers can pin a stable version. Keep generated sdk/ files committed so consumers don't have to regenerate before using your module.

Recommended release checklist:

  1. dagger check passes.
  2. dagger check --generate produces no drift.
  3. mod engine required matches the engine you support.
  4. Tag and push: git tag vX.Y.Z && git push --tags.

Troubleshooting

init fails: target already contains a Dagger module. Pass a different --path, or remove the existing dagger-module.toml if you intended to recreate it.

Decorators not recognized / runtime errors about metadata. Ensure experimentalDecorators is true in tsconfig.json and that your @object() class is exported/defined as shown. The first @object() class is the module's main object.

Calling a dependency fails with "unknown function" or missing types. Add the dependency with mod deps add, then regenerate bindings with mod generate and apply the changeset. The generated sdk/ client must include the dependency before dag.<dep>() resolves.

dagger check finds no checks. Confirm the functions are decorated with both @func() and @check() and that the class is the module's @object().

CI build fails on missing sdk/. Either commit the generated sdk/ directory, or run mod generate (or dagger generate) as a build step before calling the module.

Engine version mismatch warnings. Align dagger-module.toml's engineVersion with your engine using mod engine require/require-current/require-latest, then regenerate bindings.