scitex_agent_container API Reference
Top-level package surface re-exported from scitex_agent_container.
Use scitex-agent-container list-python-apis for the authoritative
runtime enumeration.
SciTeX Agent Container – Declarative agent management.
Provides a YAML-based framework for defining, managing, and orchestrating AI coding agent instances across container runtimes.
Public surface — CLI-tree-shaped noun submodules:
import scitex_agent_container as sac
sac.agent.list() # `sac agent list`
sac.agent.start("head-nas") # `sac agent start head-nas`
sac.db.query(table="instances") # `sac db query --table=instances`
sac.host.list() # `sac host list`
sac.skills.get("02_quick-start") # `sac skills get 02_quick-start`
Each noun submodule (agent, db, host, image, template, account, skills, mcp) re-exports its verbs under bare names that mirror the CLI subcommand tree. The same function objects power both the Python API and the MCP server (per scitex MCP §6 parity).
Lifecycle helpers that take a shared Registry instance live at
scitex_agent_container._lifecycle.lifecycle for callers that
need them. The submodule verbs go through the CLI for JSON-friendly
input/output.
Config
YAML config loading and validation for agent definitions.
- Public API:
AgentConfig, load_config, validate_config, resolve_config ContainerSpec, ClaudeSpec, HealthSpec, WatchdogSpec, RestartSpec, SkillsSpec, StartupCommand
RemoteSpec and the spec.remote field were deleted in WI-6
(handoff §6, 2026-05-20). Cross-host placement is via spec.host;
the old SSH-dispatch path is retired.
- class scitex_agent_container.config.AgentConfig(name, runtime='apptainer', image='', model='sonnet', workdir='', python_venv='', env=<factory>, env_files=<factory>, screen_name='', labels=<factory>, container=<factory>, claude=<factory>, health=<factory>, watchdog=<factory>, restart=<factory>, autonomous=<factory>, apptainer=<factory>, hooks=<factory>, listen=<factory>, extensions=<factory>, skills=<factory>, context_management=<factory>, startup_commands=<factory>, startup_prompts=<factory>, mcp_servers=<factory>, multiplexer='tmux', hosts_spec=<factory>, scheduling=<factory>, config_path='', mounts=<factory>, user='', a2a=<factory>, comms=<factory>, lineage=<factory>, kind='Agent', proxy=None, to_home='./to_home')[source]
Bases:
objectParsed agent configuration from a YAML definition file.
- container: ContainerSpec
- claude: ClaudeSpec
- health: HealthSpec
- watchdog: WatchdogSpec
- restart: RestartSpec
- autonomous: AutonomousSpec
- apptainer: ApptainerSpec
- listen: list[ListenPort]
- skills: SkillsSpec
- context_management: ContextManagementConfig
- startup_commands: list[StartupCommand]
- __init__(name, runtime='apptainer', image='', model='sonnet', workdir='', python_venv='', env=<factory>, env_files=<factory>, screen_name='', labels=<factory>, container=<factory>, claude=<factory>, health=<factory>, watchdog=<factory>, restart=<factory>, autonomous=<factory>, apptainer=<factory>, hooks=<factory>, listen=<factory>, extensions=<factory>, skills=<factory>, context_management=<factory>, startup_commands=<factory>, startup_prompts=<factory>, mcp_servers=<factory>, multiplexer='tmux', hosts_spec=<factory>, scheduling=<factory>, config_path='', mounts=<factory>, user='', a2a=<factory>, comms=<factory>, lineage=<factory>, kind='Agent', proxy=None, to_home='./to_home')
- scheduling: SchedulingSpec
- a2a: A2ASpec
- comms: CommsSpec
- lineage: LineageSpec
- class scitex_agent_container.config.ClaudeSpec(model='', channels=<factory>, flags=<factory>, raw_options=<factory>, session='continue', continue_max_age_minutes=None, resume_id='', auto_accept=True, account='', provider=None)[source]
Bases:
object- provider: ProviderSpec | None = None
- __init__(model='', channels=<factory>, flags=<factory>, raw_options=<factory>, session='continue', continue_max_age_minutes=None, resume_id='', auto_accept=True, account='', provider=None)
- class scitex_agent_container.config.ContainerSpec(runtime='none', image='scitex-agent-container:latest', volumes=<factory>, network='host', mount_host_claude=False)[source]
Bases:
object- __init__(runtime='none', image='scitex-agent-container:latest', volumes=<factory>, network='host', mount_host_claude=False)
- class scitex_agent_container.config.ContextManagementConfig(trigger_at_percent=70.0, strategy='noop', warn_before_n_checks=0, check_interval_seconds=300, state_file='~/.scitex/agent-container/state/<agent>.json')[source]
Bases:
objectContext-lifecycle policy for an agent.
Defaults mirror
strategy="noop"so absence of thecontext_managementblock preserves existing behavior (sensor disabled).- __init__(trigger_at_percent=70.0, strategy='noop', warn_before_n_checks=0, check_interval_seconds=300, state_file='~/.scitex/agent-container/state/<agent>.json')
- class scitex_agent_container.config.HealthSpec(enabled=False, interval=30, timeout=5, method='multiplexer-alive')[source]
Bases:
object- __init__(enabled=False, interval=30, timeout=5, method='multiplexer-alive')
- class scitex_agent_container.config.HookSpec(pre_start=<factory>, post_start=<factory>, pre_stop=<factory>, post_stop=<factory>, on_compact=<factory>, on_restart=<factory>, on_diff=<factory>)[source]
Bases:
objectAll hook points supported by the container.
Each entry is a list of opaque commands — shell strings or http(s) URLs. The container executes them fire-and-forget; errors are logged but never raised to the caller. Absent keys default to empty lists (feature disabled).
- __init__(pre_start=<factory>, post_start=<factory>, pre_stop=<factory>, post_stop=<factory>, on_compact=<factory>, on_restart=<factory>, on_diff=<factory>)
- class scitex_agent_container.config.HostsSpec(host='', hosts=<factory>)[source]
Bases:
objectWhere an agent should run, in either singleton or multi-instance form.
Mutually exclusive — exactly one of
hostorhostsmay be set:host(singular) — exactly one instance runs:empty / absent: local singleton (runs wherever sac is invoked)
string: pinned to that host
list: priority order; first available host wins (fallback chain)
hosts(plural) — multiple instances run, one per host:“all”: one per fleet host (replaces the old per-host mode)
list of host names: one per listed host (subset)
Validator (in
_validation.py) enforces mutual exclusion + types. Loader composes effective ids:hoststriggers the<name>-<HOST>suffix;hostkeeps the bare name.- __init__(host='', hosts=<factory>)
- class scitex_agent_container.config.ListenPort(port=0, proto='tcp', path='', name='', owner='')[source]
Bases:
objectDeclaration of a port/socket an external tool binds on behalf of an agent.
The container NEVER binds these — it just validates the shape and echoes them in
status --jsonso orchestrators can see what sidecars are expected to exist.owneris free-form (e.g."orochi") to identify the plugin that actually listens.- __init__(port=0, proto='tcp', path='', name='', owner='')
- class scitex_agent_container.config.ProviderSpec(base_url='', auth_token_env='')[source]
Bases:
objectVendor-agnostic backend override for the Claude SDK session.
When set under
spec.claude.provider, the agent’s SDK session talks to an Anthropic-SDK-compatible backend OTHER than Anthropic (DeepSeek, a self-hosted gateway, etc.) using a never-expiring, login-free API key instead of Anthropic OAuth. Lets bulk fleet work run on a cheap backend without burning Max-plan quota.The runtime injects three env vars into the container at start (see
runtimes/_apptainer_provider.py):ANTHROPIC_BASE_URL←base_urlSAC_ANTHROPIC_API_KEY← the host value of$<auth_token_env>(bridged toANTHROPIC_API_KEYfor the SDK by sac’s existing auth handoff). Fails loud at start if the env var is unset.CLAUDE_CONFIG_DIR← a clean per-agent dir — the conflict-breaker so the OAuth.credentials.jsonbind cannot win (apptainer--envis last-wins).
The model id is the provider’s own (e.g.
deepseek-chat); the claude-* regex inconfig._validationis relaxed whenever a provider is declared.Mutually exclusive with
ClaudeSpec.account(OAuth pin) — an API-key backend needs no OAuth, so declaring both is a config error the runtime rejects loudly.- base_url: str = ''
Anthropic-compatible base URL, e.g.
https://api.deepseek.com/anthropic. Required when the provider block is present.
- __init__(base_url='', auth_token_env='')
- class scitex_agent_container.config.ProxySpec(upstream='', trust='untrusted', redact=<factory>, timeout_s=30.0)[source]
Bases:
objectConfiguration for kind: AgentProxy agents.
- upstream: str = ''
REQUIRED. Full URL to the upstream A2A AgentCard endpoint. Either an explicit .well-known path or a base URL (we’ll fetch
<base>/.well-known/agent-card.jsonif a base is given).
- trust: str = 'untrusted'
One of
untrusted(default — operator must opt in to anything more permissive),local-mesh(peers on the same private network you control),trusted(cryptographically verified — reserved for future mTLS work). Surfaced on the AgentCard’sx-scitex-agent-container.trustfield.
- redact: list[str]
Substring tokens; any inbound text field containing one is refused with HTTP 400. Cheap defense-in-depth against accidentally forwarding secrets to an untrusted upstream — NOT a substitute for proper output filtering at the source.
- __init__(upstream='', trust='untrusted', redact=<factory>, timeout_s=30.0)
- class scitex_agent_container.config.RestartSpec(policy='never', max_retries=3, backoff_initial=30, backoff_max=300, backoff_multiplier=2)[source]
Bases:
object- __init__(policy='never', max_retries=3, backoff_initial=30, backoff_max=300, backoff_multiplier=2)
- class scitex_agent_container.config.SchedulingSpec(mode='per-host', preferred_host='', fallback_hosts=<factory>)[source]
Bases:
objectFleet-wide scheduling policy for an agent (shared-host layout).
modecontrols effective-id composition and launch-skip behavior:per-host(default): agent is started on every host that runssac agent start <name>; the effective id is<metadata.name>-<HOST>unless the name already ends with-<HOST>.singleton: exactly one instance fleet-wide. The effective id stays as the bare<metadata.name>. Only launched onpreferred-host; on other hosts the launch is a no-op.
fallback-hostsis recorded for observability but not acted on automatically — manual failover today.- __init__(mode='per-host', preferred_host='', fallback_hosts=<factory>)
- class scitex_agent_container.config.SkillsSpec(required=<factory>, available=<factory>, injection_mode='at-import', match_by=<factory>, match_style='exact')[source]
Bases:
object- __init__(required=<factory>, available=<factory>, injection_mode='at-import', match_by=<factory>, match_style='exact')
- class scitex_agent_container.config.StartupCommand(delay=0, command='')[source]
Bases:
object- __init__(delay=0, command='')
- class scitex_agent_container.config.WatchdogSpec(enabled=False, interval=1.5, resp_y_n='1', resp_y_y_n='2', resp_waiting='/speak-and-call')[source]
Bases:
object- __init__(enabled=False, interval=1.5, resp_y_n='1', resp_y_y_n='2', resp_waiting='/speak-and-call')
- scitex_agent_container.config.compose_effective_name(raw_name, hosts_spec, hostname)[source]
Return the effective agent id given dir-derived name + host/hosts + host.
- Return type:
- Rules:
If
hosts:is set (multi-instance), append-<hostname>so each host’s instance has a unique id. Idempotent — names that already end with-<hostname>are not double-suffixed.Otherwise (
host:set, or both empty = local singleton): keep the bareraw_name. Singleton id stays stable across hosts.
- scitex_agent_container.config.load_config(path)[source]
Load and validate a YAML config, returning an AgentConfig.
Only
scitex-agent-container/v3is accepted. Older apiVersions (v1, v2) raise loud validation errors — no backward compatibility.- Return type:
- scitex_agent_container.config.resolve_config(name_or_path)[source]
Resolve agent name or path to a config file path.
Every agent lives in its own directory; the config file is always named
spec.yaml(orspec.yml). The flat-form<base>/<name>.yamland the legacy<base>/<name>/<name>.yamlare no longer accepted — sac is dir-as-SSoT.Search order for short names (no slash, no .yaml/.yml suffix):
Project-local — first
.scitex/agent-container/agents/found walking upward from cwd. Highest priority so checked-in test agents and CI fixtures override globals.~/.scitex/agent-container/agents/<name>/spec.yaml(sac install root).$SCITEX_AGENT_CONTAINER_YAML_DIRS(colon-separated extra dirs, each searched as<dir>/<name>/spec.yaml). Plugin port for downstream orchestrators to inject extra paths without sac knowing about them.
Pass an explicit path (with / or .yaml/.yml) to bypass the search entirely.
- Return type:
- scitex_agent_container.config.resolve_hostname(gethostname=<built-in function gethostname>)[source]
Return the canonical host label for this machine.
- Resolution order (first non-empty wins):
SCITEX_AGENT_CONTAINER_HOSTNAMEenv var (manual override).SCITEX_AGENT_CONTAINER_HOSTNAMEenv var.hostname_aliases[short hostname]fromshared/config.yamlor~/.scitex/agent-container/config.yaml.socket.gethostname()short form (identity fallback).
- Parameters:
gethostname (
Callable[[],str]) – Callable returning the raw OS hostname. Defaults tosocket.gethostname(production). Tests inject a callable returning a fixed string instead of patchingsocket.- Raises:
RuntimeError – If none of the sources produces a non-empty value. This should be practically impossible (
gethostname()returns something on any configured box) but is handled loudly rather than returning the empty string.- Return type:
Runtimes
Modular TUI prompt detection and response for Claude Code.
Each prompt handler defines: - name: identifier for logging - detect(content) -> bool: whether this prompt is visible - respond(send_keys) -> None: keystrokes to accept the prompt - priority: lower = checked first (default 10)
Add new handlers by appending to PROMPT_HANDLERS or calling register_prompt().
- class scitex_agent_container.runtimes.prompts.PromptHandler(name, detect, keys=<factory>, priority=10)[source]
Bases:
objectA single TUI prompt detector and responder.
- __init__(name, detect, keys=<factory>, priority=10)
- scitex_agent_container.runtimes.prompts._detect_bypass_permissions(content)[source]
Bypass Permissions mode prompt with radio selector.
- Return type:
- Matches:
“1. No, exit” “2. Yes, I accept” “Bypass Permissions” “Enter to confirm”
- scitex_agent_container.runtimes.prompts._detect_dev_channels(content)[source]
Development channels loading confirmation.
- Return type:
- Matches:
“1. I am using this for local development” “2. Exit” “development channels” or “dangerously-load-development-channels” “Enter to confirm”
- scitex_agent_container.runtimes.prompts._detect_thinking_effort(content)[source]
Thinking effort level selector.
- Return type:
- Matches:
“1. * Medium (recommended)” or similar “thinking” in various casings “Enter to confirm”
- scitex_agent_container.runtimes.prompts._detect_skip_permissions_yn(content)[source]
Legacy y/n text prompt for skip-permissions (older Claude Code).
Matches text-based y/n prompts without radio selector.
- Return type:
- scitex_agent_container.runtimes.prompts._detect_mcp_json_edit(content)[source]
Permission prompt when Claude tries to edit .mcp.json (runtime).
Matches “1. Yes” / “1. Proceed” / “1. Allow” + “.mcp.json” + “Enter to confirm”.
- Return type:
- scitex_agent_container.runtimes.prompts._detect_press_enter_continue(content)[source]
Generic ‘Press Enter to continue’ runtime pause (context-window warning, etc).
Uses a strict last-5-lines window to avoid scrollback false positives (per pane-state-patterns.md: classify against last 5 visible lines only). Excluded: active tool calls and numbered radio selectors.
- Return type:
- scitex_agent_container.runtimes.prompts._detect_file_trust(content)[source]
‘Do you trust the files in this folder?’ prompt (first-run or new cwd).
May appear when –dangerously-skip-permissions was not propagated to a subshell. Matches the LEGACY y/n text variant; the new radio-selector variant is handled by
_detect_file_trust_radio().- Return type:
- scitex_agent_container.runtimes.prompts._detect_file_trust_radio(content)[source]
Radio-selector variant of the file-trust prompt.
Claude Code (>= ~2.1.x) asks “Is this a project you created or one you trust?” with numbered options instead of the legacy y/n text prompt. Appears on the first launch in any un-trusted workdir — including every throwaway tempdir the Haiku integration test uses.
Matches the exact option strings to avoid firing on the bypass-permissions dialog (which also says “Enter to confirm”).
- Return type:
- scitex_agent_container.runtimes.prompts._detect_external_imports(content)[source]
External CLAUDE.md file imports prompt.
Appears when
CLAUDE.md(or.claude/CLAUDE.md) contains@<absolute-path>imports pointing OUTSIDE the agent’s workdir. Triggered by the at-import skill-injection mode (sac PR #74) when skills live in~/.claude/skills/or the package source trees rather than the workspace itself.- Return type:
- Matches:
“Allow external CLAUDE.md file imports?” “1. Yes, allow external imports” “Enter to confirm”
- scitex_agent_container.runtimes.prompts._detect_login_method(content)[source]
First-run login-method picker on a fresh HOME.
Appears when Claude Code can’t find OAuth credentials at
~/.claude/.credentials.json. Even withANTHROPIC_API_KEYset in env, the 2.1.x CLI still asks which auth mode to use before it checks the env var. Blocks startup until dismissed.Matches the exact option strings to avoid false positives on any user message that happens to say “login method”.
- Return type:
- scitex_agent_container.runtimes.prompts._detect_theme_selection(content)[source]
First-run theme selection prompt.
Appears only on a fresh HOME (no
~/.claude/saved theme). On dev machines it never shows, but in CI (a clean ubuntu VM) this is the first thing Claude Code asks. Blocks every downstream startup prompt until acknowledged.Matches the radio variant: “Choose the text style…” + numbered options starting with “1. Auto (match terminal)”.
- Return type:
- scitex_agent_container.runtimes.prompts._detect_compose_pending_unsent(content)[source]
Detect unsent text sitting in the Claude Code compose buffer.
The classifier in
agent_meta._classify_pane_statematches❯[ \t]+\S(non-whitespace after the prompt marker on the same line), meaning the user has typed something but not yet pressed Enter. We mirror that pattern here so the prompts system can submit it via a plain Enter keystroke.Excluded: lines that are just the decorative separator below an empty prompt — those contain only whitespace after
❯.- Return type:
- scitex_agent_container.runtimes.prompts._detect_done(content)[source]
Check if claude is at the main input prompt (all TUI prompts done).
The status bar shows “bypass permissions” when ready.
- Return type:
- scitex_agent_container.runtimes.prompts.register_prompt(handler)[source]
Add a custom prompt handler to the registry.
- Return type: