janitor

Automated transcript trimming, LLM memory extraction, and session hygiene for OpenClaw gateways. Keeps transcripts from bloating, extracts structured memories before archiving, and prunes stale sessions.

Safety Notice

This listing is from the official public ClawHub registry. Review SKILL.md and referenced scripts before running.

Copy this and send it to your AI assistant to learn

Install skill "janitor" with this command: npx skills add halfdeadcat/session-janitor

Session Janitor

Automated transcript and session hygiene for OpenClaw gateways.

What It Does

  • Sidecar offloader — extracts large tool outputs (≥5KB) to .toolcache/ files, replacing inline content with lightweight stubs. Runs before trim on every turn.
  • Trims oversized transcripts — keeps recent exchanges + synthetic compaction summary, archives the rest
  • LLM memory extraction — uses the gateway's chat completions API to extract structured memories (facts, decisions, lessons) from trimmed content before discarding
  • Prunes stale sessions — removes old subagent/cron session entries from sessions.json
  • Archives orphan transcripts — files with no active session after a grace period
  • Cleans old archives — removes pre-trim, reset, and checkpoint files after retention period
  • Cleans orphaned checkpoints — removes *.checkpoint.*.jsonl files left behind by auto-compaction once the parent session no longer has an active transcript

Setup

bash skills/session-janitor/scripts/setup.sh

Setup auto-discovers all gateway installations (~/.openclaw/, ~/.openclaw-*/), generates config.json, installs a cron job, and installs the watcher service (systemd on Linux, launchd on macOS).

macOS Prerequisites

Install fswatch for the per-turn watcher:

brew install fswatch

The cron-based janitor works without it (trimming every 15 min instead of within ~3s of each turn).

For architecture details, trimming mechanics, and token math, see ARCHITECTURE.md.

Configuration

Edit config.json after setup to tune thresholds:

KeyDefaultDescription
trimMaxKB250Trim transcripts larger than this. Trim always fires when exceeded — there is no minimum pair count.
keepPairs10Number of recent user/assistant pairs to keep after trim. If fewer pairs exist than this, all are kept (no skip).
keepFullPairs2Most recent N assistant turns that keep full toolResult bodies (older turns are collapsed to summary lines)
minArchivePairs5Informational only — no longer blocks trim. Sessions exceeding trimMaxKB are always trimmed.
trimFullThresholdPct50Two-stage aggressive reduction when trimmed output still exceeds this % of trimMaxKB: (1) strip all assistant turns (tool args + thinking removed); (2) if still over threshold, drop all toolResult entries entirely. Set to 100 to disable.
archiveRetentionDays7Delete old archives after this many days (applies to .reset.*, .deleted.*, .pre-trim.*, .bak-*, .purged.*, .emergency-*)
keepPreTrimFiles3Max .pre-trim.* files to keep per session (oldest deleted immediately, regardless of retention period)
orphanGraceMinutes30Wait before archiving orphan transcripts
staleSubagentHours24Prune subagent session entries older than this
staleCronSessionHours24Prune cron session entries older than this
llmExtraction.enabledtrueUse LLM to extract memories from trimmed content
llmExtraction.modelopenclawModel identifier to use for extraction calls
llmExtraction.maxInputChars20000Max characters of archived content sent to LLM
llmExtraction.timeoutSecs60LLM API call timeout in seconds
llmExtraction.maxMemories15Max memories to accept from a single LLM extraction
llmExtraction.minArchived3Minimum archived messages required before running LLM extraction
llmExtraction.maxPerRun1Max LLM extractions per cron cycle (cost control)
memCli.enabledfalseStore extracted memories via mem CLI (requires mem)
cronSchedule*/15 * * * *How often to run
sidecar.enabledtrueOffload large toolResult entries to .toolcache/ files
sidecar.minEntryBytes5120Minimum toolResult content size (bytes) to trigger offload
watchdog.enabledfalseRun hung-session detector after each janitor pass
watchdog.staleMinutes5Session updatedAt age threshold to consider stuck
watchdog.alertSlacktrueSend Slack DM when a stuck session is detected
watchdog.slackTargetSlack user ID or channel to notify
watchdog.autoRestartfalseAuto-trigger safe-restart script on detection (use with caution)
watchdog.restartScriptSlack~/bin/safe-slack-restart.shSafe restart script path for Slack gateway
watchdog.restartScriptDiscord~/bin/safe-gateway-restart.shSafe restart script path for Discord gateway

Gateway Discovery

Setup scans for ~/.openclaw/openclaw.json and ~/.openclaw-*/openclaw.json. Each discovered gateway gets its own entry in config.json with:

  • Port (from gateway.port in config)
  • Auth token (from gateway.auth.token)
  • Sessions directory path

Works with single-gateway installs, multi-gateway (Discord + Slack), or any custom layout.

LLM Extraction Guardrails

  • Lockfile — only one extraction at a time across all gateways
  • Max per run — configurable cap per cron cycle (default: 1)
  • 20K char input cap — truncates middle of very long conversations
  • 60s hard timeout — SIGALRM kills hung API calls
  • Dedup — state file tracks processed trims, never re-runs
  • No context inflation — memories go to mem DB only, never to files that get auto-loaded into session context

Manual Run

bash skills/session-janitor/scripts/janitor.sh

Logs

tail -f /tmp/session-janitor.log

Watcher Service

The watcher fires trim within ~3 seconds of each turn completing (vs. waiting for the next cron cycle). setup.sh installs it automatically as a system service.

Linux (systemd):

systemctl --user status session-janitor-watcher
systemctl --user restart session-janitor-watcher
tail -f /tmp/session-janitor-watcher.log

macOS (launchd):

launchctl list | grep session-janitor
launchctl unload ~/Library/LaunchAgents/ai.openclaw.session-janitor-watcher.plist
launchctl load  ~/Library/LaunchAgents/ai.openclaw.session-janitor-watcher.plist
tail -f /tmp/session-janitor-watcher.log
  • Linux uses inotifywait (from inotify-tools) and watches via inotify kernel events.
  • macOS uses fswatch (via Homebrew) and watches via FSEvents.

Watchdog

The watchdog runs after every janitor pass and checks updatedAt on all active sessions. If any session is stale longer than watchdog.staleMinutes, it fires a Slack alert. Optional autoRestart will trigger the appropriate safe-restart script.

Alert cooldown is 10 minutes per session to prevent spam. State is tracked in the janitor state file.

# Run standalone
bash skills/session-janitor/scripts/watchdog.sh 5 ~/.openclaw/session-janitor-state.json

Sub-Agent Reporting Conventions

Workers spawned via Foreman should use prefixed sessions_send messages to prevent the parent agent from spawning new sub-agents in response to status updates:

# Intermediate status (fire-and-forget — parent relays but does NOT spawn)
sessions_send(sessionKey="{PARENT}", message="STATUS: 🤖 [label]: doing X (60%)", timeoutSeconds=0)

# Errors/failures (normal delivery — parent may intervene)
sessions_send(sessionKey="{PARENT}", message="ERROR: 🤖 [label]: what failed — details")

# Completion (no prefix, normal delivery)
sessions_send(sessionKey="{PARENT}", message="🤖 [label]: ✅ done — summary")
  • STATUS: → relay-only; timeoutSeconds=0 prevents reply-back loops
  • ERROR: → parent receives and considers recovery action
  • No prefix → completion; parent handles and relays to channel

Files

skills/session-janitor/
├── SKILL.md                              # This file
├── ARCHITECTURE.md                       # Deep dive: trimming mechanics, token math, examples
├── config.json                           # Generated by setup (gitignored)
├── config.example.json                   # Reference config
├── session-janitor-watcher.service       # Linux: systemd user service template
├── session-janitor-watcher.plist         # macOS: launchd LaunchAgent template
└── scripts/
    ├── setup.sh           # Discovery + config gen + cron + service install (Linux + macOS)
    ├── janitor.sh         # Main cron entry point
    ├── trim.py            # Transcript trimming (thinking, toolCall args, toolResult collapse)
    ├── sidecar.py         # Large toolResult offloader (→ .toolcache/ files)
    ├── watcher.sh         # Per-turn trigger (sidecar + trim; inotifywait/fswatch)
    ├── extract-llm.py     # LLM memory extraction
    ├── prune-sessions.py  # Stale session pruning
    ├── watchdog.sh        # Hung-session detector + Slack alert
    ├── test-sidecar.py    # Sidecar test suite (8 scenarios, 22 assertions)
    └── test-sidecar.sh    # Shell wrapper for sidecar tests

Source Transparency

This detail page is rendered from real SKILL.md content. Trust labels are metadata-based hints, not a safety guarantee.

Related Skills

Related by shared tags or category signals.

General

房车工业设计师

房车(RV/Motorhome/Touring Car)工业设计师技能。专注于旅居车、拖挂房车、商务房车、B型/C型房车的空间布局、外观造型、CMF设计、水电系统、人机舒适、智能控制、品牌体验设计。当用户需要房车设计方案、布局规划、设计规范、竞品分析、CMF材料工艺、人机工程、智能控制功能规划时使用此技能。

Registry SourceRecently Updated
General

china-shopping-oracle

国内全平台比价工具。Compare prices on Taobao/JD/Pinduoduo. Supports member pricing (88VIP/Plus). 电商比价、购物助手。

Registry SourceRecently Updated
General

feishu file transfer guide

Guide to upload and send local files to Feishu users via OpenClaw by obtaining tenant token, uploading files for file_key, then sending file message using Fe...

Registry SourceRecently Updated
General

Ainative React Sdk

Use @ainative/react-sdk to add AI chat and credits to React apps. Use when (1) Installing @ainative/react-sdk, (2) Using the useChat hook for chat completion...

Registry SourceRecently Updated