Skillshare skillshare-implement-feature
git clone https://github.com/runkids/skillshare
T=$(mktemp -d) && git clone --depth=1 https://github.com/runkids/skillshare "$T" && mkdir -p ~/.claude/skills && cp -r "$T/.skillshare/skills/skillshare-implement-feature" ~/.claude/skills/runkids-skillshare-skillshare-implement-feature && rm -rf "$T"
.skillshare/skills/skillshare-implement-feature/SKILL.mdImplement a feature following TDD workflow. $ARGUMENTS is a spec file path (e.g.,
specs/my-feature.md) or a plain-text feature description.
Scope: This skill writes Go code and tests. It does NOT update website docs (use
update-docs after) or CHANGELOG (use changelog after).
Workflow
Step 1: Understand Requirements
If $ARGUMENTS is a file path:
- Read the spec file
- Extract acceptance criteria and edge cases
- Identify affected packages
If $ARGUMENTS is a description:
- Search existing code for related functionality
- Identify the right package to extend
- Confirm scope with user before proceeding
Step 2: Identify Affected Files
List all files that will be created or modified:
# Typical pattern for a new command cmd/skillshare/<command>.go # Command handler cmd/skillshare/<command>_project.go # Project-mode handler (if dual-mode) internal/<package>/<feature>.go # Core logic tests/integration/<command>_test.go # Integration test
Display the file list and continue. If scope is unclear, ask the user.
Step 3: Write Failing Tests First (RED)
Write integration tests using
testutil.Sandbox:
func TestFeature_BasicCase(t *testing.T) { sb := testutil.NewSandbox(t) defer sb.Cleanup() // Setup sb.CreateSkill("test-skill", map[string]string{ "SKILL.md": "---\nname: test-skill\n---\n# Content", }) // Act result := sb.RunCLI("command", "args...") // Assert result.AssertSuccess() result.AssertOutputContains("expected output") }
Verify tests fail:
make test-int # or run specific test: go test ./tests/integration -run TestFeature_BasicCase
Step 4: Implement (GREEN)
Write minimal code to make tests pass:
- Follow existing patterns in
andcmd/skillshare/internal/ - Use
for terminal output (colors, spinners, boxes)internal/ui - Add oplog instrumentation for mutating commands:
start := time.Now() // ... do work ... e := oplog.NewEntry("command-name", statusFromErr(err), time.Since(start)) oplog.Write(configPath, oplog.OpsFile, e) - Register command in
commands map if new commandmain.go
Verify tests pass:
make test-int
Step 5: Refactor and Verify
- Clean up code while keeping tests green
- Run full quality check:
make check # fmt-check + lint + test - Fix any formatting or lint issues
Project Patterns Reference
These patterns appear throughout the codebase. Follow them when implementing new features.
Handler Split Convention
Large commands are split by concern rather than kept in a single file. When a command handler grows beyond ~300 lines, split it:
| Suffix | Purpose | Example |
|---|---|---|
| Flag parsing + mode routing (dispatch) | |
| Core handler logic | |
/ | Output rendering | |
/ | Decision/prompt logic | |
| Full-screen TUI (bubbletea) | |
| Batch operation orchestration | |
| Target/skill resolution | |
| Mode-specific context struct | |
| Output formatting helpers | |
Principle: dispatch file does ONLY flag parsing + mode routing. Logic goes in sub-files.
Dual-Mode Command Pattern
Most commands support both global (
-g) and project (-p) mode:
func handleMyCommand(args []string) error { mode, rest, err := parseModeArgs(args) if err != nil { return err } switch mode { case modeProject: return handleMyCommandProject(rest) default: return handleMyCommandGlobal(rest) } }
Create
<cmd>_project.go for project-mode handler. Use parseModeArgs() from mode.go.
TUI Components (bubbletea)
All interactive prompts use bubbletea (not survey). Key components:
— shared checklist/radio pickerchecklist_tui.go
— filterable list with detail panellist_tui.go
— multi-select checkbox listsearch_tui.go
Color palette: cyan
Color("6"), gray Color("8"), yellow #D4D93C.
Dispatch order: JSON output → TUI (if TTY + items + !
--no-tui) → empty check → plain text.
Web API Endpoint
If the feature needs a Web UI endpoint, add
internal/server/handler_<name>.go:
func (s *Server) handle<Name>(w http.ResponseWriter, r *http.Request) { // ... writeJSON(w, result) // 200 OK with JSON // writeError(w, 400, msg) // for errors }
Register in
server.go route setup. Branch on s.IsProjectMode() for mode-specific behavior.
Oplog Instrumentation
All mutating commands log to
operations.log (JSONL):
start := time.Now() // ... do work ... e := oplog.NewEntry("command-name", statusFromErr(err), time.Since(start)) e.Args = map[string]any{"key": value} oplog.Write(configPath, oplog.OpsFile, e)
Security scans write to
oplog.AuditFile instead.
Step 6: E2E Runbook (Major Features Only)
If the feature meets any of these criteria, generate an E2E runbook:
- New command or subcommand
- Changes to install/uninstall/sync flow
- Security-related (audit, hash verification, rollback)
- Multi-step user workflow (init → install → sync → verify)
- Edge cases that integration tests alone can't cover (Docker, network, file permissions)
Generate
ai_docs/tests/<slug>_runbook.md following the existing convention:
# CLI E2E Runbook: <Title> <One-line summary of what this validates.> **Origin**: <version> — <why this runbook exists> ## Scope - <bullet list of behaviors being validated> ## Environment Run inside devcontainer with `ssenv` isolation. ## Steps ### 1. Setup: <description> \```bash <commands> \``` **Expected**: <what should happen> ### 2. <Action>: <description> ... ## Pass Criteria - All steps marked PASS - <additional criteria>
Key conventions:
- YAML-free, pure Markdown
- Each step has
block +bash
blockExpected
=ss
,skillshare
= ssenv-isolated HOME~- Runbook can be executed by the
skillcli-e2e-test
If the feature does not meet the criteria above, skip this step.
Step 7: Stage and Report
- List all created/modified files
- Confirm each acceptance criterion is met with test evidence
- Remind user to run
if the feature affects CLI flags or user-visible behaviorupdate-docs
Rules
- Test-first — always write failing test before implementation
- Minimal code — only write what's needed to pass tests
- Follow patterns — match existing code style in each package
- 3-strike rule — if a test fails 3 times after fixes, stop and report what's blocking
- No docs — this skill writes code only; use
for documentationupdate-docs - No changelog — use
skill for release noteschangelog - Spec ambiguity — ask the user rather than guessing