Ralph
Ralph is a small runner around GitHub Copilot CLI (standalone) inspired by the“Ralph Wiggum” technique: run a coding agent from a clean slate, over and over, until a stop condition is met
Install / Use
/learn @soderlind/RalphQuality Score
Category
Development & EngineeringSupported Platforms
README
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
Quick Start
# 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.
How It Works
Ralph implements the "Ralph Wiggum" technique:
- Read — Copilot reads your PRD (if attached) and progress file
- Pick — It chooses the highest-priority incomplete item
- Implement — It writes code for that one feature
- Verify — It runs your tests (
pnpm typecheck,pnpm test) - Update — It marks the item complete and logs progress
- Commit — It commits the changes
- Repeat — Until all items pass or it signals completion
https://github.com/user-attachments/assets/28206ee1-8dad-4871-aef5-1a9f24625dba
Learn More
- Matt Pocock's thread
- Ship working code while you sleep (video)
- 11 Tips For AI Coding With Ralph Wiggum
- Effective harnesses for long-running agents
Configuration
Choose a Model
Set the MODEL environment variable (default: gpt-5.2):
MODEL=claude-opus-4.5 ./ralph.sh --prompt prompts/default.txt --prd plans/prd.json --allow-profile safe 10
Define Your Work Items
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
}
]
| Field | Description |
|---------------|--------------------------------------------|
| category | "functional", "ui", or custom |
| description | One-line summary |
| steps | How to verify it works |
| passes | false → true when complete |
See the plans/ folder for more context.
Use Custom Prompts
Prompts are required. Use any prompt file:
./ralph.sh --prompt prompts/my-prompt.txt --allow-profile safe 10
Note: Custom prompts require
--allow-profileor--allow-tools.
Command Reference
ralph.sh — Looped Runner
Runs Copilot up to N iterations. Stops early on <promise>COMPLETE</promise>.
./ralph.sh [options] <iterations>
Examples:
./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
ralph-once.sh — Single Run
Runs Copilot once. Great for testing.
./ralph-once.sh [options]
Examples:
./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
Options
| Option | Description | Default |
|--------------------------|--------------------------------------|-----------------------|
| --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, --help | Show help | — |
Environment:
| Variable | Description | Default |
|----------|--------------------|-----------|
| MODEL | Model to use | gpt-5.2 |
Permission Profiles
| Profile | Allows | Use Case |
|----------|----------------------------------------|------------------------------|
| locked | write only | File edits, no shell |
| safe | write, shell(pnpm:*), shell(git:*) | Normal dev workflow |
| dev | All tools | Broad shell access |
Always denied: shell(rm), shell(git push)
Custom tools: If you pass --allow-tools, it replaces the profile defaults:
./ralph.sh --prompt prompts/wp.txt --allow-tools write --allow-tools 'shell(composer:*)' 10
Demo
Try Ralph in a safe sandbox:
# 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
Project Structure
.
├── 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
Install Copilot CLI
# Check version
copilot --version
# Homebrew
brew update && brew upgrade copilot
# npm
npm i -g @github/copilot
# Windows
winget upgrade GitHub.Copilot
Testing Prompts
Run all prompts in isolated worktrees:
./test/run-prompts.sh
Logs: test/log/
Copilot CLI Notes
Ralph is just a thin wrapper around the Copilot CLI. The important flags it relies on are:
Context attachment (-p "@file ...")
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.
Tool permissions (--allow-* / --deny-*)
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)).
Reliability niceties
- Single attachment workaround: Ralph combines PRD +
progress.txtinto one context file to avoid Copilot CLI issues with multiple@fileattachments. - Pseudo-TTY capture in the harness:
test/run-prompts.shusesscript(1)to capture output even when Copilot writes directly to the TTY.
Skills (--skill)
Skills let you prepend reusable instructions into the same attached context file. Pass a comma-separated list (repeatable):
--skill wp-block-developmentloadsskills/wp-block-development/SKILL.md--skill aa,bb,ccloadsskills/aa/SKILL.md,skills/bb/SKILL.md,skills/cc/SKILL.md
Example:
./ralph.sh --prompt prompts/wordpress-plugin-agent.txt \
--skill wp-block-development,wp-cli \
--prd plans/prd.json \
--allow-profile safe \
5
License
MIT — see LICENSE.
