ai-config-sync-manager
v0.1.3
Published
OSS manager for syncing Claude and Codex developer configuration.
Maintainers
Readme
Highlights
- Continuous bidirectional sync —
claude → codexandcodex → claude, run as often as the two hosts drift; not a one-shot migration. - Strict YAML round-trip — Claude lenient YAML and Codex YAML 1.2 strict frontmatter both preserved across repeated syncs without loss or oscillation.
- Diff-first workflow —
statusto compare →sync --dry-runto preview →--applyto write. - Risk-tagged operations —
permissions,hooks, custom commands labeledsafe/partial/manual. - Backup-on-write — every overwrite snapshotted under
.backups/, FIFO retention (30). - Selector syntax —
--include skills:code-writer,instructions --exclude mcpstyle filtering. - Native semantic mapping — Claude
Write→ Codexsandbox_mode = "workspace-write", etc. - Prose-level token rewriting — Claude-only tokens (
Read,Bash,TaskCreate, headlessclaude -p) and Codex-only tokens (spawn_agent,codex exec) auto-translate across hosts and round-trip back. - Zero runtime dependencies — single ESM file, Node built-ins only.
- Thin host plugins —
/config-manager:*for Claude,config-manager-*for Codex.
Why this exists
Claude Code and Codex use the same concepts (instructions / skills / mcp / permissions / hooks) but in different files, formats, and names:
| Concept | Claude | Codex |
|---|---|---|
| Instructions | ~/.claude/CLAUDE.md | ~/.codex/AGENTS.md |
| Skills | ~/.claude/skills/ | ~/.codex/skills/ |
| Settings | ~/.claude/settings.json | ~/.codex/config.toml |
| MCP | ~/.claude/.mcp.json | [mcp_servers.*] in config.toml |
Hand-rolling the sync invites drift, semantic loss, and accidental secret leaks. This CLI keeps the two hosts aligned while preserving host-native meaning.
This tool is built for two hosts in continuous use — where drift accumulates daily and round-trip integrity matters across repeated syncs — not for a one-shot, one-way migration.
One-shot migrators typically copy Claude-only vocabulary (tool names like Read / Bash, prose like Use the Bash tool, in-line Agent({...}) calls) into the generated file as prompt guidance and flag it for manual review. This CLI instead auto-rewrites those tokens to their host equivalents (Read → workspace-write, TaskCreate → spawn_agent, claude -p → codex exec, …) and round-trips them back when syncing the other direction — so the same content stays correct for both hosts with no manual fix-ups.
Quick Start
npm install -g ai-config-sync-manager
ai-config-sync connect # register the plugin for any detected host (Claude / Codex)
ai-config-sync status # show drift across global + project scopes
ai-config-sync sync # preview changes (--dry-run by default)
ai-config-sync sync --apply # apply with automatic backupsconnect only registers plugins for hosts it actually finds (~/.claude for Claude, ~/.codex or ~/.agents for Codex). Hosts that are missing are reported as skipped and no directories are created — install the host first, then rerun connect.
Requirements
- Node.js ≥ 20
- Claude Code and/or Codex CLI installed (host plugins are auto-registered by
connectwhen the matching host directory exists)
Table of Contents
| Category | Sections | | --- | --- | | Commands | Bundled CLI · Host plugin commands · Flags | | Workflow | Selector syntax · Ignore rules · Sync direction · Scopes | | Safety | Safety defaults · Risk levels · Retention | | Mapping | Native mapping · Areas · Paraphrase · Hidden markers · Unsupported | | Reference | Install resolution · Local dev · Gotchas · API surface · What's next |
Commands
Bundled CLI
After npm install -g, the same binary is on PATH as ai-config-sync — equivalent to ./bin/ai-config-sync.mjs from a source clone.
ai-config-sync connect
ai-config-sync status
ai-config-sync status --json
ai-config-sync status --scope global
ai-config-sync status --scope project
ai-config-sync status --include skills:code-writer,instructions --exclude mcp
ai-config-sync sync --dry-run
ai-config-sync sync --scope project --dry-run
ai-config-sync sync --scope global --apply
ai-config-sync sync --include instructions,skills:code-writer --exclude mcp --dry-run
ai-config-sync sync --from claude --to codex
ai-config-sync sync --from codex --to claude
ai-config-sync reference
ai-config-sync paraphrase| Command | Purpose |
| --- | --- |
| connect | Detect installed hosts and register the matching plugin |
| status | Compare global + project config across both hosts |
| status --json | Machine-readable diff |
| sync --dry-run | Preview the merge plan without writing |
| sync --apply | Apply the plan, snapshot to .backups/ |
| reference | Emit / persist a self-generated markdown reference |
| paraphrase | Line-level override archive for instruction wording |
Host plugin commands
| Host | Connect | Status | Sync | Paraphrase |
| --- | --- | --- | --- | --- |
| Claude | /config-manager:connect | /config-manager:status | /config-manager:sync | /config-manager:paraphrase |
| Codex | config-manager-connect | config-manager-status | config-manager-sync | config-manager-paraphrase |
Flags
Per-subcommand flag reference, mirroring <command> --help output. Shared flags (--include / --exclude / --scope / --map) get a one-line summary; full syntax lives in the linked section.
connect
| Flag | Description |
| --- | --- |
| -h, --help | Show connect help |
ai-config-sync connectstatus
| Flag | Description |
| --- | --- |
| --json | Print the full status report as JSON |
| --compact | One compact line per diff entry |
| --tree | Scope/area/item tree output |
| --scope global\|project\|all | Limit scope (default: all = global + project) |
| --include area[:item][,...] | Include selector — see Selector syntax |
| --exclude area[:item][,...] | Exclude selector — see Selector syntax |
| -h, --help | Show status help |
ai-config-sync status --scope project --tree --include skills:code-writersync
| Flag | Description |
| --- | --- |
| --dry-run | Preview without writing (default; mutually exclusive with --apply) |
| --apply | Apply with backups |
| --plan-json | Print the sync plan as JSON |
| --from claude\|codex | Source host (overrides AI_CONFIG_SYNC_HOST) |
| --to claude\|codex | Target host (overrides AI_CONFIG_SYNC_HOST) |
| --scope global\|project\|all | Limit scope (default: all = global + project) |
| --include area[:item][,...] | Include selector — see Selector syntax |
| --exclude area[:item][,...] | Exclude selector — see Selector syntax |
| -h, --help | Show sync help |
When --from / --to are omitted, direction follows Sync direction.
ai-config-sync sync --scope project --include mcp:notion --applyreference
| Flag | Description |
| --- | --- |
| --output <path> | Write the reference markdown to <path> (parent directories created) |
| -h, --help | Show reference help |
ai-config-sync reference --output ~/.ai-config-sync-manager/reference.mdparaphrase
| Flag | Description |
| --- | --- |
| --apply | Rewrite files + register overrides + persist new map entries (default: dry-run) |
| --register | Skip rewriting; only register overrides where the effective map already equates both sides — see Paraphrase |
| --map token=paraphrase[,...] | Inline token-to-paraphrase pairs (free-form prose accepted) — see --map syntax |
| --non-interactive | Skip TTY prompts for tokens missing from paraphrase-map.json |
| --json | Print the result as JSON |
| --scope global\|project\|all | Limit scope (default: all = global + project) |
| --include area[:item][,...] | Include selector — see Selector syntax |
| --exclude area[:item][,...] | Exclude selector — see Selector syntax |
| -h, --help | Show paraphrase help |
ai-config-sync paraphrase --map "Read=read the file,Write=write to the file" --applySelector syntax
--include narrows the plan first, then --exclude removes matches. Both accept area or area:item syntax; itemized areas (skills, permissions, hooks, agents, mcp, commands) accept glob items.
ai-config-sync sync --include skills:code-writer,instructions --exclude mcp --dry-run
ai-config-sync sync --include "permissions:Write*" --exclude "permissions:Bash(rm:*)" --dry-runAreas
| Area | Itemized? | Apply granularity |
| --- | --- | --- |
| instructions | — | file merge |
| skills | yes | per skill |
| agents | yes | per agent |
| mcp | yes | per server |
| permissions | yes | item-by-item patch |
| hooks | yes | item-by-item patch |
| commands | yes | per command |
| plugins | yes | status only (read-only diff; sync skips this area) |
Ignore rules
Persistent ignore lives at one of:
<project>/.ai-config-sync-manager/status-ignore.json(project scope, checked first)~/.ai-config-sync-manager/rules/status-ignore.json(global)
Each exclude entry is a string selector (area:item or path glob) or an object whose fields combine with AND. term is a line-level mask — lines containing the substring are removed from both sides before the diff, so the conflict can disappear without hiding unrelated changes.
{
"version": 1,
"exclude": [
"skills:legacy-skill",
"permissions:Bash",
"~/.codex/agents/archive-*.toml",
{ "scope": "global", "area": "agents", "item": "refactor-*" },
{ "area": "skills", "host": "claude", "path": "~/.claude/skills/coderabbit-review" },
{ "area": "skills", "term": ".claude/docs/repo-analysis/" },
{ "area": "agents", "host": "claude", "path": "~/.claude/agents/*.md", "term": "TODO: do not sync" }
]
}The active path and rule count are echoed in status output as Status ignore: <path> rules: [...] (N hidden).
Sync direction
| Trigger | Default direction |
| --- | --- |
| AI_CONFIG_SYNC_HOST=codex (Codex plugin invocation) | codex → claude |
| Otherwise (Claude plugin / direct CLI) | claude → codex |
| --from <host> --to <host> | Explicit override |
Scopes
| Scope | Path coverage |
| --- | --- |
| global | ~/.claude/**, ~/.codex/** |
| project | <cwd>/.claude/**, <cwd>/.codex/**, <cwd>/AGENTS.md, <cwd>/CLAUDE.md |
| default / all | global + project |
Safety defaults
- Dry-run first —
syncdefaults to dry-run;--applyis required for any write. - Backups on every write — atomic snapshot to
.backups/<area>/<host>/<timestamp>/...before overwrite. - Risk labels — high-impact entries (
permissions,hooks, custom commands) marked with their risk level in the diff. - Strict-vocab guard — host-only tokens (e.g. Codex
update_plan) flagged on cross-host copy. - Secret pass-through — MCP env values are copied by default; set
AI_CONFIG_SYNC_STRIP_SECRETS=1to redact. - Schema version — baseline state requires
schemaVersion: 1; unknown versions abort.
Risk levels
| Level | Meaning | Behavior |
| --- | --- | --- |
| safe | Lossless, deterministic mapping | Auto-applied |
| partial | Maps to a near-equivalent on the other host | Auto-applied with annotation |
| manual | No safe automatic equivalent | Listed in the plan but always review before --apply |
Retention
| Directory | Keep | Strategy |
| --- | --- | --- |
| .backups/<area>/<host>/ | 30 | FIFO (oldest pruned on next write) |
| ~/.config/ai-config-sync/status-details/ | 100 | FIFO |
Native mapping (Claude ↔ Codex)
| Claude | Codex |
| --- | --- |
| permissions.allow: ["Write"] | sandbox_mode = "workspace-write" |
| command-like permissions.allow (e.g. Bash(npm:*)) | approval_policy = "on-request" |
| .mcp.json server entries | [mcp_servers.<name>] TOML tables |
| hooks.PreToolUse / PostToolUse | mapped where a Codex equivalent exists, else manual |
| ~/.claude/skills/<name>/SKILL.md | ~/.codex/skills/<name>/SKILL.md |
Full mapping reference: rules/.
Paraphrase
Some tokens are mutually exclusive between hosts — Read, Write, Edit, Glob, mcp__* only exist on Claude; update_plan, spawn_agent, apply_patch only exist on Codex (full list: rules/host-strict-vocab.json). When such a token leaks into the wrong host's file, the terminology map cannot translate it, so status keeps reporting the line as a manual-review mismatch forever.
This is what keeps strict YAML round-trip stable across repeated syncs: without it, host-specific vocabulary would oscillate or accumulate as drift on every cycle.
paraphrase resolves these by rewriting both sides to a shared word and registering a per-line override so future status runs treat the pair as in sync.
ai-config-sync paraphrase # dry-run preview
ai-config-sync paraphrase --apply # rewrite + register
ai-config-sync paraphrase --map "Read=Inspect,Write=Author" --apply # inline mapping
ai-config-sync paraphrase --register --include skills:foo --apply # register only (no rewrite)| Flag | Purpose |
| --- | --- |
| --apply | Rewrite files, append to paraphrase-overrides.json, persist new entries to paraphrase-map.json (default: dry-run) |
| --map token=paraphrase[,...] | Inline token-to-paraphrase pairs; layered on top of the file map |
| --register | Skip rewriting; only register an override when the effective map already makes both sides byte-equal |
| --non-interactive | Skip TTY prompts for tokens missing from the map |
| --scope global\|project\|all | Limit paraphrase scope (default: both) |
| --include / --exclude | Same selector syntax as status / sync |
| --json | Machine-readable result |
--map syntax
Token=Paraphrasefor unambiguous tokens listed inhost-strict-vocab.json.- Prefix with
claude_only:orcodex_only:to disambiguate (e.g.claude_only:Read=Inspect). - Comma-separated to chain entries:
--map "Read=Inspect,codex_only:update_plan=Plan refresh". - Tokens not present in
host-strict-vocab.jsonare rejected unless prefixed. - Paraphrase can be free-form prose, not just a single word. Pick wording that reads naturally on the opposite host so the rewritten line still makes sense in context — e.g. when masking a Claude-only token, choose phrasing a Codex prompt would actually use (
Read=read the file,Write=write to the file,Glob=glob for files). Quote the value when it contains spaces.
Map files (layered: project → home → repo)
| File | Role |
| --- | --- |
| rules/paraphrase-map.json | Token → paraphrase entries grouped under claude_only / codex_only |
| rules/paraphrase-overrides.json | Per-line override archive (host paths, line numbers, anchor texts) |
Both files follow the same precedence as terminology rules: <project>/rules/<file>.json → ~/.ai-config-sync-manager/rules/<file>.json → <repo>/rules/<file>.json.
Examples
# Preview rewrites for every drifted host-only token
ai-config-sync paraphrase
# Apply with an inline map for two common offenders
ai-config-sync paraphrase --map "Read=Inspect,Write=Author" --apply
# Scope to a single agent file in global config
ai-config-sync paraphrase --scope global --include agents:code-writer --apply
# Side already pre-paraphrased outside the CLI — just record the override
ai-config-sync paraphrase --register --include skills:code-writer \
--map "Read=Inspect,Write=Emit" --apply
# Natural-language intent — slash-command agent translates this into the
# matching --map flags on the fly (no need to spell out token=paraphrase pairs)
/config-manager:paraphrase rewrite to Codex-compatible wordingStale overrides
Overrides are auto-invalidated when the pinned anchor text on either host drifts, so manual edits cleanly retire the recorded pairing without leaving stale entries. The active / stale counts are echoed in status output.
Hidden markers
HTML comment markers the call compiler emits inside transformed text. They are reverse-direction-stable and ignored by humans during normal use.
| Marker | Meaning |
| --- | --- |
| <!-- ai-config-sync:agent-call ... --> | Supported call transformed (see Agent call compiler). |
| <!-- ai-config-sync:stripped ... --> | Unsupported call removed; original archived under ~/.ai-config-sync-manager/backups/<timestamp>/unsupported-calls.json. |
| <!-- ai-config-sync:manual-review reason="..." --> | Call left intact because the tolerant scanner could not parse it; needs manual translation. |
Unsupported
Items the engine deliberately does not sync, with the reason and the surface where they remain visible:
| Surface | Behavior | Why |
| --- | --- | --- |
| TaskCreate / TaskUpdate / TeamCreate SDK calls | Stripped with ai-config-sync:stripped marker; original payload archived under the backup root (unsupported-calls.json) | Codex has no native todo/task tracker tool nor an atomic agent-team primitive |
| Symlinked skills (~/.claude/skills/<name> is a symlink) | Reported by status as unsupported (action: manual review); excluded from sync --apply | Whether to copy the link or materialize target content is an unresolved policy. Resolve manually by either rewriting as a real directory on the source host, or applying via --include skills:<name> after the policy lands |
| memory / implicit context / agent runtime state | Out of scope for now (no read, no write) | Storage layout, redaction rules, and conflict policy not yet settled. Tracked in What's next — first phase will be read-only status |
| Host plugin installations (~/.claude/plugins/installed_plugins.json, ~/.agents/plugins/marketplace.json) | Reported by status --scope global as unsupported (action: manual review); excluded from sync --apply. Self-managed config-manager@ai-config-sync-manager / ai-config-sync-manager are filtered out so they never surface as drift | Plugin install/remove crosses package-manager-like boundaries (marketplace metadata + tree copy + per-host install commands). Status surfaces install hints — /plugin install <name>@<source> for Claude, edit ~/.agents/plugins/marketplace.json for Codex. |
Install resolution
The plugin launcher resolves the CLI in this order:
AI_CONFIG_SYNC_ROOTenv (dev override)- PATH
ai-config-sync(npm install -gornpm link) npm exec --yes --package=ai-config-sync-manager@<pin>fallback
After npm install -g, every host calls the same npm package, so two hosts cannot drift to different versions.
Local dev from this repo
npm install runs the prepare script, which builds dist/ (without touching active plugin caches) so the launcher and host plugin trees are ready to inspect immediately after clone.
npm run dev and npm run prod toggle between linking the local clone for active development and switching back to a published npm release for verification.
npm install # also runs `prepare` -> build:dist --skip-sync
npm run dev # npm link + build:dist (active plugin cache sync) — local dev mode
npm run prod # unlink + npm i -g ai-config-sync-manager (latest stable) + plugin reinstall hint
npm run prod beta # opt into a tagged dist-tag (e.g. `beta`, `next`) during prerelease cycles
npm test # node:test integration suite
npm run check # opt-in JSDoc / @ts-check
npm run lint
npm run format:checkInside a clone, invoke the CLI as ./bin/ai-config-sync.mjs <command> (the published ai-config-sync shim only exists after npm install -g or npm link).
Gotchas
- No programmatic API.
bin/ai-config-sync.mjsexecutes on import. Do notimportit from another module — see API surface. - Symlink skills,
TaskCreate/TaskUpdate/TeamCreate, and memory/runtime state are not synced — see Unsupported for the per-surface behavior. - Codex host inversion — when invoked through the Codex plugin,
AI_CONFIG_SYNC_HOST=codexflips the default direction tocodex → claude. Use--from/--tofor an explicit override. - MCP env values are copied verbatim by default — opt in to redaction with
AI_CONFIG_SYNC_STRIP_SECRETS=1. --applyis final, but reversible: every write creates a.backups/snapshot.--dry-runis the default for a reason.
API surface
This is a CLI tool, not a library. There is no programmatic API — import-ing this package from another Node module is not supported and the bundled bin/ai-config-sync.mjs is not designed to be loaded as a library (it executes the command on import). All functionality is exposed through the ai-config-sync command and the host plugins. If you need programmatic access to a specific function (mapping rules, plan generation, status diff), open an issue describing the use case so the surface can be designed deliberately.
What's next
| Item | Status | Notes |
| --- | --- | --- |
| Additional host integrations (Gemini CLI,Cursor, …) | Planned | The launcher pattern is reusable. Each new host gets its own integrations/<host>-plugin/ after a survey of its plugin/extension spec and config storage layout. |
| Memory / context sync | Deferred (RFC-first) | memory, implicit context, and agent runtime state currently sit outside the sync surface. The first phase will be read-only discovery / status; --apply is reserved for opt-in selectors (e.g. --include memories:<name>) once storage location, schema, redaction, and conflict policy are settled. |
| Skill symlink full support | Deferred | Symlinked skills appear in status only. sync will engage once the link-preserve vs target-materialize policy is finalized. |
| Plugin sync (plugins area) | Status-only today | status --scope global lists user-installed plugins from both hosts as unsupported with install hints. Bidirectional sync requires designing the cross-host mapping for marketplace metadata, install commands, and plugin tree copy semantics. |
| Extra mappings (rules/*.json import, TOML parser swap) | Tracked | Mechanical refactors with no user-visible API change. |
If any of these unblocks your workflow, an issue with the concrete use case helps prioritize the order.
License
MIT
