Doctor
fased doctor is the repair + migration tool for Fased. It fixes stale
config/state, checks health, and provides actionable repair steps.
Quick start
Headless / automation
What it does (summary)
- Optional pre-flight update for git installs (interactive only).
- UI protocol freshness check (rebuilds Control UI when the protocol schema is newer).
- Health check + restart prompt.
- Skills status summary (eligible/missing/blocked).
- Config normalization for legacy values.
- OpenCode Zen provider override warnings (
models.providers.opencode). - Legacy on-disk state migration (sessions/agent dir/WhatsApp auth).
- State integrity and permissions checks (sessions, transcripts, state dir).
- Config file permission checks (chmod 600) when running locally.
- Model auth health: checks OAuth expiry, can refresh expiring tokens, and reports auth-profile cooldown/disabled states.
- Extra workspace dir detection (
~/fased). - Sandbox image repair when sandboxing is enabled.
- Legacy service migration and extra gateway detection.
- Gateway runtime checks (service installed but not running; cached launchd label).
- Channel status warnings (probed from the running gateway).
- Supervisor config audit (launchd/systemd/schtasks) with optional repair.
- Gateway runtime best-practice checks (Node vs Bun, version-manager paths).
- Gateway port collision diagnostics (default
18789). - Security warnings for open DM policies.
- Gateway auth warnings when shared Gateway auth is missing or weak (local mode; offers token generation).
- systemd linger check on Linux.
- Source install checks (pnpm workspace mismatch, missing UI assets, missing tsx binary).
- Writes updated config + wizard metadata.
Detailed behavior and rationale
0) Optional update (git installs)
If this is a git checkout and doctor is running interactively, it offers to update (fetch/rebase/build) before running doctor.1) Config normalization
If the config contains legacy value shapes (for examplemessages.ackReaction
without a channel-specific override), doctor normalizes them into the current
schema.
2) Legacy config key migrations
When the config contains deprecated keys, other commands refuse to run and ask you to runfased doctor.
Doctor will:
- Explain which legacy keys were found.
- Show the migration it applied.
- Rewrite
~/.fased/fased.jsonwith the updated schema.
routing.allowFrom→channels.whatsapp.allowFromrouting.groupChat.requireMention→channels.whatsapp/telegram/imessage.groups."*".requireMentionrouting.groupChat.historyLimit→messages.groupChat.historyLimitrouting.groupChat.mentionPatterns→messages.groupChat.mentionPatternsrouting.queue→messages.queuerouting.bindings→ top-levelbindingsrouting.agents/routing.defaultAgentId→agents.list+agents.list[].defaultrouting.agentToAgent→tools.agentToAgentrouting.transcribeAudio→tools.media.audio.modelsbindings[].match.accountID→bindings[].match.accountId- For channels with named
accountsbut missingaccounts.default, move account-scoped top-level single-account channel values intochannels.<channel>.accounts.defaultwhen present identity→agents.list[].identityagent.*→agents.defaults+tools.*(tools/elevated/exec/sandbox/subagents)agent.model/allowedModels/modelAliases/modelFallbacks/imageModelFallbacks→agents.defaults.models+agents.defaults.model.primary/fallbacks+agents.defaults.imageModel.primary/fallbacksbrowser.ssrfPolicy.allowPrivateNetwork→browser.ssrfPolicy.dangerouslyAllowPrivateNetwork
2b) OpenCode Zen provider overrides
If you’ve addedmodels.providers.opencode (or opencode-zen) manually, it
overrides the built-in OpenCode Zen catalog. That can force every model onto a
single API or zero out costs. Doctor warns so you can remove the override and
restore per-model API routing + costs.
3) Legacy state migrations (disk layout)
Doctor can migrate older on-disk layouts into the current structure:- Sessions store + transcripts:
- from
~/.fased/sessions/to~/.fased/agents/<agentId>/sessions/
- from
- Agent dir:
- from
~/.fased/agent/to~/.fased/agents/<agentId>/agent/
- from
- WhatsApp auth state (Baileys):
- from legacy
~/.fased/credentials/*.json(exceptoauth.json) - to
~/.fased/credentials/whatsapp/<accountId>/...(default account id:default)
- from legacy
fased doctor.
4) State integrity checks (session persistence, routing, and safety)
The state directory holds sessions, credentials, logs, and config. If it vanishes, you lose those local records unless you have backups elsewhere. Doctor checks:- State dir missing: warns about catastrophic state loss, prompts to recreate the directory, and reminds you that it cannot recover missing data.
- State dir permissions: verifies writability; offers to repair permissions
(and emits a
chownhint when owner/group mismatch is detected). - Session dirs missing:
sessions/and the session store directory are required to persist history and avoidENOENTcrashes. - Transcript mismatch: warns when recent session entries have missing transcript files.
- Main session “1-line JSONL”: flags when the main transcript has only one line (history is not accumulating).
- Multiple state dirs: warns when multiple
~/.fasedfolders exist across home directories or whenFASED_STATE_DIRpoints elsewhere (history can split between installs). - Remote mode reminder: if
gateway.mode=remote, doctor reminds you to run it on the remote host (the state lives there). - Config file permissions: warns if
~/.fased/fased.jsonis group/world readable and offers to tighten to600.
5) Model auth health (OAuth expiry)
Doctor inspects OAuth profiles in the auth store, warns when tokens are expiring/expired, and can refresh them when safe. If the Anthropic Claude Code profile is stale, it suggests runningclaude setup-token (or pasting a setup-token).
Refresh prompts only appear when running interactively (TTY); --non-interactive
skips refresh attempts.
Doctor also reports auth profiles that are temporarily unusable due to:
- short cooldowns (rate limits/timeouts/auth failures)
- longer disables (billing/credit failures)
6) Hooks model validation
Ifhooks.gmail.model is set, doctor validates the model reference against the
catalog and allowlist and warns when it won’t resolve or is disallowed.
7) Sandbox image repair
When sandboxing is enabled, doctor checks Docker images and offers to build or switch to legacy names if the current image is missing.8) Gateway service migrations and cleanup hints
Doctor detects legacy gateway services (launchd/systemd/schtasks) and offers to remove them and install the Fased service using the current gateway port. It can also scan for extra gateway-like services and print cleanup hints. Profile-named Fased gateway services are considered first-class and are not flagged as “extra.”9) Security warnings
Doctor emits warnings when a provider is open to DMs without an allowlist, or when a policy is configured in a dangerous way.10) systemd linger (Linux)
If running as a systemd user service, doctor ensures lingering is enabled so the gateway stays alive after logout.11) Skills status
Doctor prints a quick summary of eligible/missing/blocked skills for the current workspace.12) Gateway auth checks (local token)
Doctor warns when shared Gateway auth is missing or weak on a local gateway and offers to generate a token. Usefased doctor --generate-gateway-token to force
token creation in automation.
13) Gateway health check + restart
Doctor runs a health check and offers to restart the gateway when it looks unhealthy.14) Channel status warnings
If the gateway is healthy, doctor runs a channel status probe and reports warnings with suggested fixes.15) Supervisor config audit + repair
Doctor checks the installed supervisor config (launchd/systemd/schtasks) for missing or outdated defaults (e.g., systemd network-online dependencies and restart delay). When it finds a mismatch, it recommends an update and can rewrite the service file/task to the current defaults. Notes:fased doctorprompts before rewriting supervisor config.fased doctor --yesaccepts the default repair prompts.fased doctor --repairapplies recommended fixes without prompts.fased doctor --repair --forceoverwrites custom supervisor configs.- You can always force a full rewrite via
fased gateway install --force.
16) Gateway runtime + port diagnostics
Doctor inspects the service runtime (PID, last exit status) and warns when the service is installed but not actually running. It also checks for port collisions on the gateway port (default18789) and reports likely causes (gateway already
running, SSH tunnel).
17) Gateway runtime best practices
Doctor warns when the gateway service runs on Bun or a version-managed Node path (nvm, fnm, volta, asdf, etc.). WhatsApp + Telegram channels require Node,
and version-manager paths can break after upgrades because the service does not
load your shell init. Doctor offers to migrate to a system Node install when
available (Homebrew/apt/choco).