Getting started with Jubilant

In this tutorial, we’ll learn how to install Jubilant, use it to run Juju commands, and write a simple charm integration test.

The tutorial assumes that you have a basic understanding of Juju and have already installed it. Learn how to install the Juju CLI.

Install Jubilant

Jubilant is published to PyPI, so you can install and use it with your favorite Python package manager:

$ pip install jubilant
# or
$ uv add jubilant

Like the Ops framework used by charms, Jubilant requires Python 3.8 or above.

Check your setup

To check that Jubilant is working, use it to add a Juju model and check its status:

$ uv run python
>>> import jubilant
>>> juju = jubilant.Juju()
>>> juju.add_model('test')
>>> juju.status()
Status(
  model=ModelStatus(
    name='test',
    type='caas',
    controller='k8s',
    cloud='my-k8s',
    version='3.6.4',
    region='localhost',
    model_status=StatusInfo(
      current='available',
      since='22 Mar 2025 12:34:12+13:00',
    ),
  ),
  machines={},
  apps={},
  controller=ControllerStatus(timestamp='12:34:17+13:00'),
)

Compare the status to what’s displayed when using the Juju CLI directly:

$ juju status --model test
Model  Controller  Cloud/Region      Version  SLA          Timestamp
test   k8s         my-k8s/localhost  3.6.4    unsupported  12:35:05+13:00

Model "test" is empty.

Write a charm integration test

We recommend using pytest for writing tests. You can define a pytest fixture to create a temporary Juju model for each test. The jubilant.temp_model context manager creates a randomly-named model on entry, and destroys the model on exit.

Here is a module-scoped fixture called juju, which you would normally define in conftest.py:

@pytest.fixture(scope='module')
def juju():
    with jubilant.temp_model() as juju:
        yield juju

Integration tests in a test file would use the fixture, operating on the temporary model:

def test_deploy(juju: jubilant.Juju):
    juju.deploy('snappass-test')
    status = juju.wait(jubilant.all_active)
    assert status.apps['snappass-test'].scale == 1

You may want to adjust the scope of your juju fixture. For example, to create a new model for every test function (pytest’s default behavior), omit the scope:

@pytest.fixture
def juju():
    ...

Use a custom wait condition

When waiting on a condition with Juju.wait, you can use pre-defined helpers including jubilant.all_active and jubilant.any_error. You can also define custom conditions for the ready and error parameters. This is typically done with inline lambda functions.

For example, to test that the myapp charm starts up with application status “unknown”:

def test_unknown(juju: jubilant.Juju):
    juju.deploy('myapp')
    juju.wait(
        lambda status: status.apps['myapp'].app_status.current == 'unknown',
    )

There are also is_* properties on the AppStatus and UnitStatus classes for the common statuses: is_active, is_blocked, is_error, is_maintenance, and is_waiting.

For example, to wait till myapp is active and yourapp is blocked, and to raise an error if any app or unit goes into error state:

def test_custom_wait(juju: jubilant.Juju):
    juju.deploy('myapp')
    juju.deploy('yourapp')
    juju.wait(
        lambda status: (
            status.apps['myapp'].is_active and
            status.apps['yourapp'].is_blocked
        ),
        error=jubilant.any_error,
    )

Fall back to Juju.cli if needed

Many common Juju commands are already defined on the Juju class, such as deploy and integrate.

However, if you want to run a Juju command that’s not yet defined in Jubilant, you can fall back to calling the Juju.cli method. For example, to fetch a model configuration value using juju model-config:

>>> import json
>>> import jubilant
>>> juju = jubilant.Juju(model='test')
>>> stdout = juju.cli('model-config', '--format=json')
>>> result = json.loads(stdout)
>>> result['automatically-retry-hooks']['Value']
True

By default, Juju.cli adds a --model=<model> parameter if the Juju instance has a model set. To prevent this for commands not specific to a model, specify include_model=False:

>>> stdout = juju.cli('controllers', '--format=json', include_model=False)
>>> result = json.loads(stdout)
>>> result['controllers']['k8s']['uuid']
'cda7763e-05fc-4e55-80ab-7b39badaa50d'

Use concierge in CI

We recommend using concierge to set up Juju when running your integration tests in CI. It will install Juju with a provider like MicroK8s and bootstrap a controller for you. For example, using GitHub Actions:

- name: Install concierge
  run: sudo snap install --classic concierge

- name: Install Juju and bootstrap
  run: |
      sudo concierge prepare \
          --juju-channel=3/stable \
          --charmcraft-channel=3.x/stable \
          --preset microk8s

- name: Run integration tests
  run: |
      charmcraft pack
      uv run pytest tests/integration -vv --log-level=INFO

Next steps

You’ve now learned the basics of Jubilant! To learn more:

If you have any problems or want to request new features, please open an issue.