Apply CLAUDE.md "Format — registries ALWAYS caveman" rule to existing entries via /caveman:compress. Drops articles, filler, and hedging while preserving: - Technical terms (rtk, claude plugin install, framer-motion, etc.) - IDs unchanged (BDR-XXX, LRN-XXX, BLK-XXX, EVAL-XXX) - Dates unchanged (2026-04-22 etc.) - Code blocks and quoted error strings exact - Commit refs (892de28,7b57b2e,d3c79f0,64d6ca7) Files: decisions.md, learnings.md, blockers.md, journal.md, evals.md. Token reduction: ~40% on session-start memory load. Pre-compression backups saved as *.original.md (gitignored next). Co-Authored-By: Claude <noreply@anthropic.com>
10 KiB
10 KiB
| type | entry_prefix | schema | rules | ||||||||||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
| learnings_registry | LRN |
|
|
Learnings registry (LRN)
Index
| ID | Date | Pattern | Applies to |
|---|---|---|---|
| LRN-001 | 2026-04-22 | rtk shape-compression breaks pipes |
any pipeline chaining rtk curl/cat/read into jq, python -c, awk |
| LRN-002 | 2026-04-23 | Moving report-file paths requires grepping bash READS, not just WRITES | any refactor that moves a generated file used by a dispatcher |
| LRN-003 | 2026-04-27 | Claude Code disable* settings use sentinel string "disable", not boolean |
any change to permissions.defaultMode or related blocker keys |
| LRN-004 | 2026-04-27 | framer-motion rebranded motion Nov 2024 — different packages per framework |
any new project recommending animation lib; auditing legacy imports |
| LRN-005 | 2026-05-03 | claude plugin install does NOT enable — separate claude plugin enable required |
every plugin installer targeting ALWAYS-ON status |
| LRN-006 | 2026-05-03 | caveman-shrink (and any MCP middleware proxy) non-functional without upstream wrapper |
any MCP middleware/proxy package — never claude mcp add it bare |
LRN-001 — rtk shape-compression silently breaks downstream parsers
- Date: 2026-04-22
- Pattern: when tracking tool (
rtk) intercepts stdout and returns schematized/compressed representation instead of raw payload, every downstream parser breaks silently — user (or LLM) never seesrtk's output, only parser error. - Context:
rtk curlreplaces raw JSON output with tokenized version, regardless of TTY vs pipe. Claude Code hooks auto-rewritecurl→rtk curl, so behavior impossible to anticipate without knowing hook. - Future application: for any tool auto-rewriting standard commands, explicitly verify pipe behavior. Documented workaround:
exclude_commands=["curl"]in~/.config/rtk/config.toml, orrtk proxy. SeeBLK-001.
LRN-002 — Moving report-file paths requires grepping bash READS, not just WRITES
- Date: 2026-04-23
- Pattern: when moving write path of generated file (report, artifact, cache), must also grep places that READ that file — not only those that write it. Dispatchers (orchestrator skills dispatching to agent then parsing result) typically contain bash commands like
test -s X.md,grep ... X.md,wc -l X.md— refs invisible if only grep for "write" or "output path". - Context:
.claude/audits/refactor (commit5c5e82c). First pass: updated write paths across 5 skills (seo/geo/harden/validate/code-clean) and 3 agents. User asked for verify-gate. They re-grepped, found 10+ bare bash refs (e.g.test -s HARDEN.md,grep -oE ... VALIDATE.md) missed — dispatchers broken (looking at project root while agent writing to.claude/audits/). Fixed in commit5c5e82c(bundled with same commit). - Future application:
- Before declaring file-path migration "complete", grep basename (
grep -rn "HARDEN\.md") plus full path — catch bare bash usages. - If file used in pipelines (
test,grep,wc,cat,head), search for those verbs explicitly. - Verify-gates save work: one extra round forced exhaustive re-grepping. Without it, two dispatchers shipped broken.
- Before declaring file-path migration "complete", grep basename (
LRN-003 — Claude Code disable* settings use sentinel string "disable", not boolean
- Date: 2026-04-27
- Pattern: Claude Code blocker-style settings (
disableAutoMode,disableBypassPermissionsMode) use literal string"disable"as sentinel. Key absent = feature available; value"disable"turns blocker on. Any other value (includingfalse,true,null) has no effect — doc explicitly states this. - Context: switching
permissions.defaultModeto"auto"whiledisableAutoMode: "disable"still present would have failed at startup ("auto mode unavailable"). Namingdisable<Foo>: "disable"reads ambiguously — easy to assume boolean toggle and leave key in place. - Future application:
- Before changing
defaultMode, audit matchingdisable*key in samepermissionsblock. If present with value"disable", remove it. - Same logic for
bypassPermissionsmode anddisableBypassPermissionsMode. - Don't trust doc's naming — read value semantics. Sentinel strings beat booleans here because harness can distinguish "unset" from "explicitly off" (admin policy).
- Before changing
- Reference: commit
1421578, dochttps://code.claude.com/docs/en/settings.
LRN-004 — framer-motion rebranded motion (Nov 2024) — different packages per framework
- Date: 2026-04-27
- Pattern:
framer-motionrenamedmotionNovember 2024. Rename not cosmetic: bundles React (motion/react), Svelte, vanilla-JS support under single npm package, while Vue gets own parallel packagemotion-v. Legacy packageframer-motionstill installs and works but in maintenance mode — recommending it in new framework default locks projects into legacy import paths day one. Detection of "is animation already covered" must include both names plus broader anim ecosystem (gsap,lottie-react,react-spring,popmotion,@formkit/auto-animate) to avoid double-installs. - Context: building animation-lib auto-install in
/init-projectand/onboard. Initial user phrasing "framer-motion" (old name remembered). Picking package name without verifying rename would have shipped legacy imports in every new scaffold. - Future application:
- For React / Next.js / Remix / Astro+React / Svelte:
motion(import { motion } from 'motion/react'). - For Vue 3 / Nuxt:
motion-v(separate package, separate API). - For React Native: do NOT recommend
motion— usereact-native-reanimated(motion targets DOM). - When auditing existing projects, check both
framer-motionandmotionkeys inpackage.jsondeps; treat either as "animation already covered". - Before adopting any "industry default" lib in framework, verify canonical package name current — naming churn (rebrand, scope change
@org/lib, fork) common in JS land.
- For React / Next.js / Remix / Astro+React / Svelte:
- Reference: helper
lib/animation-lib-check.sh, BDR-005.
LRN-005 — claude plugin install does NOT enable — claude plugin enable separate step
- Date: 2026-05-03
- Pattern: Claude Code CLI splits "available" from "active" for marketplace plugins.
claude plugin install --scope user name@sourceonly copies plugin into~/.claude/plugins/cache/<marketplace>/<plugin>/<version>/. Does NOT writename@source: trueinto user'ssettings.json:enabledPluginsmap. Without explicitclaude plugin enable name@source, plugin sits dormant — installed but unloaded. Symmetric withclaude plugin disable, which keeps cache and only removes enabledPlugins entry. - Context: discovered auditing why
security-guidanceandsuperpowerswere ✘ disabled inclaude plugin listdespite project'sinstall-plugins.shsummary banner declaring them "ALWAYS ON". Root cause:install_plugin()only ranclaude plugin install, neverenable. Bug stayed invisible because hardcodedprintf "│ ✅ ON : security-guidance rtk superpowers │"insession-start.shprinted same names regardless of actual state — lying banner agreed with lying install. - Future application:
- For any plugin meant ALWAYS ON, follow
claude plugin installwithclaude plugin enable name@source(idempotent — no-op if already enabled). - Detect "actually enabled" via
enabledPlugins[name@source] === trueinsettings.json, NOT presence of cache dir. Pattern implemented inlib/detect-plugins.sh:plugin_enabled()(filesystem grep, no subprocess). - Any banner / status display claiming plugin on must read state, never hardcode names. Hardcoded labels turn single bug into two co-conspiring bugs masking each other.
- For any plugin meant ALWAYS ON, follow
- Reference: commit
2ec7935,lib/detect-plugins.sh:plugin_enabled,install-plugins.sh:enable_plugin().
LRN-006 — caveman-shrink (and any MCP middleware proxy) needs upstream wrapper to function
- Date: 2026-05-03
- Pattern: some MCP packages are middleware proxies, not standalone servers. They wrap upstream MCP server and transform its responses (e.g.
caveman-shrinkcompresses prose fields). Running them bare viaclaude mcp add proxy-name -- npx -y proxy-pkgregisters server that errors immediately with "missing upstream command" — every health check fails, and Claude Code reports MCP broken until human intervenes. CLIclaude mcp adddoesn't validate that configured command launches working stdio MCP, so bad registration silently lands. - Context: when adding caveman, upstream installer auto-registers
claude mcp add caveman-shrink -- npx -y caveman-shrinkand prints "registered. wrap an upstream by editing the mcpServers entry". Following that flow leaves user with permanently failing MCP entry until they realize they must edit~/.claude.jsonmanually. - Future application:
- For any MCP that is proxy/middleware (read package docs for "upstream", "wraps", "proxy"), register under DERIVED name
<proxy>-<upstream>with upstream baked into args. Example for caveman-shrink wrapping filesystem server:claude mcp add caveman-shrink-fs --scope user -- \ npx -y caveman-shrink npx -y @modelcontextprotocol/server-filesystem /path - Detection of "is this MCP correctly set up?" must look for the derived name (
caveman-shrink-*), not the bare proxy name. Bare-name registration is treated as broken. - Default install scripts should NOT auto-register middleware MCPs — print the snippet for the user to choose an upstream. See
install-plugins.shSTEP 5.5.
- For any MCP that is proxy/middleware (read package docs for "upstream", "wraps", "proxy"), register under DERIVED name
- Reference: commit
9b20b84,lib/detect-plugins.sh:detect_caveman_shrink,install-plugins.shSTEP 5.5 MCP block.