strix/skills/tooling/python.md
Use exec_command for Python. There is no separate Strix Python executor.
Prefer writing reusable scripts to /workspace/scratch/<name>.py and
running them with python3 /workspace/scratch/<name>.py. For short
one-off transformations, python3 -c or a small here-document is fine.
The shell parameter on exec_command is for swapping POSIX shells
(bash/zsh/sh), not for picking interpreters. Put the interpreter
invocation in cmd instead: cmd="python3 -c '...'", not
shell=python3, cmd="...". The shell=<interpreter> shortcut breaks
in subtle ways — python3 works only with login=False (because the
SDK adds -l/-i), and other interpreters (node, ruby, perl)
take -e not -c so they fail even with login=False.
The sandbox image includes an installed caido_api module. Import it
explicitly when Python code needs Caido traffic or replay access:
from caido_api import (
list_requests,
list_sitemap,
repeat_request,
scope_rules,
view_request,
view_sitemap_entry,
)
All helpers are async. Use them inside asyncio.run(...) or an async
function:
import asyncio
from caido_api import list_requests, view_request
async def main():
posts = await list_requests(
httpql_filter='req.method.eq:"POST" AND req.path.cont:"/api/"',
first=50,
)
candidates = []
for edge in posts.edges:
request_id = edge.node.request.id
body = await view_request(request_id, part="request")
raw = body.request.raw.decode("utf-8", errors="replace")
if "id=" in raw or "user=" in raw:
candidates.append(request_id)
print(f"{len(candidates)} candidates")
print(candidates[:10])
asyncio.run(main())
Available helpers:
list_requests(httpql_filter=, first=50, after=, sort_by=, sort_order=, scope_id=) returns a cursor-paginated Caido SDK Connection.view_request(request_id, part="request") returns a Caido SDK request object with raw request/response bytes.repeat_request(request_id, modifications={...}) replays a captured request after modifying url, params, headers, body, or cookies.list_sitemap(scope_id=, parent_id=, depth="DIRECT", page=1) walks Caido's request-tree view of the discovered surface. Omit parent_id for root domains; pass an entry id with depth="DIRECT" or "ALL" to drill in.view_sitemap_entry(entry_id) returns one entry plus its 30 most recent related requests.scope_rules(action, allowlist=, denylist=, scope_id=, scope_name=) manages Caido scopes.For one-off arbitrary requests (e.g. probing a fresh endpoint, hitting an
external API), use exec_command with curl / httpx / requests. The
sandbox's HTTP_PROXY env routes all such traffic through Caido
automatically, so it shows up in list_requests and you can use
repeat_request to replay-and-modify any of it.
For iterative exploit work, put code in a file:
1. Create or edit `/workspace/scratch/exploit.py` with `apply_patch`.
2. Run it with `exec_command`: `python3 /workspace/scratch/exploit.py`.
3. Edit and rerun until the proof-of-concept is reliable.
The sandbox's Python lives in /app/.venv. To add a one-off dependency
for an exploit script, use uv (already in the image and much faster
than pip):
uv pip install --python /app/.venv/bin/python <package>