ECI-backed sandbox manager with an Agent interface.
Project description
English | 简体中文
eci-as-sandbox
ECI-backed sandbox manager built on alibabacloud-eci20180808 with a lightweight API for managing container-group sandboxes.
Install
pdm add eci-as-sandbox
Environment
The client auto-loads a .env file (searching upward from the working directory). You can also pass env_file=... or explicit credentials.
Required Alibaba Cloud credentials:
set ALIBABA_CLOUD_ACCESS_KEY_ID=...
set ALIBABA_CLOUD_ACCESS_KEY_SECRET=...
set ALIBABA_CLOUD_REGION_ID=cn-shanghai
Optional overrides:
set ECI_SANDBOX_ACCESS_KEY_ID=...
set ECI_SANDBOX_ACCESS_KEY_SECRET=...
set ECI_SANDBOX_REGION_ID=cn-shanghai
set ECI_SANDBOX_ENDPOINT=eci.cn-shanghai.aliyuncs.com
set ECI_SANDBOX_TIMEOUT_MS=60000
If you use a private image, configure registry credentials in ECI.
Quick start (sync)
from eci_as_sandbox import EciSandbox
client = EciSandbox()
result = client.create(
image="registry.cn-hangzhou.aliyuncs.com/eci_open/nginx:latest",
name="sandbox-demo",
cpu=1.0,
memory=2.0,
v_switch_id="vsw-xxx",
security_group_id="sg-xxx",
ports=[{"port": 8080, "protocol": "TCP"}],
)
if result.success and result.sandbox:
sandbox = result.sandbox
info = sandbox.info()
print(info.data.status)
sandbox.delete()
Async quick start
import asyncio
from eci_as_sandbox import AsyncEciSandbox
async def main() -> None:
client = AsyncEciSandbox()
result = await client.create(
image="registry.cn-hangzhou.aliyuncs.com/eci_open/nginx:latest",
name="sandbox-demo",
cpu=1.0,
memory=2.0,
v_switch_id="vsw-xxx",
security_group_id="sg-xxx",
)
if result.success and result.sandbox:
info = await result.sandbox.info()
print(info.data.status)
asyncio.run(main())
Command helpers
exec_command runs a list-form command. With sync=True (default) it uses the ECI WebSocket stream to collect output. timeout is in seconds and caps the stream duration (default 600 seconds, max 600). Use sync=False if you want the WebSocket/HTTP URLs without waiting.
result = sandbox.exec_command(
["/bin/sh", "-c", "for i in 1 2 3; do echo tick-$i; sleep 2; done"],
timeout=3,
)
print(result.output)
bash runs a shell command via bash -lc and supports exec_dir.
result = sandbox.bash(
command="pwd; ls -la",
exec_dir="/tmp",
timeout=5,
)
print(result.output)
Listing
result = client.list(limit=10, status="Running")
print(result.sandbox_ids)
Long command execution (WebSocket)
ECI's API has a 2048-byte command limit. For longer commands, use bash_ws which sends commands through WebSocket stdin (no length limit).
# Execute a very long command (e.g., inline Python script)
long_script = """python3 << 'EOF'
import json
# ... hundreds of lines of code ...
print(json.dumps({"status": "done"}))
EOF
"""
result = client.bash_ws(
sandbox_id=sandbox_id,
command=long_script,
exec_dir="/workspace",
timeout=60,
)
print(result.output)
write_file_ws writes large content to a file via WebSocket:
result = client.write_file_ws(
sandbox_id=sandbox_id,
file_path="/tmp/large_script.py",
content="# Very long file content...\n" * 1000,
timeout=30,
)
Tmux session management
For non-blocking command execution with output capture, use tmux methods. Long commands are automatically handled via WebSocket file transfer.
# Start a command in tmux (non-blocking)
start_result = client.tmux_start(
sandbox_id=sandbox_id,
command="python train.py --epochs 100",
exec_dir="/workspace",
)
print(f"Session: {start_result.session_id}")
# Poll for status and partial output
poll_result = client.tmux_poll(
sandbox_id=sandbox_id,
session_id=start_result.session_id,
)
print(f"Status: {poll_result.status}") # RUNNING, COMPLETED, NOT_FOUND
print(f"Output: {poll_result.output}")
# Wait for completion with exponential backoff
wait_result = client.tmux_wait(
sandbox_id=sandbox_id,
session_id=start_result.session_id,
timeout=300, # max wait time
)
print(f"Exit code: {wait_result.exit_code}")
print(f"Output: {wait_result.output}")
# Kill a session manually
client.tmux_kill(sandbox_id=sandbox_id, session_id=start_result.session_id)
# List all tmux sessions
list_result = client.tmux_list(sandbox_id=sandbox_id)
print(list_result.data) # [{"session_id": "...", "created": "...", "attached": False}]
API Reference
Client Methods
| Method | Description |
|---|---|
create(image, name, cpu, memory, ...) |
Create a new sandbox container |
get(sandbox_id) |
Get sandbox instance by ID |
get_sandbox(sandbox_id) |
Get sandbox info |
list(limit, status, tags, ...) |
List sandboxes |
delete(sandbox_id, force) |
Delete a sandbox |
restart(sandbox_id) |
Restart a sandbox |
exec_command(sandbox_id, command, ...) |
Execute command (list form) |
bash(sandbox_id, command, exec_dir, ...) |
Execute bash command |
bash_ws(sandbox_id, command, exec_dir, ...) |
Execute bash via WebSocket (unlimited length) |
write_file_ws(sandbox_id, file_path, content, ...) |
Write file via WebSocket (unlimited length) |
tmux_start(sandbox_id, command, ...) |
Start command in tmux session |
tmux_poll(sandbox_id, session_id, ...) |
Poll tmux session status |
tmux_wait(sandbox_id, session_id, timeout, ...) |
Wait for tmux session completion |
tmux_kill(sandbox_id, session_id) |
Kill tmux session |
tmux_list(sandbox_id) |
List all tmux sessions |
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 eci_as_sandbox-0.2.1.tar.gz.
File metadata
- Download URL: eci_as_sandbox-0.2.1.tar.gz
- Upload date:
- Size: 28.0 kB
- Tags: Source
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
e8a35e8f43b480f1c5e4ff75fc41d0f85d885769bb9b7a2ddaff9d7fa108f618
|
|
| MD5 |
ecb9e0bf78bc3f020891196077e81535
|
|
| BLAKE2b-256 |
a971f7bc9ec909180b48f05a9eea613f5b6a38b75f94a5e9208810009b826fa3
|
Provenance
The following attestation bundles were made for eci_as_sandbox-0.2.1.tar.gz:
Publisher:
publish.yml on AndersonBY/eci-as-sandbox
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
eci_as_sandbox-0.2.1.tar.gz -
Subject digest:
e8a35e8f43b480f1c5e4ff75fc41d0f85d885769bb9b7a2ddaff9d7fa108f618 - Sigstore transparency entry: 814798922
- Sigstore integration time:
-
Permalink:
AndersonBY/eci-as-sandbox@60fb0668493cae104a278b515dfa06ab06e9c4a4 -
Branch / Tag:
refs/tags/v0.2.1 - Owner: https://github.com/AndersonBY
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@60fb0668493cae104a278b515dfa06ab06e9c4a4 -
Trigger Event:
push
-
Statement type:
File details
Details for the file eci_as_sandbox-0.2.1-py3-none-any.whl.
File metadata
- Download URL: eci_as_sandbox-0.2.1-py3-none-any.whl
- Upload date:
- Size: 31.3 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? Yes
- Uploaded via: twine/6.1.0 CPython/3.13.7
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
bc0a4beb3cd486a3e3bf859208322bca73efae3c2084aabe8f927642be30ce6d
|
|
| MD5 |
ee1a83b43871118acac0dd85239f1120
|
|
| BLAKE2b-256 |
2553da022677615266539726114e84be039b5d004c7832bc79e29813589d80f0
|
Provenance
The following attestation bundles were made for eci_as_sandbox-0.2.1-py3-none-any.whl:
Publisher:
publish.yml on AndersonBY/eci-as-sandbox
-
Statement:
-
Statement type:
https://in-toto.io/Statement/v1 -
Predicate type:
https://docs.pypi.org/attestations/publish/v1 -
Subject name:
eci_as_sandbox-0.2.1-py3-none-any.whl -
Subject digest:
bc0a4beb3cd486a3e3bf859208322bca73efae3c2084aabe8f927642be30ce6d - Sigstore transparency entry: 814798927
- Sigstore integration time:
-
Permalink:
AndersonBY/eci-as-sandbox@60fb0668493cae104a278b515dfa06ab06e9c4a4 -
Branch / Tag:
refs/tags/v0.2.1 - Owner: https://github.com/AndersonBY
-
Access:
public
-
Token Issuer:
https://token.actions.githubusercontent.com -
Runner Environment:
github-hosted -
Publication workflow:
publish.yml@60fb0668493cae104a278b515dfa06ab06e9c4a4 -
Trigger Event:
push
-
Statement type: