Getting Started

This guide will help you get up and running with ConTree SDK. By the end of this guide, you’ll understand how to create clients, work with images, and run your first commands.

Configuration

ConTree SDK uses environment variables for configuration:

  • CONTREE_TOKEN: Your authentication token

  • CONTREE_BASE_URL: Your ConTree instance URL

export CONTREE_TOKEN="your_token_here"
export CONTREE_BASE_URL="https://your-instance.of.contree"

Alternatively, you can pass token and base_url directly when creating a client.

Creating a Client

The first step is to create a ConTree client. You can choose between async and sync versions depending on your application needs.

Here’s how to create a client and verify the connection by listing available images:

1# Get client
2client = Contree()
3
4# Get images (to verify that connection works)
5await client.images()

See Contree for all client options.

1# Get client
2client = ContreeSync()
3
4# Get images (to verify that connection works)
5client.images()

See ContreeSync for all client options.

Working with Images

Images are the foundation of ConTree. The simplest way to reference an image is by tag using images.use(), which creates an image object without making an API call:

image = await contree.images.use("ubuntu:latest")
result = await image.run(shell="echo hello")
image = contree.images.use("ubuntu:latest")
result = image.run(shell="echo hello").wait()

To resolve a tag or UUID upfront via an API call, use images.use(strict=True). To import from an external registry (or return an existing image if already imported), use images.oci().

See Working with Images for a full overview of available methods, examples, and what you can pass as a reference.

Running Commands

Once you have an image, you can run commands inside it. Each command execution creates a new version of the image with your changes.

Basic Command Execution

You can run various shell commands and handle their output:

 1image = client.images.use("busybox:latest")
 2print(f"Using {image=}")
 3
 4result = await image.run(shell="echo 'Hello World'")
 5print(f"Simple echo: {result.stdout=}, {result.stderr=}, {result.exit_code=}")
 6
 7result = await image.run(shell="pwd")
 8print(f"Current directory: {result.stdout=}, {result.exit_code=}")
 9
10result = await image.run(shell="ls -la")
11print(f"Directory listing: {result.stdout=}, {result.exit_code=}")
12
13result = await image.run(shell="cat -", stdin="Hello from stdin\n")
14print(f"Cat with stdin: {result.stdout=}, {result.exit_code=}")
15
16result = await image.run(shell="echo 'Error message' >&2; exit 1")
17print(f"Error command: {result.stdout=}, {result.stderr=}, {result.exit_code=}")

See run() for all command execution options.

 1image = client.images.use("busybox:latest")
 2print(f"Using {image=}")
 3
 4result = image.run(shell="echo 'Hello World'").wait()
 5print(f"Simple echo: {result.stdout=}, {result.stderr=}, {result.exit_code=}")
 6
 7result = image.run(shell="pwd").wait()
 8print(f"Current directory: {result.stdout=}, {result.exit_code=}")
 9
10result = image.run(shell="ls -la").wait()
11print(f"Directory listing: {result.stdout=}, {result.exit_code=}")
12
13result = image.run(shell="cat -", stdin="Hello from stdin\n").wait()
14print(f"Cat with stdin: {result.stdout=}, {result.exit_code=}")
15
16result = image.run(shell="echo 'Error message' >&2; exit 1").wait()
17print(f"Error command: {result.stdout=}, {result.stderr=}, {result.exit_code=}")

See run() for all command execution options.

Understanding the Results

When you run a command, you get back a result object that contains:

  • stdout: Standard output from the command

  • stderr: Standard error from the command

  • exit_code: The exit code (0 for success, non-zero for errors)

  • uuid: The UUID of the new image version created by this command