Files
rose-ash/scripts/sx-fix-writeback.sh
giles 72e461cf2c scripts: forge write-back — close the git→gitea→agentic loop
The forge already DRIVES sessions (briefing → tmux launch, sx-fix-up.sh).
This records what comes BACK, making the forge a true system of record:

- sx-fix-writeback.sh <forge-agent> [kind] [base-ref]: reads new commits on
  loops/sx-<slug>, appends a record per commit to writeback.sxsrc (idempotent,
  matched by sha), then rebuilds the forge + replays them as agentic-sx
  commit!s on agents/<forge-agent> and re-dumps forge.sxdata.
- forge-build.sxsrc: fb-writeback-records / fb-replay-writeback / fb-do-writeback
  — each real-git commit becomes an agentic-sx commit whose tree is a small
  commit.sx pointer (sha/branch/message/files); real git holds the code, the
  forge holds the index, so the CID stays small.
- writeback.sxsrc: the append-only record log (source of truth for what's
  been recorded); replayed chronologically so agent branch heads advance right.

Verified live: the sx-gate loop's first real commit (f09368e1, "pin K18
expt-overflow float-promotion") is now recorded as a test-kind agentic-sx
commit on agents/ws-W14 (session log: spawn → finding → writeback), its
commit.sx pointing back at the real-git sha.

Loop closed: forge → tmux (drive) and tmux → real-git → forge (record).

Co-Authored-By: Claude Fable 5 <noreply@anthropic.com>
2026-07-03 22:36:59 +00:00

90 lines
4.2 KiB
Bash
Executable File

#!/usr/bin/env bash
# sx-fix-writeback.sh — close the loop: record new real-git commits from a
# fix branch back into the rose-ash/sx-review forge as agentic-sx commits.
#
# The forge already DRIVES the tmux session (briefing -> launch, sx-fix-up.sh).
# This records what comes BACK, so the forge is a true system of record: each
# commit on loops/sx-<slug> becomes an agentic-sx commit on agents/<forge-agent>,
# with a small commit.sx pointer (sha/message/files) — real git holds the code.
#
# Usage: ./scripts/sx-fix-writeback.sh <forge-agent> [kind] [base-ref]
# forge-agent agentic-sx agent, e.g. ws-W14 (branch = loops/sx-<slug of agent>)
# kind agentic commit kind: test|refactor|finding|decision (default: test)
# base-ref commits are <base-ref>..<branch> (default: architecture)
#
# Idempotent: commits already recorded in writeback.sxsrc (matched by sha) are skipped.
# Re-runs forge-build + replay and re-dumps forge.sxdata.
set -euo pipefail
ROOT="$(cd "$(dirname "$0")/.." && pwd)"; cd "$ROOT"
AGENT="${1:?usage: sx-fix-writeback.sh <forge-agent> [kind] [base-ref]}"
KIND="${2:-test}"
BASE="${3:-architecture}"
SLUG="$(echo "$AGENT" | tr '[:upper:]' '[:lower:]' | tr -c 'a-z0-9' '-' | sed 's/-*$//')"
BRANCH="loops/sx-${SLUG}"
WB="plans/sx-review/writeback.sxsrc"
DUMP="plans/sx-review/forge.sxdata"
BIN="hosts/ocaml/_build/default/bin/sx_server.exe"
git show-ref --verify --quiet "refs/heads/$BRANCH" || { echo "no branch $BRANCH"; exit 1; }
[ -f "$WB" ] || { echo "missing $WB"; exit 1; }
# sanitize a commit subject for embedding in an SX string literal
sanitize() { echo "$1" | tr -d '"\\' | tr '\n' ' ' | cut -c1-120; }
added=0
# oldest-first so replay advances the branch head in order
while IFS=$'\x1f' read -r sha subj; do
[ -n "$sha" ] || continue
if grep -q ":sha \"$sha\"" "$WB"; then continue; fi
files="$(git diff-tree --no-commit-id --name-only -r "$sha" | tr '\n' ' ' | sed 's/ *$//')"
msg="$(sanitize "$subj")"
rec=" {:agent \"$AGENT\" :kind \"$KIND\" :sha \"$sha\" :branch \"$BRANCH\" :message \"$msg\" :files \"$files\"}"
# insert before the END-RECORDS marker (keeps chronological order)
awk -v r="$rec" '/;;; END-RECORDS/ && !done {print r; done=1} {print}' "$WB" > "$WB.tmp" && mv "$WB.tmp" "$WB"
echo "recorded $sha $msg"
added=$((added+1))
done < <(git log --reverse --format='%h%x1f%s' "$BASE..$BRANCH")
if [ "$added" -eq 0 ]; then
echo "no new commits on $BRANCH to write back (all recorded)."
exit 0
fi
echo "rebuilding forge with $added new write-back record(s) + re-dumping $DUMP ..."
RAW="$(mktemp)"
{
echo '(epoch 1)'
for f in spec/stdlib.sx lib/r7rs.sx lib/persist/event.sx lib/persist/backend.sx \
lib/persist/log.sx lib/persist/kv.sx lib/artdag/dag.sx \
lib/datalog/tokenizer.sx lib/datalog/parser.sx lib/datalog/unify.sx \
lib/datalog/db.sx lib/datalog/builtins.sx lib/datalog/aggregates.sx \
lib/datalog/strata.sx lib/datalog/eval.sx lib/datalog/api.sx lib/datalog/magic.sx \
lib/git/object.sx lib/git/ref.sx lib/git/dag.sx lib/git/worktree.sx \
lib/git/diff.sx lib/git/merge.sx lib/git/porcelain.sx \
lib/relations/schema.sx lib/relations/engine.sx lib/relations/api.sx \
lib/relations/explain.sx lib/relations/federation.sx lib/relations/tree.sx \
lib/agentic/schema.sx lib/agentic/branch.sx lib/gitea/repo.sx; do
echo "(load \"$f\")"
done
echo '(epoch 2)'
echo '(load "plans/sx-review/forge-build.sxsrc")'
echo '(load "plans/sx-review/writeback.sxsrc")' # overrides fb-writeback-records
echo '(epoch 3)'
echo '(eval "(len (fb-do-writeback))")' # replay -> agentic commits
echo '(epoch 4)'
echo '(eval "(fb-forge-serialize)")'
} | timeout 300 "$BIN" > "$RAW" 2>&1
REPLAYED="$(grep -oE '\(ok 3 [0-9]+\)|\(ok-len 3 [0-9]+\)' "$RAW" | head -1)"
python3 - "$RAW" > "$DUMP" <<'PY'
import sys, re
data = open(sys.argv[1],'rb').read()
m = re.search(rb'\(ok-len 4 (\d+)\)\n', data)
if not m:
sys.stderr.write("FAILED: no dump in forge output\n"); sys.exit(1)
n = int(m.group(1)); sys.stdout.buffer.write(data[m.end():m.end()+n])
PY
echo "replay result: $REPLAYED ; forge.sxdata now $(wc -c < "$DUMP") bytes"
rm -f "$RAW"
echo "done. review: git diff plans/sx-review/{writeback.sxsrc,forge.sxdata}"