loop-init.sh 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. #!/usr/bin/env bash
  2. # Scaffold an outer-loop state spine (loop.config.yaml + STATE.md + run-log.md).
  3. #
  4. # Usage: loop-init.sh --name NAME [OPTIONS]
  5. # Input: argv flags only (no stdin).
  6. # Output: stdout = the created loop.config.yaml path (data). Under --dry-run, the
  7. # path then the rendered config. Data only.
  8. # Stderr: the creation panel, reminders, warnings, errors.
  9. # Exit: 0 created (or dry-run rendered), 2 usage, 3 template/dir not found,
  10. # 5 precondition (target dir already populated, no --force)
  11. #
  12. # Creates <dir>/<name>/ from the bundled templates, substituting name/pattern/tier/
  13. # cadence/permission_mode. Never clobbers a populated loop dir. Atomic writes.
  14. # Next step: fill the config, then `loop-audit.sh <dir>/<name>/loop.config.yaml`.
  15. #
  16. # Examples:
  17. # loop-init.sh --name pr-babysitter --pattern pr-babysitter --tier L1
  18. # loop-init.sh --name dep-sweeper --pattern dependency-sweeper --tier L2 --cadence 1d
  19. # loop-init.sh --name nightly --cadence "0 3 * * *" --dry-run
  20. set -uo pipefail
  21. readonly EX_OK=0 EX_USAGE=2 EX_NOTFOUND=3 EX_PRECOND=5
  22. # Terminal design system (skills/_lib/term.sh). stdout = the created path (data);
  23. # the creation panel frames on stderr, so detect color on fd 2. Degrade to plain
  24. # stderr lines if the shared lib is unreachable.
  25. __lib="$(cd "$(dirname "${BASH_SOURCE[0]}")/../../_lib" 2>/dev/null && pwd || true)"
  26. if [ -n "${__lib:-}" ] && [ -f "$__lib/term.sh" ]; then . "$__lib/term.sh"; term_init 2
  27. else
  28. term_panel_open() { :; }; term_panel_close() { :; }; term_panel_vert() { :; }
  29. term_status_row() { shift; printf ' - %s %s\n' "$1" "${2:-}"; }
  30. term_alert() { shift; printf ' ! %s\n' "$*"; }
  31. term_color() { shift; printf '%s' "$*"; }; TERM_DOT="|"
  32. fi
  33. HERE="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
  34. ASSETS="$HERE/../assets"
  35. CFG_TPL="$ASSETS/loop.config.template.yaml"
  36. STATE_TPL="$ASSETS/STATE.template.md"
  37. RUN_TPL="$ASSETS/run.template.md"
  38. # ── defaults ────────────────────────────────────────────────────────────────
  39. NAME=""
  40. PATTERN="custom"
  41. TIER="L1"
  42. CADENCE="1h"
  43. DIR=".loops"
  44. DRY_RUN=0
  45. FORCE=0
  46. usage() {
  47. cat <<'EOF'
  48. loop-init.sh — scaffold an outer-loop state spine.
  49. Usage:
  50. loop-init.sh --name NAME [OPTIONS]
  51. Options:
  52. --name NAME loop identifier, kebab-case (required). Names the directory.
  53. --pattern KEY catalog key (pr-babysitter, ci-sweeper, dependency-sweeper,
  54. changelog-drafter, post-merge-cleanup, issue-triage,
  55. daily-triage) or "custom" (default: custom).
  56. --tier L1|L2|L3 starting autonomy tier (default: L1).
  57. --cadence STR 10m | 1h | 6h | 1d, or a cron string (default: 1h).
  58. --dir DIR parent directory for the loop (default: .loops).
  59. --dry-run print the target path + rendered config; write nothing.
  60. --force overwrite an already-populated <dir>/<name>/ directory.
  61. -h, --help show this help and exit 0.
  62. Exit codes:
  63. 0 created (or dry-run) 2 usage 3 template/dir not found 5 dir populated
  64. Examples:
  65. loop-init.sh --name pr-babysitter --pattern pr-babysitter --tier L1
  66. loop-init.sh --name dep-sweeper --pattern dependency-sweeper --tier L2 --cadence 1d
  67. loop-init.sh --name nightly --cadence "0 3 * * *" --dry-run
  68. EOF
  69. }
  70. die_usage() { printf 'error: %s\n' "$1" >&2; echo >&2; usage >&2; exit "$EX_USAGE"; }
  71. # ── parse args ──────────────────────────────────────────────────────────────
  72. while [[ $# -gt 0 ]]; do
  73. case "$1" in
  74. --name) [[ $# -ge 2 ]] || die_usage "--name needs a value"; NAME="$2"; shift 2 ;;
  75. --pattern) [[ $# -ge 2 ]] || die_usage "--pattern needs a value"; PATTERN="$2"; shift 2 ;;
  76. --tier) [[ $# -ge 2 ]] || die_usage "--tier needs a value"; TIER="$2"; shift 2 ;;
  77. --cadence) [[ $# -ge 2 ]] || die_usage "--cadence needs a value"; CADENCE="$2"; shift 2 ;;
  78. --dir) [[ $# -ge 2 ]] || die_usage "--dir needs a value"; DIR="$2"; shift 2 ;;
  79. --dry-run) DRY_RUN=1; shift ;;
  80. --force) FORCE=1; shift ;;
  81. -h|--help) usage; exit "$EX_OK" ;;
  82. -*) die_usage "unknown flag: $1" ;;
  83. *) die_usage "unexpected positional argument: $1" ;;
  84. esac
  85. done
  86. # ── validate ────────────────────────────────────────────────────────────────
  87. [[ -n "$NAME" ]] || die_usage "--name is required"
  88. [[ "$NAME" =~ ^[a-z0-9]+(-[a-z0-9]+)*$ ]] || die_usage "--name must be kebab-case (got '$NAME')"
  89. [[ "$PATTERN" =~ ^[a-z0-9]+(-[a-z0-9]+)*$ ]] || die_usage "--pattern must be kebab-case (got '$PATTERN')"
  90. case "$TIER" in L1|L2|L3) ;; *) die_usage "--tier must be L1|L2|L3 (got '$TIER')" ;; esac
  91. # cadence: Nm/Nh/Nd OR a cron-ish string (digits, spaces, * / , -)
  92. [[ "$CADENCE" =~ ^[0-9]+[mhd]$ || "$CADENCE" =~ ^[-0-9*/,\ ]+$ ]] \
  93. || die_usage "--cadence must be like 10m/1h/1d or a cron string (got '$CADENCE')"
  94. [[ -f "$CFG_TPL" ]] || { printf 'error: config template not found at %s\n' "$CFG_TPL" >&2; exit "$EX_NOTFOUND"; }
  95. [[ -f "$STATE_TPL" ]] || { printf 'error: STATE template not found at %s\n' "$STATE_TPL" >&2; exit "$EX_NOTFOUND"; }
  96. [[ -f "$RUN_TPL" ]] || { printf 'error: run template not found at %s\n' "$RUN_TPL" >&2; exit "$EX_NOTFOUND"; }
  97. # Default permission_mode from tier (the workhorse mapping; see references/risk-tiers.md).
  98. case "$TIER" in
  99. L1|L2) PMODE="dontAsk" ;;
  100. L3) PMODE="bypassPermissions" ;;
  101. esac
  102. TARGET_DIR="$DIR/$NAME"
  103. CFG_OUT="$TARGET_DIR/loop.config.yaml"
  104. STATE_OUT="$TARGET_DIR/STATE.md"
  105. LOG_OUT="$TARGET_DIR/run-log.md"
  106. RUN_OUT="$TARGET_DIR/run.md"
  107. # Refuse a populated target unless --force.
  108. if [[ -d "$TARGET_DIR" ]] && [[ -n "$(ls -A "$TARGET_DIR" 2>/dev/null)" ]] && [[ "$FORCE" -ne 1 ]]; then
  109. printf 'error: loop directory already populated: %s (use --force to overwrite)\n' "$TARGET_DIR" >&2
  110. exit "$EX_PRECOND"
  111. fi
  112. NOW="$(date -u +%Y-%m-%dT%H:%M:%SZ)"
  113. # ── render config from template ─────────────────────────────────────────────
  114. # Line-anchored sed substitutions: identity placeholders globally, the three
  115. # tunable scalar lines by their default value. Kill-switch path carries <loop-name>.
  116. render_config() {
  117. sed -E \
  118. -e "s|<loop-name>|$NAME|g" \
  119. -e "s|<pattern-key>|$PATTERN|" \
  120. -e "s|^tier: L1|tier: $TIER|" \
  121. -e "s|^cadence: 1h|cadence: $CADENCE|" \
  122. -e "s|^permission_mode: dontAsk|permission_mode: $PMODE|" \
  123. "$CFG_TPL"
  124. }
  125. render_state() {
  126. sed -E \
  127. -e "s|<loop-name>|$NAME|g" \
  128. -e "s|<ISO-8601 Z>|$NOW|" \
  129. "$STATE_TPL"
  130. }
  131. render_log() {
  132. cat <<EOF
  133. # $NAME — run log (append-only; one line per run)
  134. # format: <ISO-Z> run#N action=<reported|proposed|none> <key=val…> outcome=<…> tokens=<N>
  135. EOF
  136. }
  137. render_run() {
  138. sed -E \
  139. -e "s|<loop-name>|$NAME|g" \
  140. -e "s|tier <L1\\|L2\\|L3>|tier $TIER|g" \
  141. "$RUN_TPL"
  142. }
  143. # ── dry-run: print and stop ─────────────────────────────────────────────────
  144. if [[ "$DRY_RUN" -eq 1 ]]; then
  145. printf '%s\n' "$CFG_OUT"
  146. {
  147. term_panel_open loop "loop ${TERM_DOT} init (dry-run)" "$NAME"
  148. term_panel_vert
  149. term_status_row skip "would create $TARGET_DIR/" "tier $TIER ${TERM_DOT} $PATTERN ${TERM_DOT} $CADENCE"
  150. term_status_row skip " loop.config.yaml" "permission_mode: $PMODE"
  151. term_status_row skip " STATE.md / run-log.md / run.md" ""
  152. term_panel_vert
  153. term_panel_close "nothing written" ""
  154. } >&2
  155. render_config
  156. exit "$EX_OK"
  157. fi
  158. # ── atomic writes ───────────────────────────────────────────────────────────
  159. mkdir -p "$TARGET_DIR" || { printf 'error: could not create %s\n' "$TARGET_DIR" >&2; exit 1; }
  160. write_atomic() { # write_atomic <dest> <content>
  161. local dest="$1" content="$2" tmp
  162. tmp="$dest.tmp.$$"
  163. printf '%s\n' "$content" > "$tmp" || { printf 'error: failed to write %s\n' "$tmp" >&2; exit 1; }
  164. mv -f "$tmp" "$dest" || { rm -f "$tmp"; printf 'error: failed to move into place: %s\n' "$dest" >&2; exit 1; }
  165. }
  166. write_atomic "$CFG_OUT" "$(render_config)"
  167. write_atomic "$STATE_OUT" "$(render_state)"
  168. write_atomic "$LOG_OUT" "$(render_log)"
  169. write_atomic "$RUN_OUT" "$(render_run)"
  170. printf '%s\n' "$CFG_OUT"
  171. {
  172. term_panel_open loop "loop ${TERM_DOT} init" "$NAME"
  173. term_panel_vert
  174. term_status_row ok "created $TARGET_DIR/" "tier $TIER ${TERM_DOT} $PATTERN ${TERM_DOT} $CADENCE"
  175. term_status_row ok " loop.config.yaml" "permission_mode: $PMODE"
  176. term_status_row ok " STATE.md / run-log.md / run.md" ""
  177. if [[ "$TIER" != "L1" ]]; then
  178. term_alert warning "tier $TIER needs a verify gate, guard, worktree, escalation + land_via — fill them before auditing"
  179. fi
  180. term_panel_vert
  181. term_panel_close "then: fill the config ${TERM_DOT} loop-audit.sh $CFG_OUT" ""
  182. } >&2
  183. exit "$EX_OK"