Async Python wrapper for OpenCode CLI (opencode run --format json)
Project description
oc-py-harness
Python async wrapper around the OpenCode CLI (opencode run --format json). Intended as a subprocess-based executor for multi-agent workflow orchestration.
Requirements
- Python 3.10+
opencodeonPATH(or pass an absolute path to the binary)
Install (local tree)
pip install -e ".[dev]"
Usage
One-shot run with aggregated result
import asyncio
from pathlib import Path
from opencode_wrapper import AsyncOpenCodeClient, RunConfig
async def main():
client = AsyncOpenCodeClient("opencode")
cfg = RunConfig(
model="anthropic/claude-sonnet-4-5",
agent="plan",
permission={"bash": "deny", "edit": "deny"},
mcp={
"demo": {
"type": "local",
"command": ["npx", "-y", "@modelcontextprotocol/server-everything"],
"enabled": True,
}
},
)
result = await client.async_run(
"Summarize the README in one sentence.",
Path("/path/to/repo"),
run_cfg=cfg,
timeout_s=600,
)
print(result.exit_code, result.final_text)
asyncio.run(main())
Set RunConfig(record_thinking=True) when you want OpenCode reasoning/thinking
parts included in result.events and log_file JSON lines. This only maps to
OpenCode's display/output flag --thinking; it does not change model reasoning
effort. Use variant separately if you intentionally want a provider-specific
reasoning effort.
Stream structured JSON events
async def stream_example():
client = AsyncOpenCodeClient()
cfg = RunConfig(permission={"*": "allow"})
async for event in client.async_stream("List top-level files.", workspace=".", run_cfg=cfg):
print(event)
Parallel agents (asyncio.gather)
async def multi():
# startup_concurrency=1 serialises SQLite initialisation to avoid a known
# WAL-pragma race in opencode when many instances start simultaneously.
# startup_delay_s controls how long each slot is held before the next
# process is allowed to start (default 0.3 s).
client = AsyncOpenCodeClient(startup_concurrency=1, startup_delay_s=0.3)
ws = Path("/path/to/monorepo")
results = await asyncio.gather(*[
client.async_run(
f"Explain services/{svc}.",
ws / "services" / svc,
run_cfg=RunConfig(agent="explore"),
timeout_s=600,
# max_retries=2 (default): retry automatically if opencode crashes
# during SQLite startup before giving up.
)
for svc in ["api", "worker", "gateway"]
])
return results
Configuration injection
Per-call JSON is merged and passed as OPENCODE_CONFIG_CONTENT (see OpenCode config). Use RunConfig fields:
| Field | Purpose |
|---|---|
permission |
permission map (allow / deny, patterns) |
mcp |
MCP server definitions |
tools |
Enable/disable tools (including MCP globs) |
instructions |
Instruction file paths / glob patterns to inject |
config_overrides |
Any extra top-level config keys to deep-merge |
Optional env tuning: disable_autoupdate=True sets OPENCODE_DISABLE_AUTOUPDATE=1.
Note: ask is intentionally rejected in subprocess mode (no interactive terminal); use allow or deny.
User config isolation
By default, RunConfig.inherit_user_config=False makes each child opencode
process see a sanitized copy of the host's global OpenCode config. The wrapper
keeps only provider-selection keys ($schema, provider,
disabled_providers, enabled_providers) and drops capability/configuration
keys such as mcp, agent, command, tools, plugin, skills,
instructions, permission, and model.
This keeps benchmark and orchestration runs reproducible while still allowing
provider configuration and opencode auth credentials to work. Project-level
config discovered from the workspace is not suppressed.
Set inherit_user_config=True to restore the legacy behavior of inheriting the
host OpenCode config as-is. For reproducible runs, pass model, permission,
mcp, tools, and instructions explicitly through RunConfig.
CLI arguments
RunConfig maps to flags such as --agent, -m, -f, --attach, --title, etc. Prompt text is appended as the final opencode run message argument.
Tests
Unit tests (no real OpenCode / no API calls):
pytest -q -m "not integration"
Integration tests (real opencode run, needs working provider auth — slow, may incur API usage):
pytest -m integration -q tests/test_integration_opencode.py
Multi-agent weather workflow (10 parallel city lookups + 1 summary — 11 API calls, not run by default):
OPENCODE_MULTI_AGENT_WEATHER=1 pytest -m integration -v tests/test_integration_multi_agent_weather.py
Optional: OPENCODE_WEATHER_SEQUENTIAL=1 runs the 10 city calls one-by-one (easier on rate limits).
Per-stage timeouts: OPENCODE_WEATHER_PER_CITY_TIMEOUT_S, OPENCODE_WEATHER_SUMMARY_TIMEOUT_S (default: same as OPENCODE_INTEGRATION_TIMEOUT_S).
| Env | Meaning |
|---|---|
OPENCODE_BINARY |
Absolute path to opencode if not on PATH |
OPENCODE_INTEGRATION=0 |
Skip integration tests |
OPENCODE_INTEGRATION_TIMEOUT_S |
Per-test timeout seconds (default 300) |
OPENCODE_MULTI_AGENT_WEATHER=1 |
Enable 11-call weather integration test |
OPENCODE_ENABLE_EXA |
Passed through / defaulted to 1 in that test for web search tools |
Default pytest -q runs all tests; use -m "not integration" in CI without OpenCode.
Concurrency notes
When running many tasks with asyncio.gather, three protections are enabled by default:
Startup serialisation — startup_concurrency=1 and startup_delay_s=0.3 limit how many processes enter SQLite startup at once, reducing WAL-initialisation race crashes.
DB isolation — isolate_db=True gives each run a private XDG_DATA_HOME, so concurrent runs do not contend on the same opencode.db during tool execution.
Automatic retry — async_run(max_retries=2, retry_delay_s=1.0) retries known SQLite-startup crashes with short backoff. Non-SQLite failures still fail fast.
Set startup_concurrency=0, isolate_db=False, and max_retries=0 to opt out.
Notes
- Event shapes from
--format jsonmay change between OpenCode versions; unknown fields are preserved in each parsed dict. - For fully non-interactive automation, prefer explicit
permission(allow/deny) over relying on interactiveaskprompts.
Project details
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 py_opencode_wrapper-0.2.2.tar.gz.
File metadata
- Download URL: py_opencode_wrapper-0.2.2.tar.gz
- Upload date:
- Size: 31.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 |
6960970d4c04564165707b5df3affe398fa3bc5f25f80de481e707c111ec2982
|
|
| MD5 |
210726148fd62d645d8eecfdc0966dd9
|
|
| BLAKE2b-256 |
44e14b9d49d3897007cc8a50243ca33aa79abde6964b4114360be959d868cc4b
|
Provenance
The following attestation bundles were made for py_opencode_wrapper-0.2.2.tar.gz:
Publisher:
release.yml on idailylife/oc_py_wrapper
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
py_opencode_wrapper-0.2.2.tar.gz -
Subject digest:
6960970d4c04564165707b5df3affe398fa3bc5f25f80de481e707c111ec2982 - Sigstore transparency entry: 1543717886
- Sigstore integration time:
-
Permalink:
idailylife/oc_py_wrapper@284141bfcce125edd28d5bced2d999ef690df979 -
Branch / Tag:
refs/tags/v0.2.2 - Owner: https://github.com/idailylife
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@284141bfcce125edd28d5bced2d999ef690df979 -
Trigger Event:
release
-
Statement type:
File details
Details for the file py_opencode_wrapper-0.2.2-py3-none-any.whl.
File metadata
- Download URL: py_opencode_wrapper-0.2.2-py3-none-any.whl
- Upload date:
- Size: 16.7 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 |
b9a1bd3d607b1ae172f2da6a84a1fdbe642f38954f2c5dfc35d736c3eb3adba6
|
|
| MD5 |
32d9f14be7ed6d178203a1b898b06c33
|
|
| BLAKE2b-256 |
d357d13ba50ace4d7734e9cdc64d0084ea35294af580daabdea29fea85aaf50c
|
Provenance
The following attestation bundles were made for py_opencode_wrapper-0.2.2-py3-none-any.whl:
Publisher:
release.yml on idailylife/oc_py_wrapper
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
py_opencode_wrapper-0.2.2-py3-none-any.whl -
Subject digest:
b9a1bd3d607b1ae172f2da6a84a1fdbe642f38954f2c5dfc35d736c3eb3adba6 - Sigstore transparency entry: 1543718001
- Sigstore integration time:
-
Permalink:
idailylife/oc_py_wrapper@284141bfcce125edd28d5bced2d999ef690df979 -
Branch / Tag:
refs/tags/v0.2.2 - Owner: https://github.com/idailylife
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
release.yml@284141bfcce125edd28d5bced2d999ef690df979 -
Trigger Event:
release
-
Statement type: