> ## Documentation Index
> Fetch the complete documentation index at: https://docs.canton.network/llms.txt
> Use this file to discover all available pages before exploring further.

# Sandbox

> Run a local Canton node for rapid development and testing with dpm sandbox.

The Sandbox is a single-participant Canton node that runs locally on your machine. Start it with `dpm sandbox` for quick development cycles without the overhead of Docker or a full network.

## Starting the Sandbox

```bash theme={"theme":{"light":"github-light","dark":"github-dark"}}
dpm sandbox
```

This starts a Canton participant node with an in-memory ledger. The Ledger API becomes available at `localhost:6865` by default. The process runs in the foreground -- press Ctrl+C to stop it.

The Sandbox uses the SDK version from your project's `daml.yaml`, so make sure you have run `dpm install` first.

## What the Sandbox Provides

* A single Canton node
* Ledger API (gRPC) endpoint for command submission and transaction streaming
* JSON API endpoint for HTTP-based access
* Admin API for party allocation and DAR uploads
* In-memory storage (state is lost when the Sandbox stops)

The Sandbox does not include a synchronizer, wallet services, or Canton Coin. For those features, use [LocalNet](/sdks-tools/development-tools/localnet).

## Common Development Workflows

### Upload and Test Contracts

After compiling your Daml code, upload the DAR to the running Sandbox and interact with it:

```bash theme={"theme":{"light":"github-light","dark":"github-dark"}}
# In terminal 1: start the sandbox
dpm sandbox

# In terminal 2: upload the compiled DAR
# (using the gRPC Admin API or Canton Console)
```

### Connect a Backend

Point your Java or TypeScript backend at the Sandbox's Ledger API endpoint. The Sandbox supports the same gRPC and JSON API interfaces as production validators, so your backend code does not need to change.

```
# Ledger API (gRPC): localhost:6865
# JSON API: available on a separate port (check sandbox output)
```

### Allocate Parties

Use the Admin API or Canton Console to create test parties:

```bash theme={"theme":{"light":"github-light","dark":"github-dark"}}
# Via Canton Console (if using the interactive console)
participant.parties.enable("Alice")
participant.parties.enable("Bob")
```

## When to Use Sandbox vs LocalNet

| Scenario                               | Sandbox | LocalNet |
| -------------------------------------- | ------- | -------- |
| Testing contract logic with Ledger API | Yes     | Yes      |
| Testing codegen serialization          | Yes     | Yes      |
| Multi-validator workflows              | No      | Yes      |
| Canton Coin and payments               | No      | Yes      |
| Wallet integration                     | No      | Yes      |
| Splice API interactions                | No      | Yes      |
| Quick iteration, no Docker             | Yes     | No       |

**General rule**: Use the Sandbox for fast feedback during contract development and single-participant integration testing. Switch to LocalNet when your tests need multiple validators, payments, or Splice services.

## When to Use Sandbox vs Daml Script

Daml Script (`dpm test`) and the Sandbox serve different testing needs:

* **Daml Script** runs test scenarios in-process without a ledger. It is the fastest way to test contract logic, authorization rules, and multi-party workflows. Use it for unit testing your Daml model.

* **The Sandbox** runs a real Canton node. Use it when you need to test your application's integration with the Ledger API -- verifying that codegen-generated types serialize correctly, that your backend connects and authenticates properly, and that JSON API requests produce expected results.

## Stopping the Sandbox

Press Ctrl+C in the terminal where the Sandbox is running. Since the Sandbox uses in-memory storage, all data is lost on shutdown. This is by design -- each test run starts with a clean ledger.

## Related Pages

* [dpm command reference](/sdks-tools/cli-tools/dpm) -- `dpm sandbox` options
* [LocalNet](/sdks-tools/development-tools/localnet) -- Full multi-validator environment for integration testing
* [Daml Script](/sdks-tools/cli-tools/daml-script) -- In-process testing without a running node
* [Testing Daml Contracts](/appdev/modules/m3-testing) -- Complete testing guide

# Sandbox<span id="component-howtos-application-development-daml-sandbox" />

Sandbox is a program for running a Canton ledger with your Daml code. The ledger uses the simplest topology possible: a single Participant Node connected to a Synchronizer Node. Use the sandbox when you need access to a Canton ledger running your Daml code and matching the target Participant Node topology is not required.

## Install

Install the Sandbox by installing dpm.

## Configure

To configure the Sandbox, use the command line.

### Command line configuration

To view all available command line configuration options for Sandbox, run `dpm sandbox --help` in your terminal:

```none theme={"theme":{"light":"github-light","dark":"github-dark"}}
Usage: dpm sandbox [--port ARG] [--admin-api-port ARG]
                    [--sequencer-public-port ARG] [--sequencer-admin-port ARG]
                    [--mediator-admin-port ARG] [--json-api-port ARG]
                    [--json-api-port-file PATH] [--canton-port-file PATH]
                    [--static-time | --wall-clock-time] [--canton-help]
                    [-c|--config FILE] [--port-file PATH] [--dar PATH] [ARG]

Available options:
  --json-api-port ARG      Port that the HTTP JSON API should listen on, omit to
                           disable it
  --json-api-port-file PATH
                           File to write canton json-api port when ready
  --canton-port-file PATH  File to write canton participant ports when ready
  --canton-help            Display the help of the underlying Canton JAR instead
                           of the Sandbox wrapper. This is only required for
                           advanced options.
  -c,--config FILE         Set configuration file(s). If several configuration
                           files assign values to the same key, the last value
                           is taken.
  --port-file PATH         File to write ledger API port when ready
  --dar PATH               DAR file to upload to sandbox
  --shutdown-stdin-close   Shut down when stdin is closed, disabled by default
  -h,--help                Show this help text
```

Any unrecognized command-line arguments will be treated as arguments to be supplied to the underlying Canton JAR. Display these options by running `dpm sandbox --canton-help` in your terminal.

### Canton configuration

Behind the scenes, Sandbox runs an underlying Canton ledger with a default configuration file to initialize a participant named `sandbox`, a sequencer named `sequencer1`, and a mediator named `mediator1`.

Configure the underlying Canton ledger further in one of two ways:

* Specify additional configuration file(s) to apply on top of the default configuration using `--config filepath`. If several configuration files assign to the same key, the last value is taken.

* Set the value of a specific key with `-C key=value`. For example, to override the ledger-api port of the sandbox participant, the command would be:

  ```none theme={"theme":{"light":"github-light","dark":"github-dark"}}
  dpm sandbox -C canton.participants.sandbox.ledger-api.port=9999
  ```

### Canton Declarative API

The declarative API allows specifying DARs, Parties and Users desired to be present on the Canton ledger. The ledger will automatically take care of uploading the DARs, and creating the parties and users.

* upload local DARs

```none theme={"theme":{"light":"github-light","dark":"github-dark"}}
canton.parameters.enable-alpha-state-via-config = yes
canton.parameters.state-refresh-interval = 5s

canton.participants.sandbox.alpha-dynamic.dars = [
  { location = "./my-asset.dar" },
]
```

## Operate

Start Canton with a single participant:

```none theme={"theme":{"light":"github-light","dark":"github-dark"}}
$ dpm sandbox
Starting Canton sandbox.
Listening at port 6865
Canton sandbox is ready.
```

### Interacting with Sandbox's ledger

Once the sandbox is running, you may interact with it the same way you would for any Canton instance. For example, you may upload dars to it, or run scripts against it:

```none theme={"theme":{"light":"github-light","dark":"github-dark"}}
$ dpm sandbox --dar <path to DAR>
$ dpm script --ledger-host localhost --port 6865 --dar <path to DAR> --script-name <script name in DAR>
```

Because `dpm sandbox` is a Canton instance, all documentation for using Canton applies.

### Connecting to Sandbox's console

Once you have a Sandbox running locally (i.e. after running `dpm sandbox`) you may connect to Sandbox remotely by running the `dpm canton-console` command in a separate terminal:

```none theme={"theme":{"light":"github-light","dark":"github-dark"}}
$ daml canton-console
   _____            _
  / ____|          | |
 | |     __ _ _ __ | |_ ___  _ __
 | |    / _` | '_ \| __/ _ \| '_ \
 | |___| (_| | | | | || (_) | | | |
  \_____\__,_|_| |_|\__\___/|_| |_|

  Welcome to Canton!
  Type `help` to get started. `exit` to leave.

@
```

You can quit the session by running the `exit` command.

#### Built-in documentation

The Canton console comes with built-in documentation. You can use the `help` command to get online documentation for top-level commands. Many objects in the console also have built-in help that you can access by invoking the `help` method on them.

For example, you can ask for help on the `health` object by typing:

```scala theme={"theme":{"light":"github-light","dark":"github-dark"}}
health.help
```

Or go more in-depth about specific items within that object, as in the following example:

```scala theme={"theme":{"light":"github-light","dark":"github-dark"}}
health.help("status")
```

#### Interact with the Sandbox

One of the objects available in the Canton console represents the Sandbox itself. The object is called `sandbox` and you can use it to interact with the Sandbox. For example, you can list the DARs loaded on the Sandbox by running the following command:

```scala theme={"theme":{"light":"github-light","dark":"github-dark"}}
sandbox.dars.list()
```

Among the various features available as part of the console, you can manage parties and packages, check the health of the Sandbox, perform pruning operations, and more. Consult the built-in documentation mentioned above and the main documentation for the Canton console to learn about further capabilities.

#### How it works

Canton offers a console where you can run administrative or debugging commands.

When you run the Sandbox using `dpm sandbox`, you are effectively starting an in-memory instance of Canton with a single sync domain and a single participant.

As such, you can interact with the running Sandbox using the console, just like you would in a production environment.

For an in-depth guide on how to use this tool against a production, staging or testing environment, consult the main documentation for the Canton console.

### Testing your Daml contracts

Sandbox is primarily used as the first step in testing your Daml contracts in isolation

### Run with authorization

By default, Sandbox accepts all valid Ledger API requests without performing any request authorization.

To start Sandbox with authorization using [JWT-based](https://jwt.io/) access tokens as described in the Authorization documentation, create a config file that specifies the type of authorization service and the path to the certificate, then supply that config file to Sandbox with `dpm sandbox --config auth.conf`.

```none theme={"theme":{"light":"github-light","dark":"github-dark"}}
canton.participants.sandbox.ledger-api.auth-services = [{
    // type can be
    //   jwt-rs-256-crt
    //   jwt-es-256-crt
    //   jwt-es-512-crt
    //   jwt-rs-256-jwks with an additional url
    //   unsafe-jwt-hmac-256 with an additional secret
    type = jwt-rs-256-crt
    certificate = my-certificate.cert
}]
```

The settings under `auth-services` are described in detail in [API configuration documentation](/global-synchronizer/reference/security-configuration#jwt-authorization)

#### Generate JSON web tokens (JWT)

To generate access tokens for testing purposes, use the [jwt.io](https://jwt.io/) web site.

#### Generate RSA keys

To generate RSA keys for testing purposes, use the following command

```none theme={"theme":{"light":"github-light","dark":"github-dark"}}
openssl req -nodes -new -x509 -keyout sandbox.key -out sandbox.crt
```

which generates the following files:

* `sandbox.key`: the private key in PEM/DER/PKCS#1 format
* `sandbox.crt`: a self-signed certificate containing the public key, in PEM/DER/X.509 Certificate format

#### Generate EC keys

To generate keys to be used with ES256 for testing purposes, use the following command

```none theme={"theme":{"light":"github-light","dark":"github-dark"}}
openssl req -x509 -nodes -days 3650 -newkey ec:<(openssl ecparam -name prime256v1) -keyout ecdsa256.key -out ecdsa256.crt
```

which generates the following files:

* `ecdsa256.key`: the private key in PEM/DER/PKCS#1 format
* `ecdsa256.crt`: a self-signed certificate containing the public key, in PEM/DER/X.509 Certificate format

Similarly, you can use the following command for ES512 keys:

```none theme={"theme":{"light":"github-light","dark":"github-dark"}}
openssl req -x509 -nodes -days 3650 -newkey ec:<(openssl ecparam -name secp521r1) -keyout ecdsa512.key -out ecdsa512.crt
```

### Run with TLS

To enable TLS, you need to specify the private key for your server and the certificate chain. This enables TLS for both the gRPC Ledger API and the Canton Admin API. When enabling client authentication, you also need to specify client certificates which can be used by Canton’s internal processes. Note that the identity of the application will not be proven by using this method, i.e. the `application_id` field in the request is not necessarily correlated with the CN (Common Name) in the certificate. Below, you can see an example config. For more details on TLS, refer to Canton’s documentation on TLS configuration.

```none theme={"theme":{"light":"github-light","dark":"github-dark"}}
canton.participants.sandbox.ledger-api {
  tls {
    // the certificate to be used by the server
    cert-chain-file = "./tls/ledger-api.crt"
    // private key of the server
    private-key-file = "./tls/ledger-api.pem"
    // trust collection, which means that all client certificates will be verified using the trusted
    // certificates in this store. if omitted, the JVM default trust store is used.
    trust-collection-file = "./tls/root-ca.crt"
    // define whether clients need to authenticate as well (default not)
    client-auth = {
      // none, optional and require are supported
      type = require
      // If clients are required to authenticate as well, we need to provide a client
      // certificate and the key, as Canton has internal processes that need to connect to these
      // APIs. If the server certificate is trusted by the trust-collection, then you can
      // just use the server certificates. Otherwise, you need to create separate ones.
      admin-client {
        cert-chain-file = "./tls/admin-client.crt"
        private-key-file = "./tls/admin-client.pem"
      }
    }
  }
}
```

### Dev Protocol

To enable the canton dev protocol:

```bash theme={"theme":{"light":"github-light","dark":"github-dark"}}
cat <<EOF
canton.participants.sandbox.parameters.alpha-version-support = true
canton.sequencers.sequencer1.parameters.alpha-version-support = true
canton.parameters {
    non-standard-config = yes
    alpha-version-support = yes
}
EOF > dev-protocol.conf

CANTON_PROTOCOL_VERSION=dev dpm sandbox -c dev-protocol.conf
```

## Troubleshoot

### Failed to bind to address

By default, Sandbox reserves five ports for its Canton services:

* `6865` for the participant's Ledger API
* `6866` for the participant's Admin API
* `6867` for the sequencer's public API
* `6868` for the sequencer's admin API
* `6869` for the mediator's admin API

The Sandbox will also bind to the port specified in the `--json-api-port`, if any.

When one of these ports is already used by an existing process, Sandbox will emit an error that contains the following text:

```none theme={"theme":{"light":"github-light","dark":"github-dark"}}
Failed to bind to address /127.0.0.1:<port number>
```

This is most commonly either caused by an existing process that is already listening on that port, or if you do not have the permissions to bind to that address.

On Linux, the `lsof -n -i` command lists what processes are already listening to a port. For example, if an existing Java program is already listening to 6865, `lsof` would look as follows:

```none theme={"theme":{"light":"github-light","dark":"github-dark"}}
$ lsof -n -i
...
java       707977 username       77u  IPv6 67556378      0t0  TCP 127.0.0.1:6865 (LISTEN)
...
```

If killing the existing process isn't an option, or if you don't have the permission to bind to a given port, you can reconfigure the ports of a given node using the top-level options described below:

* Use --port= to override binding to `6865`
* Use --admin-api-port= to override binding to `6866`
* Use --sequencer-public-port= to override binding to `6867`
* Use --sequencer-admin-port= to override binding to `6868`
* Use --mediator-admin-port= to override binding to `6869`
* Use `--json-api-port` to change the port to which the JSON API binds.

### SDK not installed

If the `daml.yaml` file of the project you are currently in specifies a version of the dpm SDK that is not installed, you may get the following error message:

```none theme={"theme":{"light":"github-light","dark":"github-dark"}}
SDK not installed. Cannot run command without SDK.
```

To fix this, you can:

* Install the SDK as instructed to by the error, or
* Change the SDK version in the project's `daml.yaml` file, or
* Change directories to be outside of the project, where the default Daml version that is already installed on your system will be used.
