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>
90 lines
4.2 KiB
Bash
Executable File
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}"
|