RED-7 (example-priming): the STEP-2 worked example named live IDs (LRN-014 + LRN-016) and modeled merging them — but they are complementary (header-ids vs checkbox-CSS), a merge the skill's own rule forbids. Live IDs in an example prime the skill to act on those exact entries on real data. Fictionalized the whole STEP-2 example to 9xx IDs (cannot match a live registry); the merge example now models a same-concept merge. Closed by a DETERMINISTIC test (run-deterministic.sh RED-7: the example must carry only 9xx ids) per LRN-046, not a flaky behavioral fixture. The test caught its own ugrep false-green first (a leading-dash pattern parsed as an option) — fixed via /usr/bin/grep, the same dodge the skill's verify already uses at line 189. RED-8 (added-negation inversion): re-reviewed, consciously accepted as a documented limit in BACKLOG — remote (compression subtracts tokens), and an FP-safe increase check is non-trivial (needs the HEAD entry-id set to exclude legit new/merged 0->N); a noisy guard is worse than the honest limit on a destructive skill (LRN-047). Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com> Claude-Session: https://claude.ai/code/session_01C6bUdvHnajCNzgVQefZowj
14 KiB
| name | description | argument-hint | allowed-tools | ||||||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
| prune-memory | Use when .claude/memory/ registries grow too large or noisy — superseded entries verbose, similar entries cluttering, journal stale, caveman style drifted. Curates the 5 registries via mark-superseded + merge + inline caveman compression. Append-only safe (no hard delete). Git is the backup. Triggers: "prune memory", "compact memory", "clean memory", "memory hygiene", "trier memoire", "nettoyer memoire", "registres trop longs", "compresse les memoires". |
|
|
/prune-memory — Memory registry curation
Operates on .claude/memory/ in the current project (CWD). Curates the
5 registries: decisions.md, learnings.md, blockers.md, journal.md,
evals.md.
Core principles
- Git is the backup. Skill writes in-place. PRECHECK refuses to run if working tree dirty on registry files.
- Append-only friendly. Marks entries
status: superseded by <new-ID>orstatus: deprecatedinstead of deleting. Body of old entry stays for history. - IDs stable. Never renumber. Merges create a new ID; sources keep their ID with superseded status.
- Caveman style enforced. Per CLAUDE.md memory rule, all writes are caveman-style English. Compression rewrites prose to fragments.
- User approves every category. No silent changes.
Quick reference
| When | Use |
|---|---|
| Add new entry this session | /close |
| Curate: obsolete + merge + caveman | /prune-memory (this skill) |
STEP 0 — PRECHECK
test -d .claude/memory/ || { echo "no .claude/memory/ in $(pwd)"; exit 1; }
# RED-2 guard: a dirty tree is a HARD stop, enforced in-band (not a prose
# "STOP"). Git is the only backup; refuse to write over uncommitted state.
if [ -n "$(git status --short .claude/memory/ 2>/dev/null)" ]; then
git status --short .claude/memory/
echo "DIRTY: commit or stash .claude/memory/ first. Git is the only backup."
exit 1
fi
If working tree is dirty on any registry file → STOP with: "Commit or
stash pending changes in .claude/memory/ first. Skill writes in-place.
Git is the only backup."
STEP 1 — AUDIT (per registry)
For each target registry (filter by $ARGUMENTS or all 5):
Read file. Classify candidates into A/B/C/D below. Use today's date for age comparisons. Today's date is in the system context.
A. Obsolete — mark-superseded candidates
- Decisions
status: proposedolder than 90 days → proposestatus: deprecated(no follow-up). - Decisions whose body contains "superseded by " but Index row still
says
accepted→ propose Index row fix tosuperseded by <that-ID>. - Blockers
status: openwhose root cause matches a commit in last 30 days (grepgit log --since=30.days --grep=<keyword>) → proposestatus: resolvedwith commit ref. - Journal entries older than 180 days with zero cross-reference from
later entries → propose collapse into 1-line month summary
(
## YYYY-MMheading replaces detail). SAFETY-CRITICAL EXCEPTION (deterministic): an entry whose body holds an operational permanent rule is INTOUCHABLE — never collapse, summarize, or reword it, regardless of age or cross-reference. Trigger: any line withNEVER/ALWAYS/PERMANENT, or a negation + imperative (must not,do not,never deploy…). The detail IS the value; keep it verbatim.
B. Similar — merge candidates
- Two+ entries sharing root keyword in title (e.g.
pandoc,client-handover,CSS overlay). - Shared file paths in
**Reference**:lines. - Same week + adjacent IDs + same domain.
- Use semantic judgment on overlap; don't merge complementary entries that cover different angles of one concept.
C. Bloated — inline caveman-rewrite candidates
- Body > 150 words AND prose-heavy.
- Detect filler density: count
\b(the|a|an|just|really|basically|actually|simply)\bmatches; if > 5% of word count → bloated.
D. Index drift
- Body
## (BDR|LRN|BLK|EVAL)-NNNheading exists but no matching row in## Indextable → propose Index backfill. - Index row exists but body entry missing → propose
status: deleted (orphaned)tombstone or removal of Index row (user decides).
STEP 2 — PRESENT PLAN ★ MANDATORY STOP
Print one block per registry. Example:
PRUNE PLAN — decisions.md (N entries → M after if approved)
(IDs below are FICTIONAL — 9xx range, never a live registry entry — so this
worked example cannot PRIME a real prune. Apply the same shapes to real IDs.)
[A. Obsolete — mark superseded]
BDR-901 — example proposed decision — status: proposed since <90+ days ago>
→ mark: status: deprecated (no follow-up after 90 days)
BDR-902 — example decision — body says superseded by BDR-903
→ fix Index: status = "superseded by BDR-903"
[B. Similar — merge]
LRN-901 + LRN-902 — SAME concept (two notes on the identical bug)
→ propose: merge into NEW LRN-904 with both bodies appended +
caveman pass; sources marked status: superseded by LRN-904
(merge ONLY same-concept entries — complementary/different-angle stays split)
[C. Bloated — inline caveman rewrite]
BDR-902 — body 612 words, filler density 7.2% → ~380 expected (-38%)
[D. Index drift]
(none)
Approve per category? (all / a / b / c / d / edit <ID> / skip)
Wait for user input. Default = nothing applied.
STEP 3 — APPLY APPROVED CHANGES
Order: safe → destructive.
- Index drift fixes — no body changes. Backfill missing rows, mark orphans.
- Status flag updates — Index row status field only. Body untouched.
- Merges — write new merged entry (next-free ID); source IDs marked
status: superseded by <new-ID>in Index (body kept verbatim for history). Merged body:- Preserves all
**Reference**:lines (dedupe identical paths). - Caveman pass on prose during merge.
- Keeps frontmatter fields: id (new), date (today), title, status (accepted), references (union).
- Preserves all
- Inline caveman compression — preserve frontmatter exactly (id,
date, title, status, references). Rewrite prose body to fragments:
- NEGATION GUARD (deterministic, overrides every rule below): never
rewrite a sentence containing a negation token (
not,never,no,cannot, or any*n'tcontraction). Keep such sentences VERBATIM — dropping a filler next to anot/nevercan silently invert meaning. Compression touches negation-free sentences only. - Drop articles (
a,an,the) — negation-free sentences only. - Drop filler (
just,really,basically,actually,simply) — negation-free sentences only. - Short synonyms (
bignotextensive,fixnotimplement a solution for). - Keep code blocks, URLs, error messages, file paths VERBATIM.
- Keep IDs (BDR-XXX, LRN-XXX, commit hashes) verbatim.
- NEGATION GUARD (deterministic, overrides every rule below): never
rewrite a sentence containing a negation token (
After each write, regenerate Index from body when rows changed.
STEP 4 — VERIFY
# Filename → ID-prefix map. Hard-mapped because filenames don't share
# their first 3 chars with the prefix (decisions → BDR, not DEC).
# A prior version derived the prefix via `basename | cut -c1-3`, which never
# matched any heading and made verify a no-op (false-clean signal).
declare -A PREFIX_MAP=(
[decisions]=BDR
[learnings]=LRN
[blockers]=BLK
[evals]=EVAL
)
# All body entries have Index rows; no orphans
for fname in decisions learnings blockers evals; do
f=".claude/memory/${fname}.md"
[ -f "$f" ] || continue
prefix="${PREFIX_MAP[$fname]}"
/usr/bin/grep -oE "^## (${prefix})-[0-9]+" "$f" | while read marker; do
id="${marker##\#\# }"
/usr/bin/grep -q "^| ${id} " "$f" || echo "MISSING INDEX: $id in $f"
done
/usr/bin/grep -oE "^\| (${prefix})-[0-9]+ " "$f" | while read row; do
id=$(echo "$row" | awk '{print $2}')
# RED-6 fix: match id at a word boundary (space OR end-of-line) so a
# title-less heading "## BDR-009" is not flagged as a false orphan.
/usr/bin/grep -qE "^## ${id}( |\$)" "$f" || echo "ORPHAN INDEX: $id in $f"
done
done
# RED-5 fidelity guard (count-based, per-entry x per-category). STEP 0 ensured
# a clean tree, so git HEAD is the pre-prune backup. Fails the run if any
# negation/permanent token COUNT drops within an entry vs HEAD -- immune to the
# line-sharing false positives a removed-line grep produces. The STEP 3.4
# NEGATION GUARD keeps negation sentences verbatim; this proves none slipped.
# Journal entries are date-keyed and legitimately collapse, so the journal is
# restricted to {never,always,permanent} -- the markers the STEP 1.A safety
# exception protects from collapse (keys stay stable; casual not/no in a benign
# collapsed entry is not a loss). Contraction *n't is covered upstream by A.
census() { # reads a registry file on stdin -> "KEY:CAT<TAB>COUNT" per entry
awk '
/^## /{ id=$2 }
{ L=tolower($0); gsub(/[^a-z]+/," ",L); n=split(L,w," ")
for(i=1;i<=n;i++){ c=w[i]
if(c=="never") a[id":never"]++
else if(c=="always") a[id":always"]++
else if(c=="permanent") a[id":perm"]++
else if(c=="cannot") a[id":cannot"]++
else if(c=="not") a[id":not"]++
else if(c=="no") a[id":no"]++ } }
END{ for(k in a) if(a[k]>0) print k"\t"a[k] }'
}
fidelity_check() { # $1 = registry basename; returns 1 (and prints) on a drop
local fname="$1" f=".claude/memory/$1.md" cats drop
[ -f "$f" ] || return 0
git diff --quiet -- "$f" 2>/dev/null && return 0
if [ "$fname" = journal ]; then cats='never|always|perm'
else cats='never|always|perm|cannot|not|no'; fi
# Tag working "W" / HEAD "H" explicitly -- NOT NR==FNR, which misclassifies
# when the working census is empty (a fully-deleted safety entry = the case
# we most need to catch).
drop=$( { census < "$f" | awk '{print "W\t"$0}'
git show HEAD:"$f" | census | awk '{print "H\t"$0}'
} | awk -F'\t' -v cats="^($cats)\$" '
$1=="W" { w[$2]=$3; next }
{ n=split($2,p,":"); if (p[n] !~ cats) next
if ((w[$2]+0) < $3) print " "$2" (HEAD="$3" now="(w[$2]+0)")" }')
if [ -n "$drop" ]; then
echo "FIDELITY FAIL ($f): a negation/permanent token dropped within an entry:"
printf '%s\n' "$drop"; return 1
fi
return 0
}
FIDFAIL=0
for fname in decisions learnings blockers journal evals; do
fidelity_check "$fname" || FIDFAIL=1
done
[ "$FIDFAIL" = 1 ] && echo "Do NOT certify this run. Revert with: git checkout .claude/memory/"
echo "(blank above = OK)"
wc -l .claude/memory/*.md | grep -v "\.original\.md"
Report:
PRUNE COMPLETE
decisions.md : 226 → 184 lines (-19%)
learnings.md : 190 → 165 lines (-13%)
blockers.md : no candidates
journal.md : 88 → 62 lines (-30%)
evals.md : no candidates
INDEX SANITY: OK
NEXT: review `git diff .claude/memory/`, then `/commit-change`
What NOT to prune
- Journal entries < 30 days old.
- Decisions / learnings with commit references < 14 days old.
- Entries the user marked
status: acceptedin the current session. - The current session's just-capitalized entries (read
journal.mdtail to identify them).
Common mistakes
| Mistake | Fix |
|---|---|
| Renumbering IDs after a merge | IDs are stable. Sources keep their ID + status: superseded. New merged entry gets next-free ID. |
| Hard-deleting "obsolete" entries | Forbidden by append-only rule. Use status: deprecated. Body stays. |
| Compressing code blocks / URLs / error messages | Caveman compression touches PROSE only. Code, URLs, IDs, error quotes stay verbatim. |
| Running on a dirty working tree | PRECHECK blocks this. Commit first. |
| Compressing the current session's journal entry | Excluded by "What NOT to prune" — current session capitalization is still useful. |
| Merging complementary entries that cover different angles | Merge only when same concept, same scope. Different angles = keep separate. |
Failure paths
| Situation | Behavior |
|---|---|
.claude/memory/ missing |
STOP: no .claude/memory/ in current directory; run from project root |
| Working tree dirty on registry files | STOP per PRECHECK; tell user to commit/stash first |
User says skip at STEP 2 |
Exit cleanly, no writes |
| Merge produces an entry > 600 words | Re-split — merge was too greedy. Re-prompt user to keep separate. |
| Index sanity FAILED at STEP 4 | Print exact missing/orphan IDs. Do NOT auto-fix — user re-runs or hand-edits. |
| Caveman compression result < 20% of original AND original had code blocks | Revert that entry's compression — flag as needing manual rewrite (likely stripped technical detail). |
| Same file already compressed in same session | Skip C-category for that file; warn user that double-pass risks technical drift. |
Rules
- No silent writes — every change goes through STEP 2 approval gate.
- No renumbering — IDs are stable across all operations.
- No hard delete in v1 — only mark superseded. (Hard delete opt-in may arrive in v2 if explicit demand surfaces.)
- Working tree must be clean before any write — git is the only backup.
- Caveman compression touches prose only; code/URLs/error quotes verbatim per CLAUDE.md memory format rule.
TDD note (skill itself)
v1 ships without baseline test scenarios per superpowers:writing-skills Iron Law. Recommended before relying on the skill in production:
- RED: spawn subagent, give it a real
.claude/memory/snapshot, ask "prune obsolete entries". Document what it does naturally. - GREEN: invoke
/prune-memoryon the same snapshot. Verify it follows STEP 0–4 + respects append-only rule. - REFACTOR: log any new rationalizations the subagent finds; add counters to the "Common mistakes" / "Failure paths" tables.
Until TDD is done, the skill is v1-untested. STEP 2 approval gate is the human safety net.