Run a full general purpose agent from one static Go binary.
ReAct, filesystem and shell tools, MCP, Skills, optional OpenAI-compatible API with an embedded UI, scheduler, and long-term memory.
More screenshots
| Chat | History |
|---|---|
![]() |
![]() |
| Scheduler (list + job) | Settings |
![]() |
![]() |
Screenshots at 1920×1080 from the embedded UI (coddy http + Vite dev). Spec and dev workflow: docs/ui.md, layout tokens: DESIGN.md.
Coddy is a distroless-friendly harness: drop it into minimal images (scratch, distroless, read-only workspaces) without a full OS shell. The harness layer (ACP RPC, sessions, prompts, providers) stays the same if you tighten the toolset or drive it from automation instead of an IDE. The design also targets container fleets - many Coddy instances in Docker (orchestrator-defined limits, read-only rootfs, mounted workspace) with full control of each container, similar in spirit to agent OS / swarm-style agents, not a single shared chat pool.
- Features
- Quick start
- How to update
- Operating modes
- Editor and IDE integration
- Rules
- Skills
- MCP server integration
- Messenger gateway
- Configuration (reference)
- Architecture
- Documentation
- Examples (ACP over stdio)
- Persistent sessions
- Development
- License
- Harness-first - ACP server, session lifecycle, prompts, LLM backends, MCP merge, distroless-ready binary
- ReAct loop - LLM alternates between reasoning, acting (tool calls), and observing results (coding-agent persona out of the box)
- Two operating modes -
agent(full tool access) andplan(planning + text files only) - Rules - auto-discovers
.cursor/rules/,.coddy/rules/,.claude/rules/, and.codex/rules/under the session cwd - see Rules - Skills - slash commands and
SKILL.mdpacks fromskills.dirs(defaults include~/.cursor/skills) - see Skills - MCP server integration - connect any MCP server for additional tools
- Multi-provider LLM - OpenAI, Anthropic, Ollama, any OpenAI-compatible API
- ACP protocol - Coddy is an ACP server (
coddy acp); pair it with editors or scripts that implement an ACP client (see Editor and IDE integration) - Messenger gateway - optional Telegram bot adapter (
-tags gateway.telegram); per-user sessions, group isolation modes, admin ACL; extensible to Discord, Slack, etc. — see Messenger Gateway
Coddy is an ACP server (coddy acp). Cursor, Zed, scripts, and the bundled coddy http UI are clients that share the same CODDY_HOME sessions when configured with the same home directory.
Full setup (Cursor external agent, Zed agent_servers, shared sessions, rules vs skills): docs/editor-integration.md. Protocol details: docs/acp-protocol.md, harness examples: examples/acp/.
Linux / macOS - release binary plus ~/.coddy bootstrap:
curl -fsSL https://coddy.dev/install.sh | bashWindows (PowerShell)
irm https://coddy.dev/install.ps1 | iexCreates ~/.coddy/config.yaml from the release config.example.yaml when missing. Puts coddy on PATH (Unix: ~/.local/bin; Windows: %LOCALAPPDATA%\Programs\coddy). Full installer options: docs/install.md.
Then set a provider key in ~/.coddy/config.yaml (or OPENAI_API_KEY in the environment) and run coddy http for the UI, or coddy acp for an editor client.
Docker - same full binary in ghcr.io/coddy-project/coddy-agent: docker compose up -d (see Docker).
Upgrade later with coddy update -y (How to update).
Other installation methods (build from source, Go install, manual)
Prerequisites for building
- Go - same minor version as
go.mod(currently 1.25). - Git - used by the Makefile for the embedded version string.
- Node.js / npm - only if you build with
httpandui(the Makefile runsui-buildfor embedded assets).
Install with Go (lean module default, no http / UI tags)
go install github.com/EvilFreelancer/coddy-agent/cmd/coddy@latestFor coddy http, the bundled SPA, scheduler, and memory, use a release binary (install script above) or build from source below.
Recommended full binary from source
git clone https://github.com/EvilFreelancer/coddy-agent
cd coddy-agent
make build TAGS="http ui scheduler memory"
make install # copies build/coddy to ~/.local/bin or /usr/local/binOr download archives from GitHub Releases.
Manual go build
When TAGS includes http and ui, run make ui-build first.
make ui-build
VERSION="$(make -s print-version)"
go build -tags=http,ui,scheduler,memory \
-ldflags "-X github.com/EvilFreelancer/coddy-agent/internal/version.Version=${VERSION}" \
-o build/coddy \
./cmd/coddy/Lean ACP-only binary: make build (no http / UI / scheduler / memory tags).
Build reference: docs/build.md.
coddy -v prints the embedded version. coddy acp --help lists ACP flags (--home, --cwd, --config, etc.).
Use Makefile variable TAGS with spaces (make build TAGS="http ui scheduler memory"). go build uses commas (-tags=http,ui,scheduler,memory).
| Tag | Enables | Docs |
|---|---|---|
memory |
Long-term memory copilot (memory.enabled in YAML); with http, session memory REST under /coddy/sessions/{id}/memory/* |
external/memory/README.md |
http |
coddy http, REST gateway, /docs, /openapi.yaml |
docs/http-api.md |
ui |
Embedded SPA on / (needs http) |
docs/ui.md, DESIGN.md |
scheduler |
Scheduler daemon and coddy_scheduler_* tools; with http, /coddy/scheduler REST |
docs/scheduler.md, external/scheduler/README.md |
gateway.telegram |
Telegram bot adapter — coddy gateway subcommand, per-user sessions, access control |
docs/gateway.md |
gateway |
All messenger adapters (superset of gateway.telegram; add Discord/Slack without changing the core) |
docs/gateway.md |
Extended narrative and Docker alignment - docs/build.md.
Release images are published on GitHub Container Registry as ghcr.io/coddy-project/coddy-agent (tags such as latest and X.Y.Z, linux/amd64 and linux/arm64). Each SemVer git tag also gets GitHub Release archives (Linux, Windows, macOS Intel and Apple Silicon) - see docs/build.md. The default image includes http, ui, scheduler, and memory - the same feature set as make build TAGS="http ui scheduler memory".
1. Config and workspace (from the repo root, or any directory where you keep config.yaml):
cp config.example.yaml config.yaml
mkdir -p workspace coddy_home
# Edit config.yaml: at least one provider api_key (or rely on OPENAI_API_KEY etc. in compose)2. Start with Compose (pull published image, no local build):
docker compose pull
docker compose up -dTo build the image locally instead, use docker-compose.dev.yml: docker compose -f docker-compose.dev.yml up -d --build.
3. Open the bundled UI in a browser on the host:
http://127.0.0.1:12345/
The SPA is served on GET / by coddy http. Pick a model in the composer (YAML backends from GET /v1/models), choose agent or plan mode, then send a message - the UI creates a session and streams the reply via POST /v1/responses. Agent files and shell tools use the mounted workspace (./workspace → /workspace in the container). Live YAML editing: http://127.0.0.1:12345/#/settings.
Sanity check without a browser: curl -sS http://127.0.0.1:12345/v1/models | head.
There is no login on the HTTP surface - expose port 12345 only on trusted networks. Full compose options, volumes, and CI image tags: docs/docker.md. Smoke script: examples/httpserver/docker.sh.
CODDY_HOME(orcoddy acp --home) is the agent state directory. Default~/.coddy. The process createssessions/andskills/under it. Config defaults to$CODDY_HOME/config.yaml.CODDY_CWD(orcoddy acp --cwd) is the default session working directory whensession/newsends an emptycwd. Default is the process current directory at startup. Editors that pass a path insession/newuse that path instead.
CODDY_HOME defaults to ~/.coddy. Unless you set CODDY_CONFIG or pass --config, the primary config file is config.yaml at $CODDY_HOME/config.yaml.
Copy the example and edit it:
mkdir -p ~/.coddy && cp config.example.yaml ~/.coddy/config.yamlIf $CODDY_HOME/config.yaml is absent, the loader may use config.yaml in the process working directory (useful when running from a repository clone). See docs/config.md.
Providers and models
providers- named backends (type:openaifor OpenAI and OpenAI-compatible HTTP APIs,anthropicfor Anthropic). Eachnamemust be ASCII letters, digits, hyphen, or underscore, starting with a letter (it becomes the prefix in model ids). Each row hasapi_key(literal,${ENV}expanded when the file loads, or empty to readNAME_API_KEYfrom the environment at LLM call time, withNAMEderived fromproviders[].namein uppercase and hyphens mapped to underscores), and optionallyapi_basewhen the API is not the vendor default.models- selectable models. Eachmodelstring is<provider_name>/<api_model_id>whereprovider_namematchesproviders[].name. Tunables includemax_tokens,temperature, and optionalmax_context_tokens.agent-modelpicks the default ReAct model (must match onemodels[].modelentry).max_turnsandmax_tokens_per_turnbound one user turn.
Example (openai provider and gpt-5.4-mini; store secrets in the environment, not in git):
providers:
- name: openai
type: openai
api_key: "${OPENAI_API_KEY}"
models:
- model: "openai/gpt-5.4-mini"
max_tokens: 400000
temperature: 0.2
agent:
model: "openai/gpt-5.4-mini"
max_turns: 35
max_tokens_per_turn: 128000Then export the key the YAML references:
export OPENAI_API_KEY="sk-..."Other setups (Anthropic, Ollama, a non-default api_base, and env-based defaults) are covered in config.example.yaml and docs/config.md.
Official CLI binaries are published on GitHub Releases (assets such as coddy_0.9.3_linux_amd64.tar.gz). Each release matches the full feature set from make build TAGS="http ui scheduler memory".
coddy update downloads the archive for your OS/architecture and replaces the binary you invoked (symlinks resolved). That is the usual path after make install (~/.local/bin/coddy) or when you run ./build/coddy update to refresh a local build artifact.
1. See what you run today
which coddy
coddy -v2. Check for a newer release
coddy update --checkExit code 0 means you are already on the latest published X.Y.Z (or newer). Exit code 1 means a newer release is available.
3. Install
coddy update # asks [y/N]
coddy update -y # no prompt4. Confirm
coddy -v
coddy http --help # only when the binary includes -tags=http (release builds do)Common flags
| Flag | Purpose |
|---|---|
--check |
Only report whether an update exists (no download). |
-y / --yes |
Install without confirmation. |
--version X.Y.Z |
Install a specific release, not only "latest". |
--repo owner/name |
Alternate GitHub repo (default coddy-project/coddy-agent). |
Notes
- Update the same binary you intend to use. If
which coddypoints at~/.local/bin/coddy, runcoddy updatefrom that install, not a different copy onPATH. $CODDY_HOME(config, sessions, skills) is untouched; only the executable changes.- To build from source or change tags, use
make buildinstead. For containers, usedocker compose pull. See docs/update.md for platform tables, limitations, and other upgrade paths.
Full task execution mode. The agent has access to all tools:
- Read and write files
- Execute shell commands (with permission prompt)
- Search codebase
- Call MCP server tools
Best for: code generation, refactoring, debugging, feature implementation.
Planning and documentation mode. Restricted tools:
- Read files (no write to code files)
- Write/edit text and markdown files
- Search codebase
When the plan is ready, switch to agent mode yourself for full tools and implementation.
Best for: architecture planning, writing specs, design documents, code review.
Use your editor session mode selector (or session/set_config_option).
Project rules (injected as {{.Rules}}) are discovered under the session working directory from .coddy/rules, .cursor/rules, .claude/rules, and .codex/rules when rules.auto_discover is true. See docs/rules.md.
Rule files often use Cursor-style frontmatter, for example:
---
description: "Go coding standards"
globs: ["**/*.go"]
alwaysApply: false
---
Write all comments in English.
Use fmt.Errorf("context: %w", err) for error wrapping.Slash commands and SKILL.md packs (injected as {{.Skills}}) are loaded from skills.dirs. Defaults: ${CODDY_HOME}/skills, ${CWD}/.skills, ~/.cursor/skills, ~/.claude/skills. See docs/config.md (skills) and docs/skills.md.
Connect external tools via MCP servers. Configured globally in config.yaml or
passed per-session by the ACP client.
Example adding a GitHub MCP server in config:
mcp_servers:
- name: "github"
command: "npx"
args: ["-y", "@modelcontextprotocol/server-github"]
env:
- name: "GITHUB_PERSONAL_ACCESS_TOKEN"
value: "${GITHUB_TOKEN}"See MCP Integration Guide for details.
Build with -tags gateway.telegram (Telegram only) or -tags gateway (all adapters) to enable coddy gateway.
make build TAGS="gateway.telegram"
./build/coddy gateway --config ~/.coddy/config.yamlMinimal config addition (config.yaml):
gateways:
telegram:
enabled: true
token: "${TELEGRAM_BOT_TOKEN}"
admins: [YOUR_USER_ID]
default_access: "admins" # all | admins | group:<name>
default_isolation: "admin" # individual | shared | adminEach user or chat gets its own isolated session. In group chats the bot responds only when @mentioned or replied to. /clear (no space) starts a fresh session.
Full guide — access levels, group isolation modes, per-chat overrides, and how to write adapters for new messengers: docs/gateway.md.
Full configuration reference in docs/config.md.
Key settings:
providers:
- name: local
type: openai
api_key: "${OPENAI_API_KEY}"
api_base: "${OPENAI_API_BASE}"
models:
- model: "local/gpt-4o"
max_tokens: 8192
temperature: 0.2
agent:
model: "local/gpt-4o"
max_turns: 30
tools:
require_permission_for_commands: trueACP client (editor / script / CI) Messenger (Telegram, …)
| |
JSON-RPC 2.0 over stdio gateway Hub (per adapter goroutine)
| |
ACP Server Layer session.Manager (shared)
| |
Session Manager ─────────────────────────────┘
|
ReAct Agent Loop
/ | | \
LLM Tools Skills MCP
See Architecture docs for full details.
- Build from source - prerequisites,
make build,TAGSvsgo build -tags,build/coddy - Updating Coddy -
coddy update, release assets,PATHvsmake install - Docker - GHCR image,
docker compose, bundled UI athttp://127.0.0.1:12345/ - Architecture - system design and component overview
- ACP Protocol - protocol reference and message formats
- ReAct Agent - ReAct loop design and tool specifications
- Configuration - full config file reference
- HTTP API - REST gateway (
-tags=http) and embedded UI (-tags=http,ui); includes/coddy/configfor live YAML editing from the SPA (#/settings). - Embedded UI - functional spec, Vite dev workflow, build tags
- DESIGN.md - UI tokens and layout (English)
- AGENTS.md - repo map and contributor notes for automation
- Rules - project rules (
.cursor/rules,.coddy/rules, …) - Skills - slash commands and
skills.dirs - MCP Integration - MCP server integration guide
- Messenger Gateway - Telegram bot adapter, session isolation, ACL, and how to write new adapters
examples/acp/acp_e2e_todo.py is a newline-delimited JSON-RPC harness against coddy acp ( stdbuf -oL, permission auto-reply, nil-result responses). Use it as reference when building your own minimal client rather than chaining naive echo lines into a pipe.
examples/acp/acp_e2e_memory.py drives build/coddy, an isolated CODDY_HOME, and RPA_API_KEY to verify recall, persist, and optional prune of markdown under $CODDY_HOME/memory. See the script docstring for flags. Overview of all harnesses - examples/README.md.
By default, coddy acp and coddy http store each session bundle under $CODDY_HOME/sessions/<sessionId>/ (default ~/.coddy/sessions/) with session.json, messages.json, an assets/ directory, and todos/active.md (plus todos/archive/ when completed lists are replaced). Override the root with coddy acp --sessions-dir, coddy http --sessions-dir, or sessions.dir in config.yaml. If the sessions directory cannot be created, startup fails with an error.
coddy sessions listprints stored sessions (--sessions-dirand--cwdfilters supported).coddy acp --session-id <id>makes the nextsession/neweither reopen snapshots for that folder (if present) or create a fresh bundle whose directory name matches that id.session/loadrestores history and notifies the client;session/listlists bundles for ACP-aware clients.
The coddy todo tools keep the active checklist mirrored to todos/active.md. A wholesale coddy_todo_plan_replace while items are incomplete is rejected until you finish rows or run coddy_todo_plan_archive; replacing when every row is completed moves the prior active.md into todos/archive/ (todo-<nanos>.md). coddy_todo_plan_archive finishes open rows to completed, writes todos/archive/plan_<unix_seconds>.md, then clears the session plan when persistence is on.
When the persisted plan is non-empty, the agent injects ### Current todo checklist plus rendered markdown checklist lines into the system prompt template (embedded defaults, or files under prompts.dir using prompts.agent_prompt and prompts.plan_prompt, which default to agent.md and plan.md) via {{if .TodoList}} … {{end}}. That block is omitted when there is nothing to track. Before each LLM call inside one session/prompt turn, Coddy refreshes that system message so a todo list created or updated earlier in the same ReAct episode stays visible immediately.
# Run tests
go test ./...
make test
# Example harnesses (see examples/README.md): ./examples/build_coddy.sh && ./examples/test_acp.sh && ./examples/test_httpserver.sh
# Full-featured local binary (HTTP + UI + scheduler), same defaults as Docker
make build TAGS="http ui scheduler memory"
./build/coddy -v # same as --version
# Run with debug logging (ACP mode); optional --log-output, --log-file, --log-format
coddy acp --log-level debug
# Single-line sanity check only (responses may omit JSON-RPC "result" for nil payloads; prefer examples/acp/acp_e2e_todo.py)
echo '{"jsonrpc":"2.0","id":0,"method":"initialize","params":{"protocolVersion":1,"clientCapabilities":{}}}' | coddy acpThis project is licensed under the MIT License, see the LICENSE file in the repository root for details.




