Local agent-first experiment workbench CLI
Project description
ALab
ALab is a local, agent-first Python CLI workbench for iterative experiments. It lets external agents work in isolated Git worktrees, run repeatable evaluations, submit final results, and inspect visible prior experiment evidence through explicit collaboration boundaries.
ALab V1 is intentionally local-only: no server, sync service, web UI, built-in agent launcher, or account system. It owns the local project records, source snapshots, experiment lifecycle, runner execution, logs, artifacts, and visibility rules; agents remain external CLI operators.
Highlights
- Local CLI workbench for projects, sources, experiments, runs, submissions, logs, artifacts, annotations, and audits.
- Context-aware command surface:
alab helpand command preflight show only what the current project, experiment, inspection checkout, token, or explicit key can use. - Git-backed experiment isolation: each experiment is an isolated branch/worktree with a worktree token for run and submit operations.
- Reproducible project setup: project config controls runner, reward, artifact capture, environment, secrets, mutable paths, and visibility.
- Runner support for local subprocesses, Docker images/Dockerfiles, Harbor verifiers, and SkyDiscover Python/Docker evaluators.
- Collaboration boundary, not strong local security: root/admin keys and experiment tokens gate CLI capabilities, while local project records remain plaintext.
- Secret hygiene: raw keys/tokens are not stored; generated raw keys are printed once, experiment tokens stay in token files, and
secret_envvalues are not rendered or exported. - Open-source documentation set with English canonical docs and synchronized Chinese
*_cn.mdcompanions.
Current Status
The current V1 implementation is runnable and has a closed current-worktree evidence ledger. The product contract remains docs/blueprint.md; detailed requirement evidence lives in docs/completion_audit.md; current progress and active queues live in docs/progress.md and docs/progress_pipeline.md.
Environment Requirements
Required:
- macOS or Linux. Windows is not part of V1 acceptance testing.
- Python 3.11 or newer.
- Git.
uvfor the project-local Python environment and locked dependency resolution.
Optional:
- Docker, only for Docker runner and Harbor/SkyDiscover Docker evaluator workflows.
- Network access to Python package indexes for dependency-installing evaluator tests.
- Network access to GitHub for live SkyDiscover catalog validation.
- Codex CLI or another external agent runtime if you want autonomous workers; ALab itself does not launch agents.
Local environment variables are documented in .env.example. Real .env files are ignored; do not commit root keys, project admin keys, experiment tokens, or secret values.
Installation
ALab is distributed as the alab-cli Python package and installs the alab console script. Once the package is published to a Python package index, install it directly with pip:
python -m pip install alab-cli
alab help
Until then, install from a checkout or Git URL:
python -m pip install "git+https://github.com/bebetterest/ALab.git"
Or, from an already cloned checkout:
git clone https://github.com/bebetterest/ALab.git ALab
cd ALab
python -m pip install .
alab help
For editable local development with the installed command:
python -m pip install -e .
alab help
If you prefer an isolated CLI tool environment, uv can install the same console script:
uv tool install --editable .
alab help
If pip or uv installs the command into a directory that is not on PATH, add the displayed script directory to your shell path.
For repository development and locked local verification, use the checkout environment instead:
uv sync --locked
uv run --locked alab help
Run the default validation suite:
uv run --locked pytest -q
uv run --locked ruff check
If a local package mirror is slow or unavailable, use the official Python index for the current command:
UV_CACHE_DIR=.uv-cache UV_DEFAULT_INDEX=https://pypi.org/simple uv run --locked pytest -q
If dependency downloads are unavailable but the current Python environment already has the required test dependencies, the source tree can still be exercised directly:
PYTHONPATH=src python -m pytest -q
Quick Start
This local runner example creates an isolated ALab home under the repository, initializes a project from a small Python source tree, creates one experiment, runs evaluation, and submits the result.
The commands below assume alab is installed. If you are working only from the checkout, replace alab with uv run --locked alab while inside the repository, or with uv run --project /absolute/path/to/ALab --locked alab from an experiment worktree.
Create a demo source and config:
mkdir -p .alab-demo/source
cat > .alab-demo/source/main.py <<'PY'
print("reward=1.0")
PY
cat > .alab-demo/alab.project.toml <<'TOML'
[runner]
type = "local"
command = ["python", "main.py"]
timeout_seconds = 60
working_directory = "."
env_mode = "sanitized"
[reward]
type = "stdout_regex"
direction = "maximize"
primary_metric = "reward"
pattern = "reward=([0-9.]+)"
TOML
Initialize an ALab home. The root key is printed once:
ALAB_HOME="$PWD/.alab-demo/home" alab auth init
Initialize the project with the printed root key:
ALAB_HOME="$PWD/.alab-demo/home" alab project init local \
--config .alab-demo/alab.project.toml \
--source-path .alab-demo/source \
--name "Demo" \
--task "Keep the reward output passing" \
--key <root-key>
Create an experiment. Public experiment creation is enabled by default for local agent bootstrap:
ALAB_HOME="$PWD/.alab-demo/home" alab exp create \
--project <project-id> \
--name "attempt-1"
Enter the rendered worktree path, run the evaluator, and submit after a passed run:
cd <worktree-path>
ALAB_HOME="/absolute/path/to/ALab/.alab-demo/home" alab status
ALAB_HOME="/absolute/path/to/ALab/.alab-demo/home" alab run --message "baseline demo run"
ALAB_HOME="/absolute/path/to/ALab/.alab-demo/home" alab submit \
--message "final demo candidate" \
--summary "The demo candidate prints a parseable reward." \
--feedback "The latest run passed with reward 1.0." \
--ref none
Useful next commands:
alab help
alab observe experiments list
alab observe runs list --exp <exp-id>
alab observe experiments best
Core Concepts
- Home: local ALab state root containing SQLite records, config, caches, backups, and project storage.
- Project: task, goal, config, source registry, validation records, visibility policy, and project admin boundary.
- Source: immutable snapshot imported from a local path, Git repo, empty source, Harbor task source, or SkyDiscover initial program.
- Experiment: isolated Git branch/worktree bound to exactly one source and one config version.
- Run: one evaluator execution for an experiment commit, with status, reward, logs, artifacts, and warning codes.
- Submit: closes an experiment with final summary, feedback, final run, final commit, and explicit refs.
- Inspection checkout: read-only checkout for observing/exporting scoped experiment evidence without becoming submit-capable.
Configuration
Project behavior is controlled by TOML config:
[runner]: runner type, command/shell, working directory, timeout, Docker fields, Harbor task refs, or SkyDiscover task refs.[reward]: reward type and primary metric. Supported reward families includeexit_code,file,stdout_regex,harbor, andskydiscover.[artifacts]and[logs]: captured output roots, glob patterns, and byte limits.[env]and[secret_env]: explicit environment injection. Secret values are local plaintext but are not rendered or exported.[mutable]: paths the experiment may change when running or submitting.- Visibility/public bootstrap policy: controls how experiments can see prior work and whether local no-key experiment creation is allowed.
See docs/spec_runners_adapters.md, docs/spec_project_source_experiment.md, and docs/spec_observe_collaboration.md for the detailed contracts.
Examples
See examples for the runnable example matrix. The current examples cover a local scoring loop, a Dockerized clinic-order fulfillment planner with artifact export, a Harbor hidden-verifier incident classifier, a collaborative incident triage lifecycle workflow, and the SkyDiscover circle-packing Codex single-worker protocol.
The repository also includes Codex-facing role skills under skills. They are external runbooks for operating ALab through the CLI as an experiment worker, project controller, or global admin; they do not add an embedded agent launcher to ALab.
Testing And Development
Default checks:
UV_CACHE_DIR=.uv-cache UV_DEFAULT_INDEX=https://pypi.org/simple uv run --locked pytest -q
UV_CACHE_DIR=.uv-cache UV_DEFAULT_INDEX=https://pypi.org/simple uv run --locked ruff check
Opt-in integration gates are excluded from the default suite:
ALAB_RUN_REAL_DOCKER=1 UV_CACHE_DIR=.uv-cache UV_DEFAULT_INDEX=https://pypi.org/simple uv run pytest -m real_docker
ALAB_RUN_REAL_SKYDISCOVER_PYTHON=1 UV_CACHE_DIR=.uv-cache UV_DEFAULT_INDEX=https://pypi.org/simple uv run pytest -m real_skydiscover_python
ALAB_RUN_NETWORKED_SKYDISCOVER_PYTHON=1 UV_CACHE_DIR=.uv-cache UV_DEFAULT_INDEX=https://pypi.org/simple uv run pytest -m networked_skydiscover_python
ALAB_RUN_NATIVE_SKYDISCOVER_PYTHON=1 UV_CACHE_DIR=.uv-cache UV_DEFAULT_INDEX=https://pypi.org/simple uv run pytest -m native_skydiscover_python
ALAB_RUN_LIVE_SKYDISCOVER_CATALOG=1 UV_CACHE_DIR=.uv-cache UV_DEFAULT_INDEX=https://pypi.org/simple uv run pytest -m live_skydiscover_catalog
Notes:
uv.lockis tracked because CI and local validation useuv run --locked.- Keep local cache/output paths ignored (
.uv-cache/,.pytest_cache/,.ruff_cache/,.alab-demo/,.env). - GitHub Actions runs the default lint and pytest suite on pull requests and pushes to
main; real Docker and live/networked SkyDiscover gates remain manual workflow inputs. - Pushes to
maincheck PyPI for the currentpyproject.tomlpackage version; if that exact version is missing, CI builds and publishes through PyPI Trusted Publishing, otherwise it skips publishing. - The PyPI
alab-cliproject must trust repositorybebetterest/ALab, workflowci.yml, and environmentpypibefore the first automated publish can succeed.
Security And Data Model
ALab V1 is a local collaboration boundary, not a multi-user security product:
- Raw root/admin keys are printed only at creation/regeneration.
- Raw experiment tokens are written to token files and are not printed.
- Credential verifiers are stored; raw credential secrets are not.
- Project records are local plaintext SQLite/filesystem data.
secret_envvalues are local plaintext, redacted from rendered logs where configured, and never exported by config commands.- Artifact exports are exact captured bytes and are not automatically redacted.
Repository Structure
.
├── .github/
│ └── workflows/
│ └── ci.yml
├── docs/
│ ├── assets/
│ │ └── readme-header.png
│ ├── blueprint.md
│ ├── blueprint_cn.md
│ ├── completion_audit.md
│ ├── completion_audit_cn.md
│ ├── progress.md
│ ├── progress_cn.md
│ ├── progress_pipeline.md
│ └── progress_pipeline_cn.md
├── examples/
│ ├── README.md
│ ├── README_cn.md
│ ├── collaboration_observe_lifecycle/
│ ├── docker_file_reward_artifacts/
│ ├── harbor_verifier_minimal/
│ ├── local_agent_scoreboard/
│ └── skydiscover_circle_packing_codex/
├── skills/
│ ├── alab-experiment-worker/
│ ├── alab-project-controller/
│ └── alab-global-admin/
├── src/
│ └── alab/
├── tests/
│ ├── test_smoke.py
│ ├── test_cli_contract.py
│ ├── test_runner_docker.py
│ ├── test_runner_harbor.py
│ └── test_runner_skydiscover.py
├── LICENSE
├── .env.example
├── pyproject.toml
├── uv.lock
├── README.md
└── README_cn.md
Local-only agent notes such as AGENTS.md and CORE.md are intentionally git-ignored and are not part of the public repository layout.
Documentation
- English documentation is canonical.
- Synchronized Chinese documents use the
*_cn.mdnaming pattern. - docs/blueprint.md is the V1 product overview.
- docs/spec_cli.md, docs/spec_storage_auth_context.md, docs/spec_project_source_experiment.md, docs/spec_lifecycle.md, docs/spec_runners_adapters.md, docs/spec_observe_collaboration.md, and docs/spec_tests.md define subsystem contracts.
- docs/progress.md, docs/progress_pipeline.md, docs/progress_closed_gaps.md, and docs/progress_log.md track current state, active queues, closed gaps, and history.
- docs/completion_audit.md tracks requirement-level evidence.
Contributing
- Keep changes scoped and aligned with the blueprint/spec contracts.
- Update English docs first, then synchronize the matching Chinese
*_cn.mdfile. - Add focused tests for new behavior and run the relevant pytest/ruff commands before opening a PR.
- Do not commit real
.envfiles, raw keys, experiment tokens, generated caches, local ALab homes, or private runner outputs.
License
The project license is GPL-3.0-or-later; see LICENSE.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file alab_cli-0.1.1.tar.gz.
File metadata
- Download URL: alab_cli-0.1.1.tar.gz
- Upload date:
- Size: 369.5 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
1e737af6e800603a32c1d32977dda01b56c68cbe2d544fa1234987656efd2608
|
|
| MD5 |
9b02336d76287eced3b95ae3252083af
|
|
| BLAKE2b-256 |
71b53f6e37ed70350beeec5899a409f10c91adb272c4649e16d8f5e1eb68e9cc
|
Provenance
The following attestation bundles were made for alab_cli-0.1.1.tar.gz:
Publisher:
ci.yml on bebetterest/ALab
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
alab_cli-0.1.1.tar.gz -
Subject digest:
1e737af6e800603a32c1d32977dda01b56c68cbe2d544fa1234987656efd2608 - Sigstore transparency entry: 1614108384
- Sigstore integration time:
-
Permalink:
bebetterest/ALab@36a1f43917965e16dc03000755d2013699cf5127 -
Branch / Tag:
refs/heads/main - Owner: https://github.com/bebetterest
-
Access:
private
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
ci.yml@36a1f43917965e16dc03000755d2013699cf5127 -
Trigger Event:
push
-
Statement type:
File details
Details for the file alab_cli-0.1.1-py3-none-any.whl.
File metadata
- Download URL: alab_cli-0.1.1-py3-none-any.whl
- Upload date:
- Size: 138.0 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.12
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
6413c0561ddcdedf1a49eee0da2041ddef8b2830f153edbca07bfb03c040fb47
|
|
| MD5 |
baffc6e316ad046a98af2f2f04587d9c
|
|
| BLAKE2b-256 |
656b872a7542c2cce36e9428d444b8948e12ffdc6b35215de9d9afcec036a666
|
Provenance
The following attestation bundles were made for alab_cli-0.1.1-py3-none-any.whl:
Publisher:
ci.yml on bebetterest/ALab
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
alab_cli-0.1.1-py3-none-any.whl -
Subject digest:
6413c0561ddcdedf1a49eee0da2041ddef8b2830f153edbca07bfb03c040fb47 - Sigstore transparency entry: 1614108439
- Sigstore integration time:
-
Permalink:
bebetterest/ALab@36a1f43917965e16dc03000755d2013699cf5127 -
Branch / Tag:
refs/heads/main - Owner: https://github.com/bebetterest
-
Access:
private
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
ci.yml@36a1f43917965e16dc03000755d2013699cf5127 -
Trigger Event:
push
-
Statement type: