32b62cbacc
Removes the legacy `CLAUDE_BOTTLE_OAUTH_TOKEN` -> `CLAUDE_CODE_OAUTH_TOKEN` forward in prepare.py. Bottles that need claude-code to authenticate must declare a cred_proxy route with role: "anthropic-base-url" — there is no fallback that hands the token to the agent directly. Drops the now-dead BottleSpec.forward_oauth_token field, the CLI setter that read CLAUDE_BOTTLE_OAUTH_TOKEN from the host env at prepare time, and the forward_oauth_token=False arg in the six pipelock integration tests. PRD 0010 and README updated; the dev ~/claude-bottle.json gains an anthropic-base-url route so the implementer/researcher agents keep working. BREAKING: bottles previously relying on the implicit OAuth forward will now produce an agent environ without any Anthropic credential. Verified with --dry-run: a bottle with no anthropic-base-url route yields env_names: [] (no token at all); a bottle that declares the route yields ANTHROPIC_BASE_URL plus a non-secret placeholder for CLAUDE_CODE_OAUTH_TOKEN.
103 lines
4.1 KiB
Python
103 lines
4.1 KiB
Python
"""Integration: with pipelock's tls_interception enabled (PRD 0006),
|
|
a credential POST sent over HTTPS is blocked by pipelock's body-scan
|
|
layer — closing the gap that motivated this PRD.
|
|
|
|
End-to-end: drives `BottleBackend.prepare → launch` so the real
|
|
image build, network plumbing, pipelock_tls_init, sidecar bring-up,
|
|
and provision_ca (CA install in the agent's trust store) are all in
|
|
the loop. The probe is a single `curl --proxy "$HTTPS_PROXY" -X POST
|
|
... https://raw.githubusercontent.com/...` — curl natively does
|
|
CONNECT through the proxy, the agent's trust store now contains
|
|
pipelock's per-bottle CA so curl trusts pipelock's bumped leaf, and
|
|
pipelock sees the decrypted body and returns its known
|
|
`blocked: request body contains secret: <pattern>` 403.
|
|
|
|
The host has to be allowlisted (so the CONNECT is accepted) but NOT
|
|
in `tls_interception.passthrough_domains` (so the body actually gets
|
|
scanned). `api.anthropic.com` is passthrough'd to skip MITM on the
|
|
LLM endpoint, so this probe targets `raw.githubusercontent.com` —
|
|
also on the baked allowlist (Claude Code fetches release assets from
|
|
it) and intercepted+scanned like any non-passthrough host."""
|
|
|
|
from __future__ import annotations
|
|
|
|
import os
|
|
import shutil
|
|
import tempfile
|
|
import unittest
|
|
from pathlib import Path
|
|
|
|
from claude_bottle.backend import BottleSpec, get_bottle_backend
|
|
from claude_bottle.manifest import Manifest
|
|
from tests._docker import skip_unless_docker
|
|
|
|
|
|
# Synthetic value shaped like a GitHub Personal Access Token; not a
|
|
# real credential. Carried into the bottle as an env var so the
|
|
# probe shell can read it via $FAKE_TOKEN without ever interpolating
|
|
# the value on the bash `bottle.exec` argv.
|
|
_FAKE_TOKEN = "ghp_aB3cD4eF5gH6iJ7kL8mN9oP0qR1sT2uV3wX4yZ"
|
|
|
|
|
|
@skip_unless_docker()
|
|
class TestPipelockBlocksSecretHttpsPost(unittest.TestCase):
|
|
@unittest.skipIf(
|
|
os.environ.get("GITEA_ACTIONS") == "true",
|
|
"skipped under act_runner: docker socket mount topology breaks "
|
|
"in-process visibility of networks created on the host daemon",
|
|
)
|
|
def test_https_post_with_credential_body_is_blocked(self):
|
|
manifest = Manifest.from_json_obj({
|
|
"bottles": {
|
|
"dev": {"env": {"FAKE_TOKEN": _FAKE_TOKEN}},
|
|
},
|
|
"agents": {
|
|
"demo": {"skills": [], "prompt": "", "bottle": "dev"},
|
|
},
|
|
})
|
|
backend = get_bottle_backend()
|
|
stage_dir = Path(tempfile.mkdtemp(prefix="cb-test-stage."))
|
|
try:
|
|
spec = BottleSpec(
|
|
manifest=manifest,
|
|
agent_name="demo",
|
|
copy_cwd=False,
|
|
user_cwd=str(stage_dir),
|
|
)
|
|
plan = backend.prepare(spec, stage_dir=stage_dir)
|
|
with backend.launch(plan) as bottle:
|
|
script = (
|
|
"set -eu\n"
|
|
'curl --proxy "$HTTPS_PROXY" -s --max-time 8 \\\n'
|
|
" -w 'status=%{http_code}\\n' \\\n"
|
|
" -o /tmp/probe-body.txt \\\n"
|
|
' -X POST -d "token=$FAKE_TOKEN" \\\n'
|
|
" https://raw.githubusercontent.com/dlp-probe\n"
|
|
'echo "body=$(head -c 200 /tmp/probe-body.txt)"\n'
|
|
)
|
|
result = bottle.exec(script)
|
|
finally:
|
|
shutil.rmtree(stage_dir, ignore_errors=True)
|
|
|
|
self.assertEqual(
|
|
0, result.returncode,
|
|
f"exec wrapper failed: stdout={result.stdout!r} stderr={result.stderr!r}",
|
|
)
|
|
# Pipelock's body-scan block returns 403 with a plain-text
|
|
# body starting `blocked: ` (pinned empirically; see
|
|
# tests/unit/test_mitmproxy_verdict.py for the
|
|
# corresponding-fingerprint test, retained from PR #8 as
|
|
# general pipelock-block-shape coverage).
|
|
self.assertIn(
|
|
"status=403", result.stdout,
|
|
f"expected 403 from pipelock; got: {result.stdout!r}",
|
|
)
|
|
self.assertIn(
|
|
"body=blocked: ", result.stdout,
|
|
f"expected pipelock block body; got: {result.stdout!r}",
|
|
)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
unittest.main()
|