Compare commits
8 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 121dc84b9f | |||
| 2a67a85835 | |||
| 0bb47bd754 | |||
| ebbcae663c | |||
| fc6dd37dd9 | |||
| 33fe8d2c7a | |||
| 0db76b877a | |||
| 8006702ee7 |
@@ -9,6 +9,8 @@ on:
|
|||||||
- '.pylintrc'
|
- '.pylintrc'
|
||||||
- 'pyrightconfig.json'
|
- 'pyrightconfig.json'
|
||||||
- '.coveragerc'
|
- '.coveragerc'
|
||||||
|
# The core-coverage badge reads this list; refresh when it changes.
|
||||||
|
- 'scripts/critical-modules.txt'
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
|
|||||||
@@ -7,8 +7,8 @@
|
|||||||
[](https://gitea.dideric.is/didericis/bot-bottle/actions?workflow=test.yml)
|
[](https://gitea.dideric.is/didericis/bot-bottle/actions?workflow=test.yml)
|
||||||
[](https://github.com/PyCQA/pylint)
|
[](https://github.com/PyCQA/pylint)
|
||||||
[](https://github.com/microsoft/pyright)
|
[](https://github.com/microsoft/pyright)
|
||||||
[](https://coverage.readthedocs.io/)
|
[](https://coverage.readthedocs.io/)
|
||||||
[](https://gitea.dideric.is/didericis/bot-bottle/src/branch/main/docs/decisions/0004-coverage-policy.md)
|
[](https://gitea.dideric.is/didericis/bot-bottle/src/branch/main/docs/decisions/0004-coverage-policy.md)
|
||||||
|
|
||||||
**Problem:** Developer wants to run a coding agent without supervision, but they don't want a prompt injected or misbehaving agent wrecking their environment or exfiltrating sensitive data.
|
**Problem:** Developer wants to run a coding agent without supervision, but they don't want a prompt injected or misbehaving agent wrecking their environment or exfiltrating sensitive data.
|
||||||
|
|
||||||
|
|||||||
+38
-10
@@ -126,8 +126,30 @@ def redact_tokens(
|
|||||||
# Known secrets detector
|
# Known secrets detector
|
||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
|
|
||||||
|
# Encoded-variant cache. Provisioned secrets are stable for the life of the
|
||||||
|
# proxy, but `_encoded_variants` is on the per-request hot path — it runs for
|
||||||
|
# every secret on every redaction and known-secret scan (host, path, each
|
||||||
|
# header, body). Deriving the variant set is relatively expensive (gzip +
|
||||||
|
# nine encodings), so memoize it per distinct secret. The proxy process
|
||||||
|
# already holds these values in `os.environ`, so caching them here adds no
|
||||||
|
# new exposure.
|
||||||
|
_VARIANT_CACHE: dict[str, tuple[str, ...]] = {}
|
||||||
|
|
||||||
|
|
||||||
def _encoded_variants(secret: str) -> list[str]:
|
def _encoded_variants(secret: str) -> list[str]:
|
||||||
"""Return the secret plus common encoded variants for exfil detection."""
|
"""Return the secret plus common encoded variants for exfil detection.
|
||||||
|
|
||||||
|
The variant set is computed once per distinct secret and cached; callers
|
||||||
|
get a fresh list so they can't mutate the shared cached tuple."""
|
||||||
|
cached = _VARIANT_CACHE.get(secret)
|
||||||
|
if cached is None:
|
||||||
|
cached = _compute_encoded_variants(secret)
|
||||||
|
_VARIANT_CACHE[secret] = cached
|
||||||
|
return list(cached)
|
||||||
|
|
||||||
|
|
||||||
|
def _compute_encoded_variants(secret: str) -> tuple[str, ...]:
|
||||||
|
"""Derive the secret plus its encoded variants (uncached)."""
|
||||||
seen: set[str] = {secret}
|
seen: set[str] = {secret}
|
||||||
variants: list[str] = [secret]
|
variants: list[str] = [secret]
|
||||||
|
|
||||||
@@ -161,7 +183,7 @@ def _encoded_variants(secret: str) -> list[str]:
|
|||||||
# gzip + base64 (deterministic: mtime=0); recognisable by H4sI prefix
|
# gzip + base64 (deterministic: mtime=0); recognisable by H4sI prefix
|
||||||
_add(base64.b64encode(gzip.compress(secret_bytes, mtime=0)).decode("ascii"))
|
_add(base64.b64encode(gzip.compress(secret_bytes, mtime=0)).decode("ascii"))
|
||||||
|
|
||||||
return variants
|
return tuple(variants)
|
||||||
|
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
@@ -187,18 +209,24 @@ def _alnum_projection(text: str) -> str:
|
|||||||
|
|
||||||
|
|
||||||
def _find_partial_window(secret_alnum: str, text_alnum: str, min_len: int) -> int | None:
|
def _find_partial_window(secret_alnum: str, text_alnum: str, min_len: int) -> int | None:
|
||||||
"""Return the position in text_alnum where any min_len-char window of
|
"""Return the earliest position in text_alnum holding a min_len-char window
|
||||||
secret_alnum first appears, or None.
|
that also appears in secret_alnum, or None.
|
||||||
|
|
||||||
Slides a window of width min_len across secret_alnum and searches for
|
The secret's set of min_len-grams is small (bounded by the secret length),
|
||||||
each window in text_alnum. The first hit position is returned.
|
so building it once and sweeping the text a single time is O(len(text))
|
||||||
|
rather than the O(len(secret) * len(text)) of repeated substring searches —
|
||||||
|
which matters because this runs per provisioned secret on every request
|
||||||
|
body. Coverage is unchanged: a hit still means at least min_len consecutive
|
||||||
|
alphanumeric characters of the secret leaked into the text.
|
||||||
"""
|
"""
|
||||||
if len(secret_alnum) < min_len or len(text_alnum) < min_len:
|
if len(secret_alnum) < min_len or len(text_alnum) < min_len:
|
||||||
return None
|
return None
|
||||||
for i in range(len(secret_alnum) - min_len + 1):
|
secret_grams = {
|
||||||
window = secret_alnum[i:i + min_len]
|
secret_alnum[i:i + min_len]
|
||||||
pos = text_alnum.find(window)
|
for i in range(len(secret_alnum) - min_len + 1)
|
||||||
if pos >= 0:
|
}
|
||||||
|
for pos in range(len(text_alnum) - min_len + 1):
|
||||||
|
if text_alnum[pos:pos + min_len] in secret_grams:
|
||||||
return pos
|
return pos
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
|||||||
+41
-571
@@ -27,51 +27,36 @@ dataclass (`GitGatePlan`). The sidecar's start/stop lifecycle is
|
|||||||
backend-specific and lives on concrete subclasses (see
|
backend-specific and lives on concrete subclasses (see
|
||||||
`bot_bottle/backend/docker/git_gate.py`)."""
|
`bot_bottle/backend/docker/git_gate.py`)."""
|
||||||
|
|
||||||
|
|
||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
import dataclasses
|
import dataclasses
|
||||||
import os
|
|
||||||
import shlex
|
|
||||||
from abc import ABC
|
from abc import ABC
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
|
|
||||||
from .log import info
|
from .manifest import ManifestBottle
|
||||||
from .manifest import ManifestBottle, ManifestGitEntry
|
|
||||||
|
|
||||||
|
|
||||||
# Short network alias for git-gate inside the sidecar bundle. The
|
|
||||||
# agent's `.gitconfig` insteadOf rewrites resolve through this name.
|
|
||||||
GIT_GATE_HOSTNAME = "git-gate"
|
|
||||||
# Shared timeout (seconds) for all git-gate subprocess and CGI calls:
|
|
||||||
# git daemon (--timeout/--init-timeout), the access-hook subprocess in
|
|
||||||
# git_http_backend, and the git http-backend CGI subprocess.
|
|
||||||
GIT_GATE_TIMEOUT_SECS = 15
|
|
||||||
|
|
||||||
|
|
||||||
@dataclass(frozen=True)
|
|
||||||
class GitGateUpstream:
|
|
||||||
"""One bare repo on the gate. `name` drives the bare-repo path
|
|
||||||
(`/git/<name>.git`), the agent's URL after insteadOf rewrite
|
|
||||||
(`git://<gate>/<name>.git`), and the per-upstream credential
|
|
||||||
paths inside the gate (`/git-gate/creds/<name>-key` and
|
|
||||||
`/git-gate/creds/<name>-known_hosts`).
|
|
||||||
|
|
||||||
`identity_file` is the host-side absolute path the gate's start
|
|
||||||
step will docker-cp into the container. `known_host_key` is the
|
|
||||||
KnownHostKey string from the manifest; the gate's start step
|
|
||||||
materialises it into a known_hosts file if non-empty.
|
|
||||||
|
|
||||||
the gate credential paths inside the running sidecar."""
|
|
||||||
|
|
||||||
name: str
|
|
||||||
upstream_url: str
|
|
||||||
upstream_host: str
|
|
||||||
upstream_port: str
|
|
||||||
identity_file: str
|
|
||||||
known_host_key: str
|
|
||||||
known_hosts_file: Path = Path()
|
|
||||||
|
|
||||||
|
# Rendering and the deploy-key lifecycle live in sibling modules; the
|
||||||
|
# names are re-exported here (see __all__) so existing
|
||||||
|
# `from bot_bottle.git_gate import …` callers are unchanged.
|
||||||
|
from .git_gate_render import (
|
||||||
|
GIT_GATE_HOSTNAME,
|
||||||
|
GIT_GATE_TIMEOUT_SECS,
|
||||||
|
GitGateUpstream,
|
||||||
|
git_gate_known_hosts_line,
|
||||||
|
git_gate_render_access_hook,
|
||||||
|
git_gate_render_entrypoint,
|
||||||
|
git_gate_render_gitconfig,
|
||||||
|
git_gate_render_hook,
|
||||||
|
git_gate_upstreams_for_bottle,
|
||||||
|
_gitconfig_validate_value,
|
||||||
|
)
|
||||||
|
from .git_gate_provision import (
|
||||||
|
revoke_git_gate_provisioned_keys,
|
||||||
|
_provision_dynamic_key,
|
||||||
|
_resolve_identity_file,
|
||||||
|
)
|
||||||
|
|
||||||
@dataclass(frozen=True)
|
@dataclass(frozen=True)
|
||||||
class GitGatePlan:
|
class GitGatePlan:
|
||||||
@@ -96,540 +81,6 @@ class GitGatePlan:
|
|||||||
egress_network: str = ""
|
egress_network: str = ""
|
||||||
|
|
||||||
|
|
||||||
def git_gate_upstreams_for_bottle(bottle: ManifestBottle) -> tuple[GitGateUpstream, ...]:
|
|
||||||
"""Lift each `bottle.git` entry into a GitGateUpstream. Unique-Name
|
|
||||||
validation already ran in `manifest.ManifestBottle.from_dict`."""
|
|
||||||
return tuple(
|
|
||||||
GitGateUpstream(
|
|
||||||
name=e.Name,
|
|
||||||
upstream_url=e.Upstream,
|
|
||||||
upstream_host=e.UpstreamHost,
|
|
||||||
upstream_port=e.UpstreamPort,
|
|
||||||
identity_file=e.IdentityFile,
|
|
||||||
known_host_key=e.KnownHostKey,
|
|
||||||
)
|
|
||||||
for e in bottle.git
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def _gitconfig_validate_value(field: str, value: str) -> None:
|
|
||||||
"""Raise ValueError if value contains characters that break gitconfig line syntax."""
|
|
||||||
if "\n" in value or "\r" in value:
|
|
||||||
raise ValueError(
|
|
||||||
f"git-gate: {field} contains a newline, which would inject "
|
|
||||||
f"arbitrary gitconfig keys; rejecting manifest entry"
|
|
||||||
)
|
|
||||||
|
|
||||||
|
|
||||||
def git_gate_render_gitconfig(
|
|
||||||
entries: tuple[ManifestGitEntry, ...], gate_host: str, *, scheme: str = "git",
|
|
||||||
) -> str:
|
|
||||||
"""Render the agent's ~/.gitconfig content for git-gate
|
|
||||||
`insteadOf` rewrites. Pure host-side, no docker / smolvm;
|
|
||||||
exposed for tests + reuse across backends.
|
|
||||||
|
|
||||||
`gate_host` is the part of the URL between `<scheme>://` and the
|
|
||||||
repo path — backends differ here:
|
|
||||||
- docker: `git-gate` (the short network alias)
|
|
||||||
- smolmachines: `<bundle_ip>:<port>` (no DNS in the
|
|
||||||
TSI-allowlisted guest)
|
|
||||||
|
|
||||||
Empty `entries` returns an empty string so callers can no-op
|
|
||||||
cleanly without conditional formatting at the call site."""
|
|
||||||
if not entries:
|
|
||||||
return ""
|
|
||||||
out = [
|
|
||||||
"# bot-bottle git-gate (PRD 0008): every git operation against\n",
|
|
||||||
"# a declared upstream routes through the gate, which mirrors\n",
|
|
||||||
"# the upstream bidirectionally (gitleaks-scanned push;\n",
|
|
||||||
"# fetch-from-upstream-before-every-upload-pack via access-hook).\n",
|
|
||||||
]
|
|
||||||
for entry in entries:
|
|
||||||
_gitconfig_validate_value(f"repos[{entry.Name!r}].url", entry.Upstream)
|
|
||||||
out.append(f'[url "{scheme}://{gate_host}/{entry.Name}.git"]\n')
|
|
||||||
out.append(f"\tinsteadOf = {entry.Upstream}\n")
|
|
||||||
if entry.RemoteKey and entry.RemoteKey != entry.UpstreamHost:
|
|
||||||
port = (
|
|
||||||
f":{entry.UpstreamPort}"
|
|
||||||
if entry.UpstreamPort and entry.UpstreamPort != "22"
|
|
||||||
else ""
|
|
||||||
)
|
|
||||||
alias = (
|
|
||||||
f"ssh://{entry.UpstreamUser}@{entry.RemoteKey}{port}/"
|
|
||||||
f"{entry.UpstreamPath}"
|
|
||||||
)
|
|
||||||
_gitconfig_validate_value(f"repos[{entry.Name!r}].url (resolved alias)", alias)
|
|
||||||
out.append(f"\tinsteadOf = {alias}\n")
|
|
||||||
return "".join(out)
|
|
||||||
|
|
||||||
|
|
||||||
def git_gate_known_hosts_line(host: str, port: str, key: str) -> str:
|
|
||||||
"""Format `host[:port] key` for OpenSSH's known_hosts. Non-default
|
|
||||||
ports use the bracketed `[host]:port` form (the form OpenSSH writes
|
|
||||||
on disk for hosts reached via a non-22 port)."""
|
|
||||||
if port and port != "22":
|
|
||||||
target = f"[{host}]:{port}"
|
|
||||||
else:
|
|
||||||
target = host
|
|
||||||
return f"{target} {key}\n"
|
|
||||||
|
|
||||||
|
|
||||||
def git_gate_render_entrypoint(upstreams: tuple[GitGateUpstream, ...]) -> str:
|
|
||||||
"""Posix-sh entrypoint. One `init_repo` call per upstream, then
|
|
||||||
`exec git daemon`. The function reads
|
|
||||||
`/git-gate/creds/<name>-{key,known_hosts}` (bind-mounted into
|
|
||||||
the bundle by the renderer) and wires them into each bare repo's
|
|
||||||
config; the access-hook + pre-receive hook pick those paths up
|
|
||||||
at fetch / push time."""
|
|
||||||
lines = [
|
|
||||||
"#!/bin/sh",
|
|
||||||
"set -eu",
|
|
||||||
"",
|
|
||||||
"init_repo() {",
|
|
||||||
" name=$1",
|
|
||||||
" upstream_url=$2",
|
|
||||||
" keyfile=/git-gate/creds/${name}-key",
|
|
||||||
" hostsfile=/git-gate/creds/${name}-known_hosts",
|
|
||||||
"",
|
|
||||||
# `|| true`: PRD 0018 chunk 3+ bind-mounts these RO from the
|
|
||||||
# host, so chmod-syscalls fail with EROFS. The files already
|
|
||||||
# have the right perms on the host (SSH requires 0600 to load
|
|
||||||
# the key in the first place), so the chmod is best-effort
|
|
||||||
# cleanup for the legacy docker-cp path where the file
|
|
||||||
# landed at the host's umask perms.
|
|
||||||
" chmod 600 \"$keyfile\" 2>/dev/null || true",
|
|
||||||
" if [ -f \"$hostsfile\" ]; then",
|
|
||||||
" chmod 600 \"$hostsfile\" 2>/dev/null || true",
|
|
||||||
" fi",
|
|
||||||
"",
|
|
||||||
" repo=/git/${name}.git",
|
|
||||||
" if [ ! -d \"$repo\" ]; then",
|
|
||||||
" git init --bare \"$repo\" >/dev/null",
|
|
||||||
# --mirror=fetch sets remote.origin.fetch = +refs/*:refs/* so",
|
|
||||||
# a later `git fetch origin` mirrors the upstream's full ref",
|
|
||||||
# graph (heads, tags, notes) into the bare repo at canonical",
|
|
||||||
# paths. It does NOT set remote.origin.mirror=true, so an",
|
|
||||||
# explicit `git push origin <ref>:<ref>` still pushes one ref.",
|
|
||||||
" git -C \"$repo\" remote add --mirror=fetch origin \"$upstream_url\"",
|
|
||||||
" fi",
|
|
||||||
" git -C \"$repo\" config git-gate.identityFile \"$keyfile\"",
|
|
||||||
" git -C \"$repo\" config git-gate.knownHosts \"$hostsfile\"",
|
|
||||||
" git -C \"$repo\" config receive.denyCurrentBranch ignore",
|
|
||||||
" git -C \"$repo\" config receive.advertisePushOptions true",
|
|
||||||
" git -C \"$repo\" config http.receivepack true",
|
|
||||||
" install -m 755 /etc/git-gate/pre-receive \"$repo/hooks/pre-receive\"",
|
|
||||||
"}",
|
|
||||||
"",
|
|
||||||
"mkdir -p /git",
|
|
||||||
]
|
|
||||||
for u in upstreams:
|
|
||||||
lines.append(f"init_repo {shlex.quote(u.name)} {shlex.quote(u.upstream_url)}")
|
|
||||||
lines.extend([
|
|
||||||
"",
|
|
||||||
"exec git daemon \\",
|
|
||||||
" --reuseaddr \\",
|
|
||||||
f" --timeout={GIT_GATE_TIMEOUT_SECS} \\",
|
|
||||||
f" --init-timeout={GIT_GATE_TIMEOUT_SECS} \\",
|
|
||||||
" --base-path=/git \\",
|
|
||||||
" --export-all \\",
|
|
||||||
" --enable=receive-pack \\",
|
|
||||||
" --access-hook=/etc/git-gate/access-hook \\",
|
|
||||||
" --verbose",
|
|
||||||
])
|
|
||||||
return "\n".join(lines) + "\n"
|
|
||||||
|
|
||||||
|
|
||||||
def git_gate_render_hook() -> str:
|
|
||||||
"""The shared pre-receive hook: gitleaks-scan all incoming refs,
|
|
||||||
then forward each accepted ref to the real upstream (`origin`)
|
|
||||||
using the per-repo credential. Failure in either phase aborts
|
|
||||||
the push so the agent sees a real rejection. POSIX sh.
|
|
||||||
|
|
||||||
Two phases (scan all, then push all) keeps a hit on ref N from
|
|
||||||
half-pushing refs 1..N-1; both phases re-read stdin from a temp
|
|
||||||
file because pre-receive's stdin is a one-shot stream."""
|
|
||||||
return r"""#!/bin/sh
|
|
||||||
# git-gate pre-receive (PRD 0008). Stdin: <old> <new> <ref> per line.
|
|
||||||
set -u
|
|
||||||
|
|
||||||
refs_file=$(mktemp)
|
|
||||||
trap 'rm -f "$refs_file"' EXIT
|
|
||||||
cat > "$refs_file"
|
|
||||||
|
|
||||||
zero=0000000000000000000000000000000000000000
|
|
||||||
|
|
||||||
supervise_gitleaks_allow() {
|
|
||||||
log_opts=$1
|
|
||||||
ref=$2
|
|
||||||
report_file=$(mktemp)
|
|
||||||
if ! gitleaks git \
|
|
||||||
--log-opts="$log_opts" \
|
|
||||||
--no-banner \
|
|
||||||
--redact \
|
|
||||||
--ignore-gitleaks-allow \
|
|
||||||
--report-format=json \
|
|
||||||
--report-path="$report_file" \
|
|
||||||
--exit-code 0 \
|
|
||||||
1>&2; then
|
|
||||||
rm -f "$report_file"
|
|
||||||
echo "git-gate: gitleaks inline-suppression scan failed for $ref" >&2
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
proposal_id=$(
|
|
||||||
GITLEAKS_ALLOW_REF="$ref" python3 - "$report_file" <<'PY'
|
|
||||||
import datetime
|
|
||||||
import hashlib
|
|
||||||
import json
|
|
||||||
import os
|
|
||||||
import sys
|
|
||||||
import uuid
|
|
||||||
from pathlib import Path
|
|
||||||
|
|
||||||
report_path = Path(sys.argv[1])
|
|
||||||
queue_dir = os.environ.get("SUPERVISE_QUEUE_DIR", "")
|
|
||||||
slug = os.environ.get("SUPERVISE_BOTTLE_SLUG", "")
|
|
||||||
if not queue_dir or not slug:
|
|
||||||
sys.exit(2)
|
|
||||||
|
|
||||||
try:
|
|
||||||
raw = json.loads(report_path.read_text() or "[]")
|
|
||||||
except json.JSONDecodeError:
|
|
||||||
sys.exit(3)
|
|
||||||
if not isinstance(raw, list):
|
|
||||||
sys.exit(3)
|
|
||||||
if not raw:
|
|
||||||
sys.exit(0)
|
|
||||||
|
|
||||||
ref = os.environ.get("GITLEAKS_ALLOW_REF", "")
|
|
||||||
lines = [
|
|
||||||
"gitleaks inline suppression requires supervisor approval",
|
|
||||||
f"ref: {ref}",
|
|
||||||
"",
|
|
||||||
]
|
|
||||||
for i, finding in enumerate(raw, 1):
|
|
||||||
if not isinstance(finding, dict):
|
|
||||||
continue
|
|
||||||
file_path = finding.get("File", "")
|
|
||||||
line_no = finding.get("StartLine", finding.get("Line", ""))
|
|
||||||
rule_id = finding.get("RuleID", "")
|
|
||||||
commit = finding.get("Commit", "")
|
|
||||||
line = finding.get("Line", "")
|
|
||||||
lines.extend([
|
|
||||||
f"finding {i}:",
|
|
||||||
f" file: {file_path}",
|
|
||||||
f" line: {line_no}",
|
|
||||||
f" rule: {rule_id}",
|
|
||||||
f" commit: {commit}",
|
|
||||||
f" code: {line}",
|
|
||||||
"",
|
|
||||||
])
|
|
||||||
|
|
||||||
payload = "\n".join(lines).rstrip() + "\n"
|
|
||||||
proposal_id = str(uuid.uuid4())
|
|
||||||
proposal = {
|
|
||||||
"id": proposal_id,
|
|
||||||
"bottle_slug": slug,
|
|
||||||
"tool": "gitleaks-allow",
|
|
||||||
"proposed_file": payload,
|
|
||||||
"justification": (
|
|
||||||
"git-gate found gitleaks findings hidden by # gitleaks:allow; "
|
|
||||||
"approve only for dummy test fixtures or confirmed false positives"
|
|
||||||
),
|
|
||||||
"arrival_timestamp": datetime.datetime.now(
|
|
||||||
datetime.timezone.utc
|
|
||||||
).isoformat(),
|
|
||||||
"current_file_hash": hashlib.sha256(payload.encode("utf-8")).hexdigest(),
|
|
||||||
}
|
|
||||||
queue = Path(queue_dir)
|
|
||||||
queue.mkdir(parents=True, exist_ok=True)
|
|
||||||
path = queue / f"{proposal_id}.proposal.json"
|
|
||||||
tmp = path.with_suffix(path.suffix + ".tmp")
|
|
||||||
with tmp.open("w", encoding="utf-8") as f:
|
|
||||||
json.dump(proposal, f, indent=2)
|
|
||||||
f.write("\n")
|
|
||||||
os.chmod(tmp, 0o600)
|
|
||||||
os.replace(tmp, path)
|
|
||||||
print(proposal_id)
|
|
||||||
PY
|
|
||||||
)
|
|
||||||
rc=$?
|
|
||||||
rm -f "$report_file"
|
|
||||||
if [ "$rc" -eq 0 ] && [ -z "$proposal_id" ]; then
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
if [ "$rc" -ne 0 ]; then
|
|
||||||
echo "git-gate: cannot route # gitleaks:allow finding to supervisor; refusing push" >&2
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
queue_dir=${SUPERVISE_QUEUE_DIR:-}
|
|
||||||
response_file="$queue_dir/${proposal_id}.response.json"
|
|
||||||
timeout=${SUPERVISE_GITLEAKS_ALLOW_TIMEOUT_SECONDS:-300}
|
|
||||||
case "$timeout" in
|
|
||||||
''|*[!0-9]*)
|
|
||||||
echo "git-gate: invalid SUPERVISE_GITLEAKS_ALLOW_TIMEOUT_SECONDS=$timeout" >&2
|
|
||||||
return 1
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
echo "git-gate: queued # gitleaks:allow supervisor approval $proposal_id" >&2
|
|
||||||
echo "git-gate: approve with './cli.py supervise' to continue this push" >&2
|
|
||||||
waited=0
|
|
||||||
while [ "$waited" -lt "$timeout" ]; do
|
|
||||||
if [ -f "$response_file" ]; then
|
|
||||||
status=$(python3 - "$response_file" <<'PY'
|
|
||||||
import json
|
|
||||||
import sys
|
|
||||||
try:
|
|
||||||
with open(sys.argv[1], encoding="utf-8") as f:
|
|
||||||
raw = json.load(f)
|
|
||||||
except (OSError, json.JSONDecodeError):
|
|
||||||
sys.exit(1)
|
|
||||||
status = raw.get("status")
|
|
||||||
if not isinstance(status, str):
|
|
||||||
sys.exit(1)
|
|
||||||
print(status)
|
|
||||||
PY
|
|
||||||
) || status=""
|
|
||||||
case "$status" in
|
|
||||||
approved|modified)
|
|
||||||
mkdir -p "$queue_dir/processed"
|
|
||||||
mv -f "$queue_dir/${proposal_id}.proposal.json" "$queue_dir/processed/" 2>/dev/null || true
|
|
||||||
mv -f "$queue_dir/${proposal_id}.response.json" "$queue_dir/processed/" 2>/dev/null || true
|
|
||||||
echo "git-gate: supervisor approved # gitleaks:allow for $ref" >&2
|
|
||||||
return 0
|
|
||||||
;;
|
|
||||||
rejected)
|
|
||||||
echo "git-gate: supervisor rejected # gitleaks:allow for $ref" >&2
|
|
||||||
return 1
|
|
||||||
;;
|
|
||||||
*)
|
|
||||||
echo "git-gate: invalid supervisor response for # gitleaks:allow" >&2
|
|
||||||
return 1
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
fi
|
|
||||||
sleep 1
|
|
||||||
waited=$((waited + 1))
|
|
||||||
done
|
|
||||||
echo "git-gate: supervisor approval timed out for # gitleaks:allow; refusing push" >&2
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
|
|
||||||
# Phase 1: gitleaks scan each ref's incoming commits.
|
|
||||||
while IFS=' ' read -r old new ref; do
|
|
||||||
[ -z "$ref" ] && continue
|
|
||||||
[ "$new" = "$zero" ] && continue
|
|
||||||
if [ "$old" = "$zero" ]; then
|
|
||||||
# New ref: scan only the commits this push introduces — those
|
|
||||||
# reachable from $new but not from any ref the gate already has.
|
|
||||||
# Everything already on the gate arrived via upstream mirror-fetch
|
|
||||||
# or a previously gitleaks-scanned push, so it's already-upstream
|
|
||||||
# or already-scanned; re-scanning it (the old `$new` full-ancestry
|
|
||||||
# range) only resurfaces historical findings and blocks every new
|
|
||||||
# branch. See PRD 0028 / issue #106.
|
|
||||||
log_opts="$new --not --all"
|
|
||||||
else
|
|
||||||
log_opts="$old..$new"
|
|
||||||
fi
|
|
||||||
echo "git-gate: gitleaks scanning $ref ($log_opts)" >&2
|
|
||||||
if ! gitleaks git --log-opts="$log_opts" --no-banner --redact 1>&2; then
|
|
||||||
echo "git-gate: gitleaks rejected push to $ref" >&2
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
if ! supervise_gitleaks_allow "$log_opts" "$ref"; then
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
done < "$refs_file"
|
|
||||||
|
|
||||||
# Phase 2: forward each ref to the upstream (`origin`, configured
|
|
||||||
# in the entrypoint via `git remote add --mirror=fetch`).
|
|
||||||
keyfile=$(git config --get git-gate.identityFile)
|
|
||||||
hostsfile=$(git config --get git-gate.knownHosts)
|
|
||||||
if [ ! -f "$hostsfile" ]; then
|
|
||||||
echo "git-gate: no KnownHostKey configured for this upstream; refusing to push" >&2
|
|
||||||
echo "git-gate: add KnownHostKey to the bottle.git entry and restart the bottle" >&2
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
ssh_cmd="ssh -i $keyfile -o UserKnownHostsFile=$hostsfile -o StrictHostKeyChecking=yes -o IdentitiesOnly=yes -o BatchMode=yes -o ConnectTimeout=10"
|
|
||||||
|
|
||||||
push_option_count=${GIT_PUSH_OPTION_COUNT:-0}
|
|
||||||
case "$push_option_count" in
|
|
||||||
''|*[!0-9]*)
|
|
||||||
echo "git-gate: invalid GIT_PUSH_OPTION_COUNT=$push_option_count" >&2
|
|
||||||
exit 1
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
set --
|
|
||||||
i=0
|
|
||||||
while [ "$i" -lt "$push_option_count" ]; do
|
|
||||||
opt=$(printenv "GIT_PUSH_OPTION_$i" || :)
|
|
||||||
set -- "$@" --push-option="$opt"
|
|
||||||
i=$((i + 1))
|
|
||||||
done
|
|
||||||
|
|
||||||
while IFS=' ' read -r old new ref; do
|
|
||||||
[ -z "$ref" ] && continue
|
|
||||||
if [ "$new" = "$zero" ]; then
|
|
||||||
refspec=":$ref"
|
|
||||||
elif [ "$old" != "$zero" ] && ! git merge-base --is-ancestor "$old" "$new" 2>/dev/null; then
|
|
||||||
refspec="+$new:$ref"
|
|
||||||
else
|
|
||||||
refspec="$new:$ref"
|
|
||||||
fi
|
|
||||||
echo "git-gate: forwarding $ref to origin" >&2
|
|
||||||
if ! GIT_SSH_COMMAND="$ssh_cmd" git push "$@" origin "$refspec" 1>&2; then
|
|
||||||
echo "git-gate: upstream push failed for $ref" >&2
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
done < "$refs_file"
|
|
||||||
|
|
||||||
exit 0
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
def git_gate_render_access_hook() -> str:
|
|
||||||
"""`git daemon --access-hook` script. Runs before each protocol
|
|
||||||
service; for `upload-pack` (fetch / clone / ls-remote / pull) it
|
|
||||||
refreshes the bare repo from upstream first, so the response
|
|
||||||
reflects upstream's current state. For other services (notably
|
|
||||||
`receive-pack`) it returns 0 immediately and lets the existing
|
|
||||||
pre-receive hook gate the operation. POSIX sh.
|
|
||||||
|
|
||||||
The hook receives:
|
|
||||||
$1 service name (`upload-pack`, `receive-pack`, ...)
|
|
||||||
$2 absolute path to the resolved repo
|
|
||||||
$3 client hostname (unused)
|
|
||||||
$4 client tcp address (unused)
|
|
||||||
|
|
||||||
Fail-closed on upstream errors: the agent's fetch fails too,
|
|
||||||
so it never silently sees stale data — matches the PRD's
|
|
||||||
'equivalent to operations against the upstream' contract."""
|
|
||||||
return r"""#!/bin/sh
|
|
||||||
# git-gate access-hook (PRD 0008). $1=service $2=repo $3=host $4=peer
|
|
||||||
set -u
|
|
||||||
service=$1
|
|
||||||
repo_dir=$2
|
|
||||||
|
|
||||||
# Push path keeps its own gating in pre-receive (gitleaks +
|
|
||||||
# forward). Only refresh-from-upstream on fetch operations.
|
|
||||||
if [ "$service" != "upload-pack" ]; then
|
|
||||||
exit 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
keyfile=$(git -C "$repo_dir" config --get git-gate.identityFile 2>/dev/null || true)
|
|
||||||
hostsfile=$(git -C "$repo_dir" config --get git-gate.knownHosts 2>/dev/null || true)
|
|
||||||
if [ -z "$keyfile" ] || [ ! -f "$hostsfile" ]; then
|
|
||||||
echo "git-gate: missing credentials for $repo_dir; refusing fetch" >&2
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
ssh_cmd="ssh -i $keyfile -o UserKnownHostsFile=$hostsfile -o StrictHostKeyChecking=yes -o IdentitiesOnly=yes -o BatchMode=yes -o ConnectTimeout=10"
|
|
||||||
|
|
||||||
echo "git-gate: refreshing $repo_dir from upstream" >&2
|
|
||||||
if ! GIT_SSH_COMMAND="$ssh_cmd" git -C "$repo_dir" fetch origin --prune >&2; then
|
|
||||||
echo "git-gate: upstream fetch failed for $repo_dir; refusing to serve stale data" >&2
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Sync the bare repo's HEAD to upstream's HEAD on the first fetch
|
|
||||||
# (when it still points at the `git init --bare` default of
|
|
||||||
# refs/heads/master and upstream uses something else, the cloned
|
|
||||||
# checkout would fail with "remote HEAD refers to nonexistent ref").
|
|
||||||
# Costs one extra ls-remote on first fetch only; subsequent fetches
|
|
||||||
# skip the branch. If upstream's default branch changes after the
|
|
||||||
# gate has cached it, restart the bottle to resync.
|
|
||||||
if ! git -C "$repo_dir" rev-parse --verify HEAD >/dev/null 2>&1; then
|
|
||||||
upstream_head=$(GIT_SSH_COMMAND="$ssh_cmd" git -C "$repo_dir" \
|
|
||||||
ls-remote --symref origin HEAD 2>/dev/null \
|
|
||||||
| awk '/^ref:/ {print $2; exit}')
|
|
||||||
if [ -n "$upstream_head" ]; then
|
|
||||||
git -C "$repo_dir" symbolic-ref HEAD "$upstream_head" || true
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
exit 0
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
def _provision_dynamic_key(
|
|
||||||
entry: ManifestGitEntry,
|
|
||||||
slug: str,
|
|
||||||
stage_dir: Path,
|
|
||||||
) -> str:
|
|
||||||
"""Generate a fresh ed25519 keypair, register the public half with
|
|
||||||
the forge, and persist the private key + key ID under `stage_dir`.
|
|
||||||
|
|
||||||
Returns the host-side path to the private key file so the caller
|
|
||||||
can inject it into the GitGateUpstream as `identity_file`."""
|
|
||||||
from .deploy_key_provisioner import get_provisioner
|
|
||||||
pk = entry.Key
|
|
||||||
token = os.environ.get(pk.forge_token_env)
|
|
||||||
if token is None:
|
|
||||||
raise RuntimeError(
|
|
||||||
f"git-gate.repos[{entry.Name!r}] key.forge_token_env"
|
|
||||||
f" = {pk.forge_token_env!r}: env var is not set"
|
|
||||||
)
|
|
||||||
api_url = pk.api_url or f"https://{entry.UpstreamHost}"
|
|
||||||
provisioner = get_provisioner(pk.provider, token, api_url)
|
|
||||||
|
|
||||||
owner_repo = entry.UpstreamPath
|
|
||||||
if owner_repo.endswith(".git"):
|
|
||||||
owner_repo = owner_repo[:-4]
|
|
||||||
title = f"bot-bottle:{slug}:{entry.Name}"
|
|
||||||
|
|
||||||
info(f"provisioning deploy key for git-gate.repos[{entry.Name!r}]")
|
|
||||||
key_id, private_key_bytes = provisioner.create(owner_repo, title)
|
|
||||||
|
|
||||||
key_file = stage_dir / f"{entry.Name}-key"
|
|
||||||
key_file.write_bytes(private_key_bytes)
|
|
||||||
key_file.chmod(0o600)
|
|
||||||
|
|
||||||
id_file = stage_dir / f"{entry.Name}-deploy-key-id"
|
|
||||||
id_file.write_text(key_id)
|
|
||||||
id_file.chmod(0o600)
|
|
||||||
|
|
||||||
info(f"provisioned deploy key {key_id} for git-gate.repos[{entry.Name!r}]")
|
|
||||||
return str(key_file)
|
|
||||||
|
|
||||||
|
|
||||||
def revoke_git_gate_provisioned_keys(bottle: ManifestBottle, stage_dir: Path) -> None:
|
|
||||||
"""Revoke all deploy keys provisioned for `bottle` during prepare.
|
|
||||||
|
|
||||||
Called at teardown after containers stop. Raises if any revocation
|
|
||||||
fails — a stranded key is a security concern that the operator must
|
|
||||||
address manually."""
|
|
||||||
from .deploy_key_provisioner import get_provisioner
|
|
||||||
for entry in bottle.git:
|
|
||||||
if entry.Key.provider != "gitea":
|
|
||||||
continue
|
|
||||||
pk = entry.Key
|
|
||||||
id_file = stage_dir / f"{entry.Name}-deploy-key-id"
|
|
||||||
if not id_file.exists():
|
|
||||||
continue
|
|
||||||
key_id = id_file.read_text().strip()
|
|
||||||
token = os.environ.get(pk.forge_token_env)
|
|
||||||
if token is None:
|
|
||||||
raise RuntimeError(
|
|
||||||
f"git-gate.repos[{entry.Name!r}] key.forge_token_env"
|
|
||||||
f" = {pk.forge_token_env!r}: env var is not set;"
|
|
||||||
f" cannot revoke deploy key {key_id}"
|
|
||||||
)
|
|
||||||
api_url = pk.api_url or f"https://{entry.UpstreamHost}"
|
|
||||||
provisioner = get_provisioner(pk.provider, token, api_url)
|
|
||||||
owner_repo = entry.UpstreamPath
|
|
||||||
if owner_repo.endswith(".git"):
|
|
||||||
owner_repo = owner_repo[:-4]
|
|
||||||
info(f"revoking deploy key {key_id} for git-gate.repos[{entry.Name!r}]")
|
|
||||||
provisioner.delete(owner_repo, key_id)
|
|
||||||
info(f"revoked deploy key {key_id} for git-gate.repos[{entry.Name!r}]")
|
|
||||||
|
|
||||||
|
|
||||||
def _resolve_identity_file(entry: ManifestGitEntry, slug: str, stage_dir: Path) -> str:
|
|
||||||
"""Return the host-side SSH identity file path for this entry.
|
|
||||||
For gitea entries, provisions a fresh deploy key first."""
|
|
||||||
if entry.Key.provider == "gitea":
|
|
||||||
return _provision_dynamic_key(entry, slug, stage_dir)
|
|
||||||
return entry.IdentityFile
|
|
||||||
|
|
||||||
|
|
||||||
class GitGate(ABC):
|
class GitGate(ABC):
|
||||||
"""The per-agent git-gate. Encapsulates the host-side prepare
|
"""The per-agent git-gate. Encapsulates the host-side prepare
|
||||||
@@ -697,3 +148,22 @@ class GitGate(ABC):
|
|||||||
access_hook_script=access_hook,
|
access_hook_script=access_hook,
|
||||||
upstreams=tuple(upstreams_with_files),
|
upstreams=tuple(upstreams_with_files),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
__all__ = [
|
||||||
|
"GIT_GATE_HOSTNAME",
|
||||||
|
"GIT_GATE_TIMEOUT_SECS",
|
||||||
|
"GitGateUpstream",
|
||||||
|
"GitGatePlan",
|
||||||
|
"GitGate",
|
||||||
|
"git_gate_upstreams_for_bottle",
|
||||||
|
"git_gate_render_gitconfig",
|
||||||
|
"git_gate_known_hosts_line",
|
||||||
|
"git_gate_render_entrypoint",
|
||||||
|
"git_gate_render_hook",
|
||||||
|
"git_gate_render_access_hook",
|
||||||
|
"revoke_git_gate_provisioned_keys",
|
||||||
|
"_gitconfig_validate_value",
|
||||||
|
"_provision_dynamic_key",
|
||||||
|
"_resolve_identity_file",
|
||||||
|
]
|
||||||
|
|||||||
@@ -0,0 +1,102 @@
|
|||||||
|
"""git-gate deploy-key lifecycle for `gitea` upstreams (PRD 0047/0048).
|
||||||
|
|
||||||
|
Provisions a fresh ed25519 deploy key via the forge API at prepare time
|
||||||
|
and revokes it at teardown, so the agent never holds an upstream
|
||||||
|
credential. Split out of `git_gate.py`; the forge HTTP client is lazily
|
||||||
|
imported (`deploy_key_provisioner`) to keep its cost off the host path.
|
||||||
|
`git_gate` re-exports these names for API stability."""
|
||||||
|
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import os
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
from .log import info
|
||||||
|
from .manifest import ManifestBottle, ManifestGitEntry
|
||||||
|
|
||||||
|
def _provision_dynamic_key(
|
||||||
|
entry: ManifestGitEntry,
|
||||||
|
slug: str,
|
||||||
|
stage_dir: Path,
|
||||||
|
) -> str:
|
||||||
|
"""Generate a fresh ed25519 keypair, register the public half with
|
||||||
|
the forge, and persist the private key + key ID under `stage_dir`.
|
||||||
|
|
||||||
|
Returns the host-side path to the private key file so the caller
|
||||||
|
can inject it into the GitGateUpstream as `identity_file`."""
|
||||||
|
from .deploy_key_provisioner import get_provisioner
|
||||||
|
pk = entry.Key
|
||||||
|
token = os.environ.get(pk.forge_token_env)
|
||||||
|
if token is None:
|
||||||
|
raise RuntimeError(
|
||||||
|
f"git-gate.repos[{entry.Name!r}] key.forge_token_env"
|
||||||
|
f" = {pk.forge_token_env!r}: env var is not set"
|
||||||
|
)
|
||||||
|
api_url = pk.api_url or f"https://{entry.UpstreamHost}"
|
||||||
|
provisioner = get_provisioner(pk.provider, token, api_url)
|
||||||
|
|
||||||
|
owner_repo = entry.UpstreamPath
|
||||||
|
if owner_repo.endswith(".git"):
|
||||||
|
owner_repo = owner_repo[:-4]
|
||||||
|
title = f"bot-bottle:{slug}:{entry.Name}"
|
||||||
|
|
||||||
|
info(f"provisioning deploy key for git-gate.repos[{entry.Name!r}]")
|
||||||
|
key_id, private_key_bytes = provisioner.create(owner_repo, title)
|
||||||
|
|
||||||
|
key_file = stage_dir / f"{entry.Name}-key"
|
||||||
|
key_file.write_bytes(private_key_bytes)
|
||||||
|
key_file.chmod(0o600)
|
||||||
|
|
||||||
|
id_file = stage_dir / f"{entry.Name}-deploy-key-id"
|
||||||
|
id_file.write_text(key_id)
|
||||||
|
id_file.chmod(0o600)
|
||||||
|
|
||||||
|
info(f"provisioned deploy key {key_id} for git-gate.repos[{entry.Name!r}]")
|
||||||
|
return str(key_file)
|
||||||
|
|
||||||
|
|
||||||
|
def revoke_git_gate_provisioned_keys(bottle: ManifestBottle, stage_dir: Path) -> None:
|
||||||
|
"""Revoke all deploy keys provisioned for `bottle` during prepare.
|
||||||
|
|
||||||
|
Called at teardown after containers stop. Raises if any revocation
|
||||||
|
fails — a stranded key is a security concern that the operator must
|
||||||
|
address manually."""
|
||||||
|
from .deploy_key_provisioner import get_provisioner
|
||||||
|
for entry in bottle.git:
|
||||||
|
if entry.Key.provider != "gitea":
|
||||||
|
continue
|
||||||
|
pk = entry.Key
|
||||||
|
id_file = stage_dir / f"{entry.Name}-deploy-key-id"
|
||||||
|
if not id_file.exists():
|
||||||
|
continue
|
||||||
|
key_id = id_file.read_text().strip()
|
||||||
|
token = os.environ.get(pk.forge_token_env)
|
||||||
|
if token is None:
|
||||||
|
raise RuntimeError(
|
||||||
|
f"git-gate.repos[{entry.Name!r}] key.forge_token_env"
|
||||||
|
f" = {pk.forge_token_env!r}: env var is not set;"
|
||||||
|
f" cannot revoke deploy key {key_id}"
|
||||||
|
)
|
||||||
|
api_url = pk.api_url or f"https://{entry.UpstreamHost}"
|
||||||
|
provisioner = get_provisioner(pk.provider, token, api_url)
|
||||||
|
owner_repo = entry.UpstreamPath
|
||||||
|
if owner_repo.endswith(".git"):
|
||||||
|
owner_repo = owner_repo[:-4]
|
||||||
|
info(f"revoking deploy key {key_id} for git-gate.repos[{entry.Name!r}]")
|
||||||
|
provisioner.delete(owner_repo, key_id)
|
||||||
|
info(f"revoked deploy key {key_id} for git-gate.repos[{entry.Name!r}]")
|
||||||
|
|
||||||
|
|
||||||
|
def _resolve_identity_file(entry: ManifestGitEntry, slug: str, stage_dir: Path) -> str:
|
||||||
|
"""Return the host-side SSH identity file path for this entry.
|
||||||
|
For gitea entries, provisions a fresh deploy key first."""
|
||||||
|
if entry.Key.provider == "gitea":
|
||||||
|
return _provision_dynamic_key(entry, slug, stage_dir)
|
||||||
|
return entry.IdentityFile
|
||||||
|
|
||||||
|
|
||||||
|
__all__ = [
|
||||||
|
"revoke_git_gate_provisioned_keys",
|
||||||
|
"_provision_dynamic_key",
|
||||||
|
"_resolve_identity_file",
|
||||||
|
]
|
||||||
@@ -0,0 +1,502 @@
|
|||||||
|
"""Pure host-side rendering for the per-agent git-gate (PRD 0008).
|
||||||
|
|
||||||
|
Builds the agent's `.gitconfig` insteadOf rewrites, the known_hosts
|
||||||
|
line, and the entrypoint / pre-receive / access-hook scripts the sidecar
|
||||||
|
runs. No docker or forge calls — exposed for tests and reuse across
|
||||||
|
backends. Split out of `git_gate.py` so the control surface (`GitGate`)
|
||||||
|
and the deploy-key lifecycle (`git_gate_provision`) each read on their
|
||||||
|
own; `git_gate` re-exports these names for API stability."""
|
||||||
|
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import shlex
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
from .manifest import ManifestBottle, ManifestGitEntry
|
||||||
|
|
||||||
|
# Short network alias for git-gate inside the sidecar bundle. The
|
||||||
|
# agent's `.gitconfig` insteadOf rewrites resolve through this name.
|
||||||
|
GIT_GATE_HOSTNAME = "git-gate"
|
||||||
|
# Shared timeout (seconds) for all git-gate subprocess and CGI calls:
|
||||||
|
# git daemon (--timeout/--init-timeout), the access-hook subprocess in
|
||||||
|
# git_http_backend, and the git http-backend CGI subprocess.
|
||||||
|
GIT_GATE_TIMEOUT_SECS = 15
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass(frozen=True)
|
||||||
|
class GitGateUpstream:
|
||||||
|
"""One bare repo on the gate. `name` drives the bare-repo path
|
||||||
|
(`/git/<name>.git`), the agent's URL after insteadOf rewrite
|
||||||
|
(`git://<gate>/<name>.git`), and the per-upstream credential
|
||||||
|
paths inside the gate (`/git-gate/creds/<name>-key` and
|
||||||
|
`/git-gate/creds/<name>-known_hosts`).
|
||||||
|
|
||||||
|
`identity_file` is the host-side absolute path the gate's start
|
||||||
|
step will docker-cp into the container. `known_host_key` is the
|
||||||
|
KnownHostKey string from the manifest; the gate's start step
|
||||||
|
materialises it into a known_hosts file if non-empty.
|
||||||
|
|
||||||
|
the gate credential paths inside the running sidecar."""
|
||||||
|
|
||||||
|
name: str
|
||||||
|
upstream_url: str
|
||||||
|
upstream_host: str
|
||||||
|
upstream_port: str
|
||||||
|
identity_file: str
|
||||||
|
known_host_key: str
|
||||||
|
known_hosts_file: Path = Path()
|
||||||
|
|
||||||
|
def git_gate_upstreams_for_bottle(bottle: ManifestBottle) -> tuple[GitGateUpstream, ...]:
|
||||||
|
"""Lift each `bottle.git` entry into a GitGateUpstream. Unique-Name
|
||||||
|
validation already ran in `manifest.ManifestBottle.from_dict`."""
|
||||||
|
return tuple(
|
||||||
|
GitGateUpstream(
|
||||||
|
name=e.Name,
|
||||||
|
upstream_url=e.Upstream,
|
||||||
|
upstream_host=e.UpstreamHost,
|
||||||
|
upstream_port=e.UpstreamPort,
|
||||||
|
identity_file=e.IdentityFile,
|
||||||
|
known_host_key=e.KnownHostKey,
|
||||||
|
)
|
||||||
|
for e in bottle.git
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def _gitconfig_validate_value(field: str, value: str) -> None:
|
||||||
|
"""Raise ValueError if value contains characters that break gitconfig line syntax."""
|
||||||
|
if "\n" in value or "\r" in value:
|
||||||
|
raise ValueError(
|
||||||
|
f"git-gate: {field} contains a newline, which would inject "
|
||||||
|
f"arbitrary gitconfig keys; rejecting manifest entry"
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def git_gate_render_gitconfig(
|
||||||
|
entries: tuple[ManifestGitEntry, ...], gate_host: str, *, scheme: str = "git",
|
||||||
|
) -> str:
|
||||||
|
"""Render the agent's ~/.gitconfig content for git-gate
|
||||||
|
`insteadOf` rewrites. Pure host-side, no docker / smolvm;
|
||||||
|
exposed for tests + reuse across backends.
|
||||||
|
|
||||||
|
`gate_host` is the part of the URL between `<scheme>://` and the
|
||||||
|
repo path — backends differ here:
|
||||||
|
- docker: `git-gate` (the short network alias)
|
||||||
|
- smolmachines: `<bundle_ip>:<port>` (no DNS in the
|
||||||
|
TSI-allowlisted guest)
|
||||||
|
|
||||||
|
Empty `entries` returns an empty string so callers can no-op
|
||||||
|
cleanly without conditional formatting at the call site."""
|
||||||
|
if not entries:
|
||||||
|
return ""
|
||||||
|
out = [
|
||||||
|
"# bot-bottle git-gate (PRD 0008): every git operation against\n",
|
||||||
|
"# a declared upstream routes through the gate, which mirrors\n",
|
||||||
|
"# the upstream bidirectionally (gitleaks-scanned push;\n",
|
||||||
|
"# fetch-from-upstream-before-every-upload-pack via access-hook).\n",
|
||||||
|
]
|
||||||
|
for entry in entries:
|
||||||
|
_gitconfig_validate_value(f"repos[{entry.Name!r}].url", entry.Upstream)
|
||||||
|
out.append(f'[url "{scheme}://{gate_host}/{entry.Name}.git"]\n')
|
||||||
|
out.append(f"\tinsteadOf = {entry.Upstream}\n")
|
||||||
|
if entry.RemoteKey and entry.RemoteKey != entry.UpstreamHost:
|
||||||
|
port = (
|
||||||
|
f":{entry.UpstreamPort}"
|
||||||
|
if entry.UpstreamPort and entry.UpstreamPort != "22"
|
||||||
|
else ""
|
||||||
|
)
|
||||||
|
alias = (
|
||||||
|
f"ssh://{entry.UpstreamUser}@{entry.RemoteKey}{port}/"
|
||||||
|
f"{entry.UpstreamPath}"
|
||||||
|
)
|
||||||
|
_gitconfig_validate_value(f"repos[{entry.Name!r}].url (resolved alias)", alias)
|
||||||
|
out.append(f"\tinsteadOf = {alias}\n")
|
||||||
|
return "".join(out)
|
||||||
|
|
||||||
|
|
||||||
|
def git_gate_known_hosts_line(host: str, port: str, key: str) -> str:
|
||||||
|
"""Format `host[:port] key` for OpenSSH's known_hosts. Non-default
|
||||||
|
ports use the bracketed `[host]:port` form (the form OpenSSH writes
|
||||||
|
on disk for hosts reached via a non-22 port)."""
|
||||||
|
if port and port != "22":
|
||||||
|
target = f"[{host}]:{port}"
|
||||||
|
else:
|
||||||
|
target = host
|
||||||
|
return f"{target} {key}\n"
|
||||||
|
|
||||||
|
|
||||||
|
def git_gate_render_entrypoint(upstreams: tuple[GitGateUpstream, ...]) -> str:
|
||||||
|
"""Posix-sh entrypoint. One `init_repo` call per upstream, then
|
||||||
|
`exec git daemon`. The function reads
|
||||||
|
`/git-gate/creds/<name>-{key,known_hosts}` (bind-mounted into
|
||||||
|
the bundle by the renderer) and wires them into each bare repo's
|
||||||
|
config; the access-hook + pre-receive hook pick those paths up
|
||||||
|
at fetch / push time."""
|
||||||
|
lines = [
|
||||||
|
"#!/bin/sh",
|
||||||
|
"set -eu",
|
||||||
|
"",
|
||||||
|
"init_repo() {",
|
||||||
|
" name=$1",
|
||||||
|
" upstream_url=$2",
|
||||||
|
" keyfile=/git-gate/creds/${name}-key",
|
||||||
|
" hostsfile=/git-gate/creds/${name}-known_hosts",
|
||||||
|
"",
|
||||||
|
# `|| true`: PRD 0018 chunk 3+ bind-mounts these RO from the
|
||||||
|
# host, so chmod-syscalls fail with EROFS. The files already
|
||||||
|
# have the right perms on the host (SSH requires 0600 to load
|
||||||
|
# the key in the first place), so the chmod is best-effort
|
||||||
|
# cleanup for the legacy docker-cp path where the file
|
||||||
|
# landed at the host's umask perms.
|
||||||
|
" chmod 600 \"$keyfile\" 2>/dev/null || true",
|
||||||
|
" if [ -f \"$hostsfile\" ]; then",
|
||||||
|
" chmod 600 \"$hostsfile\" 2>/dev/null || true",
|
||||||
|
" fi",
|
||||||
|
"",
|
||||||
|
" repo=/git/${name}.git",
|
||||||
|
" if [ ! -d \"$repo\" ]; then",
|
||||||
|
" git init --bare \"$repo\" >/dev/null",
|
||||||
|
# --mirror=fetch sets remote.origin.fetch = +refs/*:refs/* so",
|
||||||
|
# a later `git fetch origin` mirrors the upstream's full ref",
|
||||||
|
# graph (heads, tags, notes) into the bare repo at canonical",
|
||||||
|
# paths. It does NOT set remote.origin.mirror=true, so an",
|
||||||
|
# explicit `git push origin <ref>:<ref>` still pushes one ref.",
|
||||||
|
" git -C \"$repo\" remote add --mirror=fetch origin \"$upstream_url\"",
|
||||||
|
" fi",
|
||||||
|
" git -C \"$repo\" config git-gate.identityFile \"$keyfile\"",
|
||||||
|
" git -C \"$repo\" config git-gate.knownHosts \"$hostsfile\"",
|
||||||
|
" git -C \"$repo\" config receive.denyCurrentBranch ignore",
|
||||||
|
" git -C \"$repo\" config receive.advertisePushOptions true",
|
||||||
|
" git -C \"$repo\" config http.receivepack true",
|
||||||
|
" install -m 755 /etc/git-gate/pre-receive \"$repo/hooks/pre-receive\"",
|
||||||
|
"}",
|
||||||
|
"",
|
||||||
|
"mkdir -p /git",
|
||||||
|
]
|
||||||
|
for u in upstreams:
|
||||||
|
lines.append(f"init_repo {shlex.quote(u.name)} {shlex.quote(u.upstream_url)}")
|
||||||
|
lines.extend([
|
||||||
|
"",
|
||||||
|
"exec git daemon \\",
|
||||||
|
" --reuseaddr \\",
|
||||||
|
f" --timeout={GIT_GATE_TIMEOUT_SECS} \\",
|
||||||
|
f" --init-timeout={GIT_GATE_TIMEOUT_SECS} \\",
|
||||||
|
" --base-path=/git \\",
|
||||||
|
" --export-all \\",
|
||||||
|
" --enable=receive-pack \\",
|
||||||
|
" --access-hook=/etc/git-gate/access-hook \\",
|
||||||
|
" --verbose",
|
||||||
|
])
|
||||||
|
return "\n".join(lines) + "\n"
|
||||||
|
|
||||||
|
|
||||||
|
def git_gate_render_hook() -> str:
|
||||||
|
"""The shared pre-receive hook: gitleaks-scan all incoming refs,
|
||||||
|
then forward each accepted ref to the real upstream (`origin`)
|
||||||
|
using the per-repo credential. Failure in either phase aborts
|
||||||
|
the push so the agent sees a real rejection. POSIX sh.
|
||||||
|
|
||||||
|
Two phases (scan all, then push all) keeps a hit on ref N from
|
||||||
|
half-pushing refs 1..N-1; both phases re-read stdin from a temp
|
||||||
|
file because pre-receive's stdin is a one-shot stream."""
|
||||||
|
return r"""#!/bin/sh
|
||||||
|
# git-gate pre-receive (PRD 0008). Stdin: <old> <new> <ref> per line.
|
||||||
|
set -u
|
||||||
|
|
||||||
|
refs_file=$(mktemp)
|
||||||
|
trap 'rm -f "$refs_file"' EXIT
|
||||||
|
cat > "$refs_file"
|
||||||
|
|
||||||
|
zero=0000000000000000000000000000000000000000
|
||||||
|
|
||||||
|
supervise_gitleaks_allow() {
|
||||||
|
log_opts=$1
|
||||||
|
ref=$2
|
||||||
|
report_file=$(mktemp)
|
||||||
|
if ! gitleaks git \
|
||||||
|
--log-opts="$log_opts" \
|
||||||
|
--no-banner \
|
||||||
|
--redact \
|
||||||
|
--ignore-gitleaks-allow \
|
||||||
|
--report-format=json \
|
||||||
|
--report-path="$report_file" \
|
||||||
|
--exit-code 0 \
|
||||||
|
1>&2; then
|
||||||
|
rm -f "$report_file"
|
||||||
|
echo "git-gate: gitleaks inline-suppression scan failed for $ref" >&2
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
proposal_id=$(
|
||||||
|
GITLEAKS_ALLOW_REF="$ref" python3 - "$report_file" <<'PY'
|
||||||
|
import datetime
|
||||||
|
import hashlib
|
||||||
|
import json
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
import uuid
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
report_path = Path(sys.argv[1])
|
||||||
|
queue_dir = os.environ.get("SUPERVISE_QUEUE_DIR", "")
|
||||||
|
slug = os.environ.get("SUPERVISE_BOTTLE_SLUG", "")
|
||||||
|
if not queue_dir or not slug:
|
||||||
|
sys.exit(2)
|
||||||
|
|
||||||
|
try:
|
||||||
|
raw = json.loads(report_path.read_text() or "[]")
|
||||||
|
except json.JSONDecodeError:
|
||||||
|
sys.exit(3)
|
||||||
|
if not isinstance(raw, list):
|
||||||
|
sys.exit(3)
|
||||||
|
if not raw:
|
||||||
|
sys.exit(0)
|
||||||
|
|
||||||
|
ref = os.environ.get("GITLEAKS_ALLOW_REF", "")
|
||||||
|
lines = [
|
||||||
|
"gitleaks inline suppression requires supervisor approval",
|
||||||
|
f"ref: {ref}",
|
||||||
|
"",
|
||||||
|
]
|
||||||
|
for i, finding in enumerate(raw, 1):
|
||||||
|
if not isinstance(finding, dict):
|
||||||
|
continue
|
||||||
|
file_path = finding.get("File", "")
|
||||||
|
line_no = finding.get("StartLine", finding.get("Line", ""))
|
||||||
|
rule_id = finding.get("RuleID", "")
|
||||||
|
commit = finding.get("Commit", "")
|
||||||
|
line = finding.get("Line", "")
|
||||||
|
lines.extend([
|
||||||
|
f"finding {i}:",
|
||||||
|
f" file: {file_path}",
|
||||||
|
f" line: {line_no}",
|
||||||
|
f" rule: {rule_id}",
|
||||||
|
f" commit: {commit}",
|
||||||
|
f" code: {line}",
|
||||||
|
"",
|
||||||
|
])
|
||||||
|
|
||||||
|
payload = "\n".join(lines).rstrip() + "\n"
|
||||||
|
proposal_id = str(uuid.uuid4())
|
||||||
|
proposal = {
|
||||||
|
"id": proposal_id,
|
||||||
|
"bottle_slug": slug,
|
||||||
|
"tool": "gitleaks-allow",
|
||||||
|
"proposed_file": payload,
|
||||||
|
"justification": (
|
||||||
|
"git-gate found gitleaks findings hidden by # gitleaks:allow; "
|
||||||
|
"approve only for dummy test fixtures or confirmed false positives"
|
||||||
|
),
|
||||||
|
"arrival_timestamp": datetime.datetime.now(
|
||||||
|
datetime.timezone.utc
|
||||||
|
).isoformat(),
|
||||||
|
"current_file_hash": hashlib.sha256(payload.encode("utf-8")).hexdigest(),
|
||||||
|
}
|
||||||
|
queue = Path(queue_dir)
|
||||||
|
queue.mkdir(parents=True, exist_ok=True)
|
||||||
|
path = queue / f"{proposal_id}.proposal.json"
|
||||||
|
tmp = path.with_suffix(path.suffix + ".tmp")
|
||||||
|
with tmp.open("w", encoding="utf-8") as f:
|
||||||
|
json.dump(proposal, f, indent=2)
|
||||||
|
f.write("\n")
|
||||||
|
os.chmod(tmp, 0o600)
|
||||||
|
os.replace(tmp, path)
|
||||||
|
print(proposal_id)
|
||||||
|
PY
|
||||||
|
)
|
||||||
|
rc=$?
|
||||||
|
rm -f "$report_file"
|
||||||
|
if [ "$rc" -eq 0 ] && [ -z "$proposal_id" ]; then
|
||||||
|
return 0
|
||||||
|
fi
|
||||||
|
if [ "$rc" -ne 0 ]; then
|
||||||
|
echo "git-gate: cannot route # gitleaks:allow finding to supervisor; refusing push" >&2
|
||||||
|
return 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
queue_dir=${SUPERVISE_QUEUE_DIR:-}
|
||||||
|
response_file="$queue_dir/${proposal_id}.response.json"
|
||||||
|
timeout=${SUPERVISE_GITLEAKS_ALLOW_TIMEOUT_SECONDS:-300}
|
||||||
|
case "$timeout" in
|
||||||
|
''|*[!0-9]*)
|
||||||
|
echo "git-gate: invalid SUPERVISE_GITLEAKS_ALLOW_TIMEOUT_SECONDS=$timeout" >&2
|
||||||
|
return 1
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
echo "git-gate: queued # gitleaks:allow supervisor approval $proposal_id" >&2
|
||||||
|
echo "git-gate: approve with './cli.py supervise' to continue this push" >&2
|
||||||
|
waited=0
|
||||||
|
while [ "$waited" -lt "$timeout" ]; do
|
||||||
|
if [ -f "$response_file" ]; then
|
||||||
|
status=$(python3 - "$response_file" <<'PY'
|
||||||
|
import json
|
||||||
|
import sys
|
||||||
|
try:
|
||||||
|
with open(sys.argv[1], encoding="utf-8") as f:
|
||||||
|
raw = json.load(f)
|
||||||
|
except (OSError, json.JSONDecodeError):
|
||||||
|
sys.exit(1)
|
||||||
|
status = raw.get("status")
|
||||||
|
if not isinstance(status, str):
|
||||||
|
sys.exit(1)
|
||||||
|
print(status)
|
||||||
|
PY
|
||||||
|
) || status=""
|
||||||
|
case "$status" in
|
||||||
|
approved|modified)
|
||||||
|
mkdir -p "$queue_dir/processed"
|
||||||
|
mv -f "$queue_dir/${proposal_id}.proposal.json" "$queue_dir/processed/" 2>/dev/null || true
|
||||||
|
mv -f "$queue_dir/${proposal_id}.response.json" "$queue_dir/processed/" 2>/dev/null || true
|
||||||
|
echo "git-gate: supervisor approved # gitleaks:allow for $ref" >&2
|
||||||
|
return 0
|
||||||
|
;;
|
||||||
|
rejected)
|
||||||
|
echo "git-gate: supervisor rejected # gitleaks:allow for $ref" >&2
|
||||||
|
return 1
|
||||||
|
;;
|
||||||
|
*)
|
||||||
|
echo "git-gate: invalid supervisor response for # gitleaks:allow" >&2
|
||||||
|
return 1
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
fi
|
||||||
|
sleep 1
|
||||||
|
waited=$((waited + 1))
|
||||||
|
done
|
||||||
|
echo "git-gate: supervisor approval timed out for # gitleaks:allow; refusing push" >&2
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
|
||||||
|
# Phase 1: gitleaks scan each ref's incoming commits.
|
||||||
|
while IFS=' ' read -r old new ref; do
|
||||||
|
[ -z "$ref" ] && continue
|
||||||
|
[ "$new" = "$zero" ] && continue
|
||||||
|
if [ "$old" = "$zero" ]; then
|
||||||
|
# New ref: scan only the commits this push introduces — those
|
||||||
|
# reachable from $new but not from any ref the gate already has.
|
||||||
|
# Everything already on the gate arrived via upstream mirror-fetch
|
||||||
|
# or a previously gitleaks-scanned push, so it's already-upstream
|
||||||
|
# or already-scanned; re-scanning it (the old `$new` full-ancestry
|
||||||
|
# range) only resurfaces historical findings and blocks every new
|
||||||
|
# branch. See PRD 0028 / issue #106.
|
||||||
|
log_opts="$new --not --all"
|
||||||
|
else
|
||||||
|
log_opts="$old..$new"
|
||||||
|
fi
|
||||||
|
echo "git-gate: gitleaks scanning $ref ($log_opts)" >&2
|
||||||
|
if ! gitleaks git --log-opts="$log_opts" --no-banner --redact 1>&2; then
|
||||||
|
echo "git-gate: gitleaks rejected push to $ref" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
if ! supervise_gitleaks_allow "$log_opts" "$ref"; then
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
done < "$refs_file"
|
||||||
|
|
||||||
|
# Phase 2: forward each ref to the upstream (`origin`, configured
|
||||||
|
# in the entrypoint via `git remote add --mirror=fetch`).
|
||||||
|
keyfile=$(git config --get git-gate.identityFile)
|
||||||
|
hostsfile=$(git config --get git-gate.knownHosts)
|
||||||
|
if [ ! -f "$hostsfile" ]; then
|
||||||
|
echo "git-gate: no KnownHostKey configured for this upstream; refusing to push" >&2
|
||||||
|
echo "git-gate: add KnownHostKey to the bottle.git entry and restart the bottle" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
ssh_cmd="ssh -i $keyfile -o UserKnownHostsFile=$hostsfile -o StrictHostKeyChecking=yes -o IdentitiesOnly=yes -o BatchMode=yes -o ConnectTimeout=10"
|
||||||
|
|
||||||
|
push_option_count=${GIT_PUSH_OPTION_COUNT:-0}
|
||||||
|
case "$push_option_count" in
|
||||||
|
''|*[!0-9]*)
|
||||||
|
echo "git-gate: invalid GIT_PUSH_OPTION_COUNT=$push_option_count" >&2
|
||||||
|
exit 1
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
set --
|
||||||
|
i=0
|
||||||
|
while [ "$i" -lt "$push_option_count" ]; do
|
||||||
|
opt=$(printenv "GIT_PUSH_OPTION_$i" || :)
|
||||||
|
set -- "$@" --push-option="$opt"
|
||||||
|
i=$((i + 1))
|
||||||
|
done
|
||||||
|
|
||||||
|
while IFS=' ' read -r old new ref; do
|
||||||
|
[ -z "$ref" ] && continue
|
||||||
|
if [ "$new" = "$zero" ]; then
|
||||||
|
refspec=":$ref"
|
||||||
|
elif [ "$old" != "$zero" ] && ! git merge-base --is-ancestor "$old" "$new" 2>/dev/null; then
|
||||||
|
refspec="+$new:$ref"
|
||||||
|
else
|
||||||
|
refspec="$new:$ref"
|
||||||
|
fi
|
||||||
|
echo "git-gate: forwarding $ref to origin" >&2
|
||||||
|
if ! GIT_SSH_COMMAND="$ssh_cmd" git push "$@" origin "$refspec" 1>&2; then
|
||||||
|
echo "git-gate: upstream push failed for $ref" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
done < "$refs_file"
|
||||||
|
|
||||||
|
exit 0
|
||||||
|
"""
|
||||||
|
|
||||||
|
|
||||||
|
def git_gate_render_access_hook() -> str:
|
||||||
|
"""`git daemon --access-hook` script. Runs before each protocol
|
||||||
|
service; for `upload-pack` (fetch / clone / ls-remote / pull) it
|
||||||
|
refreshes the bare repo from upstream first, so the response
|
||||||
|
reflects upstream's current state. For other services (notably
|
||||||
|
`receive-pack`) it returns 0 immediately and lets the existing
|
||||||
|
pre-receive hook gate the operation. POSIX sh.
|
||||||
|
|
||||||
|
The hook receives:
|
||||||
|
$1 service name (`upload-pack`, `receive-pack`, ...)
|
||||||
|
$2 absolute path to the resolved repo
|
||||||
|
$3 client hostname (unused)
|
||||||
|
$4 client tcp address (unused)
|
||||||
|
|
||||||
|
Fail-closed on upstream errors: the agent's fetch fails too,
|
||||||
|
so it never silently sees stale data — matches the PRD's
|
||||||
|
'equivalent to operations against the upstream' contract."""
|
||||||
|
return r"""#!/bin/sh
|
||||||
|
# git-gate access-hook (PRD 0008). $1=service $2=repo $3=host $4=peer
|
||||||
|
set -u
|
||||||
|
service=$1
|
||||||
|
repo_dir=$2
|
||||||
|
|
||||||
|
# Push path keeps its own gating in pre-receive (gitleaks +
|
||||||
|
# forward). Only refresh-from-upstream on fetch operations.
|
||||||
|
if [ "$service" != "upload-pack" ]; then
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
keyfile=$(git -C "$repo_dir" config --get git-gate.identityFile 2>/dev/null || true)
|
||||||
|
hostsfile=$(git -C "$repo_dir" config --get git-gate.knownHosts 2>/dev/null || true)
|
||||||
|
if [ -z "$keyfile" ] || [ ! -f "$hostsfile" ]; then
|
||||||
|
echo "git-gate: missing credentials for $repo_dir; refusing fetch" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
ssh_cmd="ssh -i $keyfile -o UserKnownHostsFile=$hostsfile -o StrictHostKeyChecking=yes -o IdentitiesOnly=yes -o BatchMode=yes -o ConnectTimeout=10"
|
||||||
|
|
||||||
|
echo "git-gate: refreshing $repo_dir from upstream" >&2
|
||||||
|
if ! GIT_SSH_COMMAND="$ssh_cmd" git -C "$repo_dir" fetch origin --prune >&2; then
|
||||||
|
echo "git-gate: upstream fetch failed for $repo_dir; refusing to serve stale data" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Sync the bare repo's HEAD to upstream's HEAD on the first fetch
|
||||||
|
# (when it still points at the `git init --bare` default of
|
||||||
|
# refs/heads/master and upstream uses something else, the cloned
|
||||||
|
# checkout would fail with "remote HEAD refers to nonexistent ref").
|
||||||
|
# Costs one extra ls-remote on first fetch only; subsequent fetches
|
||||||
|
# skip the branch. If upstream's default branch changes after the
|
||||||
|
# gate has cached it, restart the bottle to resync.
|
||||||
|
if ! git -C "$repo_dir" rev-parse --verify HEAD >/dev/null 2>&1; then
|
||||||
|
upstream_head=$(GIT_SSH_COMMAND="$ssh_cmd" git -C "$repo_dir" \
|
||||||
|
ls-remote --symref origin HEAD 2>/dev/null \
|
||||||
|
| awk '/^ref:/ {print $2; exit}')
|
||||||
|
if [ -n "$upstream_head" ]; then
|
||||||
|
git -C "$repo_dir" symbolic-ref HEAD "$upstream_head" || true
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
exit 0
|
||||||
|
"""
|
||||||
|
|
||||||
+34
-14
@@ -213,6 +213,20 @@ def _merge_git_user(
|
|||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def _manifest_with_merged_git_user(
|
||||||
|
agent: "ManifestAgent", raw_bottle: "ManifestBottle"
|
||||||
|
) -> "Manifest":
|
||||||
|
"""Build the single-value Manifest, overlaying the agent's git-gate.user
|
||||||
|
onto the bottle (agent wins on non-empty, per-field). Shared by the eager
|
||||||
|
and lazy load_for_agent paths."""
|
||||||
|
merged = _merge_git_user(agent.git_user, raw_bottle.git_user)
|
||||||
|
bottle = (
|
||||||
|
raw_bottle if merged == raw_bottle.git_user
|
||||||
|
else replace(raw_bottle, git_user=merged)
|
||||||
|
)
|
||||||
|
return Manifest(agent=agent, bottle=bottle)
|
||||||
|
|
||||||
|
|
||||||
def _resolve_effective_bottle_eager(
|
def _resolve_effective_bottle_eager(
|
||||||
agent_name: str,
|
agent_name: str,
|
||||||
agent: "ManifestAgent",
|
agent: "ManifestAgent",
|
||||||
@@ -468,11 +482,16 @@ class ManifestIndex:
|
|||||||
Always raises ManifestError if the agent is unknown or invalid.
|
Always raises ManifestError if the agent is unknown or invalid.
|
||||||
Backends call this at preflight inside _validate."""
|
Backends call this at preflight inside _validate."""
|
||||||
effective_bottle_names: tuple[str, ...] = bottle_names or ()
|
effective_bottle_names: tuple[str, ...] = bottle_names or ()
|
||||||
|
|
||||||
if self.home_md is None:
|
if self.home_md is None:
|
||||||
# Eager manifest (from_json_obj): data already parsed; filter to
|
return self._load_for_agent_eager(agent_name, effective_bottle_names)
|
||||||
# the one requested agent and its bottle so the returned Manifest
|
return self._load_for_agent_lazy(agent_name, effective_bottle_names)
|
||||||
# always holds exactly one agent and one bottle regardless of path.
|
|
||||||
|
def _load_for_agent_eager(
|
||||||
|
self, agent_name: str, bottle_names: tuple[str, ...]
|
||||||
|
) -> "Manifest":
|
||||||
|
"""Eager path (from_json_obj): data is already parsed; filter to the one
|
||||||
|
requested agent and its bottle so the returned Manifest always holds
|
||||||
|
exactly one agent and one bottle regardless of path."""
|
||||||
if agent_name not in self.agents:
|
if agent_name not in self.agents:
|
||||||
available = ", ".join(sorted(self.agents.keys())) or "(none)"
|
available = ", ".join(sorted(self.agents.keys())) or "(none)"
|
||||||
raise ManifestError(
|
raise ManifestError(
|
||||||
@@ -480,12 +499,16 @@ class ManifestIndex:
|
|||||||
)
|
)
|
||||||
agent = self.agents[agent_name]
|
agent = self.agents[agent_name]
|
||||||
raw_bottle = _resolve_effective_bottle_eager(
|
raw_bottle = _resolve_effective_bottle_eager(
|
||||||
agent_name, agent, effective_bottle_names, self.bottles
|
agent_name, agent, bottle_names, self.bottles
|
||||||
)
|
)
|
||||||
merged = _merge_git_user(agent.git_user, raw_bottle.git_user)
|
return _manifest_with_merged_git_user(agent, raw_bottle)
|
||||||
bottle = raw_bottle if merged == raw_bottle.git_user else replace(raw_bottle, git_user=merged)
|
|
||||||
return Manifest(agent=agent, bottle=bottle)
|
|
||||||
|
|
||||||
|
def _load_for_agent_lazy(
|
||||||
|
self, agent_name: str, bottle_names: tuple[str, ...]
|
||||||
|
) -> "Manifest":
|
||||||
|
"""Lazy path (resolve/from_md_dirs): read and parse the agent file and
|
||||||
|
its bottle chain from disk for the first time here."""
|
||||||
|
assert self.home_md is not None # guaranteed by load_for_agent dispatch
|
||||||
from .manifest_loader import scan_agent_names
|
from .manifest_loader import scan_agent_names
|
||||||
from .manifest_schema import validate_agent_frontmatter_keys
|
from .manifest_schema import validate_agent_frontmatter_keys
|
||||||
from .yaml_subset import YamlSubsetError, parse_frontmatter
|
from .yaml_subset import YamlSubsetError, parse_frontmatter
|
||||||
@@ -517,11 +540,10 @@ class ManifestIndex:
|
|||||||
agent_bottle = fm.get("bottle") or ""
|
agent_bottle = fm.get("bottle") or ""
|
||||||
bottles_dir = self.home_md / "bottles"
|
bottles_dir = self.home_md / "bottles"
|
||||||
raw_bottle = _resolve_effective_bottle_lazy(
|
raw_bottle = _resolve_effective_bottle_lazy(
|
||||||
agent_name, str(agent_bottle), effective_bottle_names, bottles_dir
|
agent_name, str(agent_bottle), bottle_names, bottles_dir
|
||||||
)
|
)
|
||||||
effective_bottle_name = (
|
effective_bottle_name = (
|
||||||
effective_bottle_names[-1] if effective_bottle_names
|
bottle_names[-1] if bottle_names else str(agent_bottle)
|
||||||
else str(agent_bottle)
|
|
||||||
)
|
)
|
||||||
|
|
||||||
# Build and validate the full ManifestAgent.
|
# Build and validate the full ManifestAgent.
|
||||||
@@ -539,9 +561,7 @@ class ManifestIndex:
|
|||||||
known = {effective_bottle_name} if effective_bottle_name else set()
|
known = {effective_bottle_name} if effective_bottle_name else set()
|
||||||
agent = ManifestAgent.from_dict(agent_name, agent_dict, known)
|
agent = ManifestAgent.from_dict(agent_name, agent_dict, known)
|
||||||
|
|
||||||
merged_user = _merge_git_user(agent.git_user, raw_bottle.git_user)
|
return _manifest_with_merged_git_user(agent, raw_bottle)
|
||||||
bottle = raw_bottle if merged_user == raw_bottle.git_user else replace(raw_bottle, git_user=merged_user)
|
|
||||||
return Manifest(agent=agent, bottle=bottle)
|
|
||||||
|
|
||||||
def has_agent(self, name: str) -> bool:
|
def has_agent(self, name: str) -> bool:
|
||||||
return name in self.agents
|
return name in self.agents
|
||||||
|
|||||||
@@ -1,247 +0,0 @@
|
|||||||
# PRD prd-new: Egress control plane — metering, budgets, and forced cutoff
|
|
||||||
|
|
||||||
- **Status:** Draft
|
|
||||||
- **Author:** didericis
|
|
||||||
- **Created:** 2026-06-25
|
|
||||||
- **Issue:** #251
|
|
||||||
|
|
||||||
## Summary
|
|
||||||
|
|
||||||
Add an **out-of-band egress enforcement & observability plane**: meter every
|
|
||||||
agent's token usage at the egress proxy, decrement budgets without the agent's
|
|
||||||
cooperation, and forcibly cut a bottle's egress when a budget is exhausted —
|
|
||||||
either automatically or on command from a host-level dashboard. The trigger
|
|
||||||
(usage threshold) and the action (route-drop / freeze / kill) both live in the
|
|
||||||
egress plane and run with no agent in the loop. This is distinct from the
|
|
||||||
supervise sidecar (PRD 0013), which is agent-initiated and therefore cannot
|
|
||||||
enforce a cost cutoff on a runaway agent. State (usage ledger, budgets, audit)
|
|
||||||
moves into a host-level SQLite database behind a thin repository API, the first
|
|
||||||
SQL store in an otherwise flat-file repo.
|
|
||||||
|
|
||||||
## Problem
|
|
||||||
|
|
||||||
bot-bottle can't currently do two things the cost-overrun case demands:
|
|
||||||
|
|
||||||
1. **Forced egress shutdown on limit.** When an agent crosses a token
|
|
||||||
threshold, kill its egress automatically — no human in the loop.
|
|
||||||
2. **Remote (host-level) management.** Drive agents from a single surface:
|
|
||||||
see usage, cut egress, stop bottles, to prevent cost overruns.
|
|
||||||
|
|
||||||
The existing supervise sidecar (PRD 0013) is **entirely agent-initiated**: every
|
|
||||||
action begins with the agent voluntarily calling an MCP tool and an operator
|
|
||||||
approving it. A runaway or expensive agent — exactly the cost-overrun case —
|
|
||||||
will never call `egress-block` on itself. Supervision is therefore a
|
|
||||||
**collaborative recovery** mechanism, not an **enforcement** mechanism; making
|
|
||||||
it mandatory (#249) would not deliver forced cost-cutoff.
|
|
||||||
|
|
||||||
The requirement forces a distinction the current design blurs:
|
|
||||||
|
|
||||||
- **Plane A — enforcement / observability (this PRD).** System → infrastructure.
|
|
||||||
Meter usage, cut egress on threshold or command, account for cost.
|
|
||||||
Out-of-band; independent of the agent. **Unconditional** — an enforcement
|
|
||||||
plane you can opt out of isn't enforcement.
|
|
||||||
- **Plane B — agent-facing recovery (the existing supervise sidecar).**
|
|
||||||
Agent → operator, approval-gated. Useful interactively; meaningless for a
|
|
||||||
headless agent with no operator watching its queue. Remains optional.
|
|
||||||
|
|
||||||
This PRD builds Plane A. It reframes the "always-on control" invariant of #249
|
|
||||||
as "the egress control plane is always present" — a more defensible property
|
|
||||||
than "every agent runs the agent-facing supervisor." Unsupervised
|
|
||||||
(headless/CI/ephemeral) agents stay first-class: still subject to the mandatory
|
|
||||||
meter + kill switch, they simply lack the agent-facing proposal tools they
|
|
||||||
couldn't use anyway.
|
|
||||||
|
|
||||||
## Goals / Success Criteria
|
|
||||||
|
|
||||||
- The egress proxy meters every request to a metered API host (e.g.
|
|
||||||
`api.anthropic.com`) and records authoritative token usage per bottle and per
|
|
||||||
agent provider, with no agent cooperation.
|
|
||||||
- A budget can be set at four scopes with deterministic precedence
|
|
||||||
(**agent → bottle → parent bottle → global host budget**); the
|
|
||||||
most-specific applicable budget governs.
|
|
||||||
- When usage crosses a budget, the bottle's configured **cutoff policy**
|
|
||||||
(`cutoff` | `freeze` | `kill`) fires automatically, executed host-side on the
|
|
||||||
egress plane — never via the supervise queue.
|
|
||||||
- An operator can, from a single **host-level TUI dashboard**, see live per-bottle
|
|
||||||
usage against budget and command a cutoff/stop on demand.
|
|
||||||
- Host budgets, default cutoff policy, and per-provider limits are declared in a
|
|
||||||
new host-level `~/.bot-bottle/settings.yml`, parseable by `yaml_subset.py`.
|
|
||||||
- All usage, budget state, and enforcement actions persist in a host-level
|
|
||||||
SQLite DB behind a thin repository API, so the store can later be swapped for
|
|
||||||
a cross-host cloud service.
|
|
||||||
|
|
||||||
## Non-goals
|
|
||||||
|
|
||||||
- **Remote control / cross-host control plane.** Web + mobile remote control,
|
|
||||||
cross-host budgets, and the authn/transport they require are explicitly
|
|
||||||
deferred. v1 is a **host-only TUI** with no remote surface.
|
|
||||||
- **Dollar-denominated budgets.** Budgets are token counts keyed by agent
|
|
||||||
provider, not currency. Price tables are out of scope.
|
|
||||||
- **Migrating existing flat-file state into SQLite.** Resume `metadata.json`,
|
|
||||||
transcripts, Dockerfile overrides, the supervise queue, and audit logs stay on
|
|
||||||
the filesystem. Only the *new* metering/budget/enforcement ledger is SQL.
|
|
||||||
- **Making the supervise sidecar (Plane B) mandatory.** Out of scope here; this
|
|
||||||
PRD is the answer to "what should be unconditional" (Plane A), leaving #249's
|
|
||||||
Plane-B question open.
|
|
||||||
- **Per-request hard pre-send blocking as the primary mechanism.** The gate is
|
|
||||||
budget-crossing detected at/after metering; a pre-flight estimator (below) is a
|
|
||||||
refinement, not the core enforcement path.
|
|
||||||
|
|
||||||
## Design
|
|
||||||
|
|
||||||
### Two measurements: gate vs. account
|
|
||||||
|
|
||||||
There are two distinct needs, and they want different signals:
|
|
||||||
|
|
||||||
- **Account (authoritative).** Decrement the real budget from the API
|
|
||||||
**response**, which already carries authoritative usage (Anthropic
|
|
||||||
`input_tokens` / `output_tokens`, OpenAI `usage`). The egress addon already
|
|
||||||
has a `response(flow)` hook (`bot_bottle/egress_addon.py:460`), so the real
|
|
||||||
number is available with no extra network call. **Caveat:** agent traffic is
|
|
||||||
mostly streaming SSE, so the response path must tail the stream for the final
|
|
||||||
usage event rather than parse a single JSON body — scoped explicitly as work.
|
|
||||||
- **Gate (estimate).** To block *before* sending, only the request is available,
|
|
||||||
so an estimator / provider `count_tokens` endpoint is the only option.
|
|
||||||
|
|
||||||
Calling `count_tokens` for accounting would be both less accurate *and* an extra
|
|
||||||
metered egress call per request, so accounting uses response `usage` and the
|
|
||||||
estimator is reserved for the optional pre-flight gate.
|
|
||||||
|
|
||||||
### `count_tokens` on agent providers
|
|
||||||
|
|
||||||
Add an abstract `count_tokens(request) -> int` to the `AgentProvider`
|
|
||||||
abstraction (`bot_bottle/agent_provider.py`):
|
|
||||||
|
|
||||||
- **Default** is a good-enough stdlib estimator. Prefer stdlib only; a small
|
|
||||||
pip dependency *for the sidecar* is acceptable for the fallback if stdlib
|
|
||||||
proves too inaccurate (this does not relax the package's stdlib-first stance —
|
|
||||||
it would be a sidecar-only dep, like the bundle already carries).
|
|
||||||
- **Built-in `claude`** uses Anthropic's token-counting endpoint;
|
|
||||||
**built-in `codex`** uses OpenAI's. These are exact for the gate but cost a
|
|
||||||
metered call, so they are gate-only; accounting still comes from the response.
|
|
||||||
|
|
||||||
### Budgets and precedence
|
|
||||||
|
|
||||||
Budgets are token counts keyed by **agent provider name** (the same names
|
|
||||||
bottles already use). Four scopes, most-specific wins:
|
|
||||||
|
|
||||||
```
|
|
||||||
agent → bottle → parent bottle → global (host)
|
|
||||||
```
|
|
||||||
|
|
||||||
The global host budget is the highest-priority feature to ship (the cross-host
|
|
||||||
control plane will eventually consume it); per-agent and per-bottle budgets
|
|
||||||
override it for finer control. A budget can also be supplied **at bottle
|
|
||||||
launch** (`--budget` or equivalent), overriding the settings.yml defaults for
|
|
||||||
that run. Enforcement evaluates the effective budget as the
|
|
||||||
nearest-defined scope at decrement time.
|
|
||||||
|
|
||||||
### `~/.bot-bottle/settings.yml`
|
|
||||||
|
|
||||||
New **host-level** settings file (the `~/.bot-bottle/` root, *not* the per-repo
|
|
||||||
`.bot-bottle/` — host budgets must not be committed per-repo). Parsed by
|
|
||||||
`yaml_subset.py`, so it must stay within that bounded subset (flat mappings,
|
|
||||||
scalars; no anchors, no multi-line block scalars). Shape:
|
|
||||||
|
|
||||||
```yaml
|
|
||||||
budget:
|
|
||||||
claude: 5000000 # token budget keyed by agent provider
|
|
||||||
codex: 2000000
|
|
||||||
shutdown: cutoff # default cutoff policy: cutoff | freeze | kill
|
|
||||||
```
|
|
||||||
|
|
||||||
### Forced cutoff and cutoff policy
|
|
||||||
|
|
||||||
On budget exhaustion (or an operator command), the configured per-bottle cutoff
|
|
||||||
policy fires. The three policies map onto primitives that already exist:
|
|
||||||
|
|
||||||
- **`cutoff`** (default) — drop the bottle's `routes.yaml` to empty and reload
|
|
||||||
(or isolate the bottle from the egress network); the agent/bottle keeps
|
|
||||||
running but can no longer reach metered hosts. This is the route-drop already
|
|
||||||
available on the egress plane (`bot_bottle/backend/egress_apply.py`).
|
|
||||||
- **`freeze`** — commit/snapshot state, then kill the agent/bottle; resumable
|
|
||||||
later via `bot_bottle/backend/freeze.py`.
|
|
||||||
- **`kill`** — tear the bottle down without saving state (backend teardown).
|
|
||||||
|
|
||||||
The trigger lives in the metering path and the action in the egress/backend
|
|
||||||
plane; **neither touches the supervise proposal queue** (design constraint from
|
|
||||||
#251).
|
|
||||||
|
|
||||||
### Host-level SQLite store
|
|
||||||
|
|
||||||
**Decision: introduce SQLite now, narrowly.**
|
|
||||||
|
|
||||||
- **The dependency objection doesn't apply.** `sqlite3` is in the Python stdlib,
|
|
||||||
so it does not break the AGENTS.md stdlib-first / no-runtime-pip stance — same
|
|
||||||
category as the hand-rolled `yaml_subset.py`, except the stdlib already ships
|
|
||||||
the whole engine.
|
|
||||||
- **It fits the problem.** A *global* token budget decremented concurrently by N
|
|
||||||
egress sidecars (today `~/.bot-bottle/` already has `state/`, `audit/`,
|
|
||||||
`queue/` written by parallel bottles) is a read-modify-write race. Over JSON
|
|
||||||
that means hand-rolled file locking; SQLite gives atomic transactions + WAL for
|
|
||||||
free. The per-agent/per-bottle precedence rollup plus "sum across all bottles"
|
|
||||||
is a `GROUP BY`, not an N-directory rescan.
|
|
||||||
- **It rehearses the cloud swap.** "Wrap operations in an API so we can swap to a
|
|
||||||
cloud service" maps directly onto a thin repository/DAO over SQLite → Postgres
|
|
||||||
later. A JSON-file store is a worse rehearsal than SQL.
|
|
||||||
|
|
||||||
**Costs (real but bounded):** a new paradigm in a flat-file repo needs a
|
|
||||||
`schema_version` table + idempotent startup migrations; SQLite serializes
|
|
||||||
writers, so WAL mode + `busy_timeout` are required (a non-issue at a handful of
|
|
||||||
bottles); test fixtures need temp DBs.
|
|
||||||
|
|
||||||
**Scope of the store:** one DB at `~/.bot-bottle/bot-bottle.db` behind a thin
|
|
||||||
repository API. Only the **new** metering/budget/enforcement-audit ledger lives
|
|
||||||
there. Existing per-bottle blobs (resume `metadata.json`, transcripts,
|
|
||||||
Dockerfile overrides, supervise queue) stay on the filesystem — migrating them
|
|
||||||
now is churn for no benefit and they lack the concurrency/aggregation problem.
|
|
||||||
|
|
||||||
### Host-level controller + dashboard
|
|
||||||
|
|
||||||
A single **host-level controller** owns the meter, budget evaluation, and the
|
|
||||||
cutoff actions across all bottles (cf. `bot_bottle/cli/supervise.py`'s
|
|
||||||
cross-bottle view), rather than a per-bottle daemon. v1 ships one host-level
|
|
||||||
**TUI dashboard** that reads live usage-vs-budget from the SQLite store and
|
|
||||||
offers on-demand cutoff/stop. The existing supervisor UI should eventually fold
|
|
||||||
into this same dashboard; this PRD lays the host-level surface it will move to.
|
|
||||||
|
|
||||||
## Implementation chunks
|
|
||||||
|
|
||||||
Ordered, individually mergeable:
|
|
||||||
|
|
||||||
1. **SQLite repository foundation.** `~/.bot-bottle/bot-bottle.db`, schema +
|
|
||||||
`schema_version` migrations, WAL + `busy_timeout`, thin repository API,
|
|
||||||
temp-DB test fixtures. No behavior wired yet.
|
|
||||||
2. **Metering at the egress proxy.** Parse authoritative response `usage`
|
|
||||||
(including SSE final-usage tailing) in the egress addon `response` hook;
|
|
||||||
write per-bottle / per-provider usage rows to the ledger.
|
|
||||||
3. **`settings.yml` + budget model.** Host-level `~/.bot-bottle/settings.yml`
|
|
||||||
parsed by `yaml_subset.py`; budget precedence (agent → bottle → parent →
|
|
||||||
global) and the `--budget` launch flag.
|
|
||||||
4. **Forced cutoff + cutoff policy.** Wire the threshold trigger to the
|
|
||||||
`cutoff` / `freeze` / `kill` primitives on the egress/backend plane; record
|
|
||||||
enforcement actions to the audit ledger.
|
|
||||||
5. **Host-level TUI dashboard.** Live usage-vs-budget view + on-demand
|
|
||||||
cutoff/stop, reading the store.
|
|
||||||
6. **`count_tokens` pre-flight gate (optional refinement).** Abstract method +
|
|
||||||
stdlib estimator default; Anthropic/OpenAI endpoints for built-in
|
|
||||||
claude/codex; optional pre-send block.
|
|
||||||
|
|
||||||
## Open questions
|
|
||||||
|
|
||||||
- **SSE usage tailing robustness.** Buffering streamed responses to extract the
|
|
||||||
final usage event without breaking the agent's own stream consumption — how
|
|
||||||
much of the body must the addon hold, and what's the failure mode if the
|
|
||||||
stream is interrupted mid-flight?
|
|
||||||
- **Crossing mid-request.** A single response can push usage past budget only
|
|
||||||
*after* it's already been delivered. Is post-hoc cutoff (next request blocked)
|
|
||||||
sufficient, or is a pre-flight estimator gate (chunk 6) required for v1?
|
|
||||||
- **Provider name ↔ metered host mapping.** How does the proxy attribute a
|
|
||||||
flow to an agent-provider budget key — by destination host, by bottle
|
|
||||||
identity, or both?
|
|
||||||
- **Parent-bottle budget semantics.** For `bottle extends` (PRD 0025 / 0065)
|
|
||||||
chains, does "parent bottle" mean the manifest parent, the launching bottle,
|
|
||||||
or the full ancestry summed?
|
|
||||||
- **Dashboard ↔ controller transport (even host-only).** In-process, a local
|
|
||||||
socket, or polling the SQLite store directly? Picks the seam the future remote
|
|
||||||
control plane will extend.
|
|
||||||
@@ -17,6 +17,8 @@ bot_bottle/manifest_egress.py
|
|||||||
bot_bottle/manifest_agent.py
|
bot_bottle/manifest_agent.py
|
||||||
bot_bottle/manifest_schema.py
|
bot_bottle/manifest_schema.py
|
||||||
bot_bottle/git_gate.py
|
bot_bottle/git_gate.py
|
||||||
|
bot_bottle/git_gate_render.py
|
||||||
|
bot_bottle/git_gate_provision.py
|
||||||
bot_bottle/git_http_backend.py
|
bot_bottle/git_http_backend.py
|
||||||
bot_bottle/supervise.py
|
bot_bottle/supervise.py
|
||||||
bot_bottle/yaml_subset.py
|
bot_bottle/yaml_subset.py
|
||||||
|
|||||||
@@ -0,0 +1,37 @@
|
|||||||
|
"""Unit-test package init.
|
||||||
|
|
||||||
|
Isolates ``HOME`` to a throwaway directory for the entire unit suite so
|
||||||
|
no test ever reads or writes the real ``~/.bot-bottle`` (state, queue,
|
||||||
|
and audit dirs all derive from ``supervise.bot_bottle_root()`` →
|
||||||
|
``Path.home()``). Without this, a test that takes a ``flock`` on the
|
||||||
|
real audit log can **block indefinitely** when a live bottle's supervise
|
||||||
|
sidecar holds that lock — observed as a hung ``coverage run`` at 0% CPU —
|
||||||
|
and unisolated tests otherwise pollute the developer's home dir.
|
||||||
|
|
||||||
|
Individual tests that need their own ``HOME`` still override
|
||||||
|
``os.environ['HOME']`` and restore it; they now restore to this isolated
|
||||||
|
dir rather than the real one, so isolation holds either way. Tests that
|
||||||
|
patch ``supervise.bot_bottle_root`` directly are unaffected.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import atexit
|
||||||
|
import os
|
||||||
|
import shutil
|
||||||
|
import tempfile
|
||||||
|
|
||||||
|
_real_home = os.environ.get("HOME")
|
||||||
|
_tmp_home = tempfile.mkdtemp(prefix="bot-bottle-unit-home.")
|
||||||
|
os.environ["HOME"] = _tmp_home
|
||||||
|
|
||||||
|
|
||||||
|
def _restore_home() -> None:
|
||||||
|
if _real_home is None:
|
||||||
|
os.environ.pop("HOME", None)
|
||||||
|
else:
|
||||||
|
os.environ["HOME"] = _real_home
|
||||||
|
shutil.rmtree(_tmp_home, ignore_errors=True)
|
||||||
|
|
||||||
|
|
||||||
|
atexit.register(_restore_home)
|
||||||
|
|||||||
@@ -281,6 +281,17 @@ class TestEncodedVariants(unittest.TestCase):
|
|||||||
v = self._variants()
|
v = self._variants()
|
||||||
self.assertEqual(len(v), len(set(v)))
|
self.assertEqual(len(v), len(set(v)))
|
||||||
|
|
||||||
|
def test_repeated_calls_equal(self):
|
||||||
|
# Memoization must not change observable output.
|
||||||
|
self.assertEqual(self._variants(), self._variants())
|
||||||
|
|
||||||
|
def test_returns_fresh_list_each_call(self):
|
||||||
|
# Callers mutate/iterate the result; the cached set must not be
|
||||||
|
# exposed by reference, or one caller could corrupt another's view.
|
||||||
|
first = self._variants()
|
||||||
|
first.append("MUTATED")
|
||||||
|
self.assertNotIn("MUTATED", self._variants())
|
||||||
|
|
||||||
|
|
||||||
class TestUnicodeNormalization(unittest.TestCase):
|
class TestUnicodeNormalization(unittest.TestCase):
|
||||||
def test_fullwidth_chars_normalized(self):
|
def test_fullwidth_chars_normalized(self):
|
||||||
|
|||||||
@@ -367,7 +367,7 @@ class TestDynamicKeyProvisioning(unittest.TestCase):
|
|||||||
|
|
||||||
def test_resolve_identity_file_gitea_provisions_key(self):
|
def test_resolve_identity_file_gitea_provisions_key(self):
|
||||||
entry = self._gitea_manifest().bottles["dev"].git[0]
|
entry = self._gitea_manifest().bottles["dev"].git[0]
|
||||||
with patch("bot_bottle.git_gate._provision_dynamic_key", return_value="/tmp/provisioned-key") as mock_provision:
|
with patch("bot_bottle.git_gate_provision._provision_dynamic_key", return_value="/tmp/provisioned-key") as mock_provision:
|
||||||
self.assertEqual("/tmp/provisioned-key", _resolve_identity_file(entry, "demo", self.stage))
|
self.assertEqual("/tmp/provisioned-key", _resolve_identity_file(entry, "demo", self.stage))
|
||||||
mock_provision.assert_called_once()
|
mock_provision.assert_called_once()
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,112 @@
|
|||||||
|
"""Unit: lazy (on-disk) ManifestIndex loader branches (coverage ratchet).
|
||||||
|
|
||||||
|
The eager from_json_obj path is covered by test_manifest_validation.py;
|
||||||
|
this drives the lazy resolve()/from_md_dirs path — all_agent_names with a
|
||||||
|
cwd overlay, load_for_agent on an unknown / malformed agent file, and
|
||||||
|
require_agent's names-only file-existence checks — so manifest.py's
|
||||||
|
core-module coverage doesn't depend on the integration suite."""
|
||||||
|
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import os
|
||||||
|
import shutil
|
||||||
|
import tempfile
|
||||||
|
import textwrap
|
||||||
|
import unittest
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
from bot_bottle.manifest import ManifestError, ManifestIndex
|
||||||
|
|
||||||
|
|
||||||
|
def _write(p: Path, text: str) -> None:
|
||||||
|
p.parent.mkdir(parents=True, exist_ok=True)
|
||||||
|
p.write_text(textwrap.dedent(text).lstrip("\n"))
|
||||||
|
|
||||||
|
|
||||||
|
_BOTTLE_DEV = """
|
||||||
|
---
|
||||||
|
egress:
|
||||||
|
routes:
|
||||||
|
- host: example.com
|
||||||
|
---
|
||||||
|
The dev bottle.
|
||||||
|
"""
|
||||||
|
|
||||||
|
_AGENT = """
|
||||||
|
---
|
||||||
|
bottle: dev
|
||||||
|
---
|
||||||
|
An agent.
|
||||||
|
"""
|
||||||
|
|
||||||
|
# Tab in the frontmatter indent -> YamlSubsetError on parse.
|
||||||
|
_AGENT_BAD_FM = "---\nskills:\n\t- x\n---\nbody\n"
|
||||||
|
|
||||||
|
|
||||||
|
class _LazyCase(unittest.TestCase):
|
||||||
|
def setUp(self) -> None:
|
||||||
|
self.home_root = Path(tempfile.mkdtemp(prefix="cb-home-"))
|
||||||
|
self.cwd_root = Path(tempfile.mkdtemp(prefix="cb-cwd-"))
|
||||||
|
self._orig_home = os.environ.get("HOME")
|
||||||
|
os.environ["HOME"] = str(self.home_root)
|
||||||
|
|
||||||
|
def tearDown(self) -> None:
|
||||||
|
if self._orig_home is None:
|
||||||
|
os.environ.pop("HOME", None)
|
||||||
|
else:
|
||||||
|
os.environ["HOME"] = self._orig_home
|
||||||
|
shutil.rmtree(self.home_root, ignore_errors=True)
|
||||||
|
shutil.rmtree(self.cwd_root, ignore_errors=True)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def home_cb(self) -> Path:
|
||||||
|
return self.home_root / ".bot-bottle"
|
||||||
|
|
||||||
|
@property
|
||||||
|
def cwd_cb(self) -> Path:
|
||||||
|
return self.cwd_root / ".bot-bottle"
|
||||||
|
|
||||||
|
def resolve(self) -> ManifestIndex:
|
||||||
|
return ManifestIndex.resolve(str(self.cwd_root))
|
||||||
|
|
||||||
|
|
||||||
|
class TestAllAgentNamesLazy(_LazyCase):
|
||||||
|
def test_merges_home_and_cwd_agents(self) -> None:
|
||||||
|
_write(self.home_cb / "bottles" / "dev.md", _BOTTLE_DEV)
|
||||||
|
_write(self.home_cb / "agents" / "alpha.md", _AGENT)
|
||||||
|
_write(self.cwd_cb / "agents" / "beta.md", _AGENT)
|
||||||
|
self.assertEqual(["alpha", "beta"], self.resolve().all_agent_names)
|
||||||
|
|
||||||
|
|
||||||
|
class TestLoadForAgentLazy(_LazyCase):
|
||||||
|
def test_unknown_agent_raises(self) -> None:
|
||||||
|
_write(self.home_cb / "agents" / "alpha.md", _AGENT)
|
||||||
|
with self.assertRaises(ManifestError):
|
||||||
|
self.resolve().load_for_agent("nope")
|
||||||
|
|
||||||
|
def test_malformed_frontmatter_raises(self) -> None:
|
||||||
|
_write(self.home_cb / "bottles" / "dev.md", _BOTTLE_DEV)
|
||||||
|
_write(self.home_cb / "agents" / "broken.md", _AGENT_BAD_FM)
|
||||||
|
with self.assertRaises(ManifestError):
|
||||||
|
self.resolve().load_for_agent("broken")
|
||||||
|
|
||||||
|
|
||||||
|
class TestRequireAgentLazy(_LazyCase):
|
||||||
|
def test_existing_home_agent_ok(self) -> None:
|
||||||
|
_write(self.home_cb / "agents" / "alpha.md", _AGENT)
|
||||||
|
self.resolve().require_agent("alpha") # no raise
|
||||||
|
|
||||||
|
def test_existing_cwd_agent_ok(self) -> None:
|
||||||
|
# File only under cwd -> require_agent's cwd_path branch.
|
||||||
|
_write(self.home_cb / "agents" / "alpha.md", _AGENT)
|
||||||
|
_write(self.cwd_cb / "agents" / "beta.md", _AGENT)
|
||||||
|
self.resolve().require_agent("beta") # no raise
|
||||||
|
|
||||||
|
def test_unknown_agent_raises(self) -> None:
|
||||||
|
_write(self.home_cb / "agents" / "alpha.md", _AGENT)
|
||||||
|
with self.assertRaises(ManifestError):
|
||||||
|
self.resolve().require_agent("nope")
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
unittest.main()
|
||||||
Reference in New Issue
Block a user