Skip to content

Ralph (Copilot CLI runner)

Let AI implement your features while you sleep.

Ralph runs GitHub Copilot CLI in a loop, implementing one feature at a time until your PRD is complete.

Quick Start · How It Works · Configuration · Command Reference · Demo


Terminal window
# Clone and enter the repo
git clone https://github.com/soderlind/ralph
cd ralph
# Add your work items to plans/prd.json
# Test with a single run
./ralph-once.sh --prompt prompts/default.txt --prd plans/prd.json --allow-profile safe
# Run multiple iterations
./ralph.sh --prompt prompts/default.txt --prd plans/prd.json --allow-profile safe 10

Check progress.txt for a log of what was done.


Ralph implements the “Ralph Wiggum” technique:

  1. Read — Copilot reads your PRD (if attached) and progress file
  2. Pick — It chooses the highest-priority incomplete item
  3. Implement — It writes code for that one feature
  4. Verify — It runs your tests (pnpm typecheck, pnpm test)
  5. Update — It marks the item complete and logs progress
  6. Commit — It commits the changes
  7. Repeat — Until all items pass or it signals completion

https://github.com/user-attachments/assets/28206ee1-8dad-4871-aef5-1a9f24625dba


Set the MODEL environment variable (default: gpt-5.2):

Terminal window
MODEL=claude-opus-4.5 ./ralph.sh --prompt prompts/default.txt --prd plans/prd.json --allow-profile safe 10

Create plans/prd.json with your requirements:

[
{
"category": "functional",
"description": "User can send a message and see it in the conversation",
"steps": ["Open chat", "Type message", "Click Send", "Verify it appears"],
"passes": false
}
]
FieldDescription
category"functional", "ui", or custom
descriptionOne-line summary
stepsHow to verify it works
passesfalsetrue when complete

See the plans/ folder for more context.

Prompts are required. Use any prompt file:

Terminal window
./ralph.sh --prompt prompts/my-prompt.txt --allow-profile safe 10

Note: Custom prompts require --allow-profile or --allow-tools.


Runs Copilot up to N iterations. Stops early on <promise>COMPLETE</promise>.

Terminal window
./ralph.sh [options] <iterations>

Examples:

Terminal window
./ralph.sh --prompt prompts/default.txt --prd plans/prd.json --allow-profile safe 10
./ralph.sh --prompt prompts/wp.txt --allow-profile safe 10
MODEL=claude-opus-4.5 ./ralph.sh --prompt prompts/default.txt --prd plans/prd.json --allow-profile safe 10

Runs Copilot once. Great for testing.

Terminal window
./ralph-once.sh [options]

Examples:

Terminal window
./ralph-once.sh --prompt prompts/default.txt --prd plans/prd.json --allow-profile safe
./ralph-once.sh --prompt prompts/wp.txt --allow-profile locked
MODEL=claude-opus-4.5 ./ralph-once.sh --prompt prompts/default.txt --prd plans/prd.json --allow-profile safe
OptionDescriptionDefault
--prompt <file>Load prompt from file (required)
--prd <file>Optionally attach a PRD JSON file
--skill <a[,b,...]>Prepend skills from skills/<name>/SKILL.md
--allow-profile <name>Permission profile (see below)
--allow-tools <spec>Allow specific tool (repeatable)
--deny-tools <spec>Deny specific tool (repeatable)
-h, --helpShow help

Environment:

VariableDescriptionDefault
MODELModel to usegpt-5.2
ProfileAllowsUse Case
lockedwrite onlyFile edits, no shell
safewrite, shell(pnpm:*), shell(git:*)Normal dev workflow
devAll toolsBroad shell access

Always denied: shell(rm), shell(git push)

Custom tools: If you pass --allow-tools, it replaces the profile defaults:

Terminal window
./ralph.sh --prompt prompts/wp.txt --allow-tools write --allow-tools 'shell(composer:*)' 10

Try Ralph in a safe sandbox:

Terminal window
# Setup
git clone https://github.com/soderlind/ralph && cd ralph
git worktree add ../ralph-demo -b ralph-demo
cd ../ralph-demo
# Run
./ralph-once.sh --prompt prompts/default.txt --prd plans/prd.json --allow-profile safe
./ralph.sh --prompt prompts/default.txt --prd plans/prd.json --allow-profile safe 10
# Inspect
git log --oneline -20
cat progress.txt
# Cleanup
cd .. && git worktree remove ralph-demo && git branch -D ralph-demo

.
├── plans/prd.json # Your work items
├── prompts/default.txt # Example prompt
├── progress.txt # Running log
├── ralph.sh # Looped runner
├── ralph-once.sh # Single-run script
└── test/run-prompts.sh # Test harness

Terminal window
# Check version
copilot --version
# Homebrew
brew update && brew upgrade copilot
# npm
npm i -g @github/copilot
# Windows
winget upgrade GitHub.Copilot

Run all prompts in isolated worktrees:

Terminal window
./test/run-prompts.sh

Logs: test/log/


Ralph is just a thin wrapper around the Copilot CLI. The important flags it relies on are:

Ralph passes context to Copilot by attaching a file directly in the prompt using Copilot’s @file syntax (for example: -p "@.ralph-context... Follow the attached prompt.").

Ralph builds one temporary “attachment” file per iteration that typically contains:

  • progress.txt (always)
  • PRD JSON (only if you pass --prd <file>)
  • The selected prompt file (from --prompt <file>)

This keeps the agent’s input structured and avoids inlining large blobs into command-line flags.

Ralph controls what Copilot is allowed to do by passing tool permission flags:

  • --allow-profile <safe|dev|locked>: convenience presets implemented by Ralph.
  • --allow-tools <spec>: allow a specific tool spec (repeatable). When you use this, it replaces the profile defaults.
  • --deny-tools <spec>: deny a specific tool spec (repeatable).

For shell tools, prefer the pattern form shell(cmd:*) (for example shell(git:*)).

Ralph always denies a small set of dangerous commands (currently shell(rm) and shell(git push)).

  • Single attachment workaround: Ralph combines PRD + progress.txt into one context file to avoid Copilot CLI issues with multiple @file attachments.
  • Pseudo-TTY capture in the harness: test/run-prompts.sh uses script(1) to capture output even when Copilot writes directly to the TTY.

Skills let you prepend reusable instructions into the same attached context file. Pass a comma-separated list (repeatable):

  • --skill wp-block-development loads skills/wp-block-development/SKILL.md
  • --skill aa,bb,cc loads skills/aa/SKILL.md, skills/bb/SKILL.md, skills/cc/SKILL.md

Example:

Terminal window
./ralph.sh --prompt prompts/wordpress-plugin-agent.txt \
--skill wp-block-development,wp-cli \
--prd plans/prd.json \
--allow-profile safe \
5

MIT — see LICENSE.