Dotfiles ticket-creator
Create and refine tickets for the tk ticket system. Use when the user says 'create tickets for X', 'refine ticket X', 'break this into tickets', 'seed tickets from plan', or anything about creating or refining tk tickets.
git clone https://github.com/megalithic/dotfiles
T=$(mktemp -d) && git clone --depth=1 https://github.com/megalithic/dotfiles "$T" && mkdir -p ~/.claude/skills && cp -r "$T/home/common/programs/ai/pi-coding-agent/skills/ticket-creator" ~/.claude/skills/megalithic-dotfiles-ticket-creator && rm -rf "$T"
home/common/programs/ai/pi-coding-agent/skills/ticket-creator/SKILL.mdTicket creator
Create well-structured tickets that the ticket-worker skill can consume without ambiguity.
Ticket format contract
Every ticket must have:
| Field | Required | How |
|---|---|---|
| Title | Yes | Imperative, scoped action: |
| Description | Yes | What to do, why, and file hints () |
| Acceptance criteria | Yes | Numbered, each independently verifiable. Prefer criteria that map to a command |
| Type | Yes | , , , , |
| Dependencies | When order matters | — the second arg depends on the first |
| Parent | For subtasks | — ticket is a subtask of an epic |
Good ticket example
tk create "Fix token refresh returning 401 on expired tokens" \ -d "The /auth/refresh endpoint returns 401 when the refresh token is expired. Should return 403 with a clear error message instead. Relevant code in src/auth/refresh.ts and src/auth/middleware.ts." \ --acceptance "1. POST /auth/refresh with expired token returns 403 (not 401) 2. Response body includes 'error' field with descriptive message 3. Existing tests pass 4. New test covers expired token case" \ -t bug
Bad ticket example
tk create "Fix the bug" -d "There's a bug somewhere in auth"
No acceptance criteria, no file hints, no verification command. The agent will wander.
Acceptance criteria guidelines
- Each criterion must be independently verifiable
- Prefer criteria that map to a command: "tests pass", "linter clean", "curl returns X"
- Always include "existing tests still pass" as one criterion (unless no tests exist)
- For refactors: "behavior unchanged" + "tests pass" is sufficient
- Never use vague criteria like "code is clean" or "well-structured"
Parent vs dependency
Use
--parent when one ticket is a logical subtask of an epic (e.g., "implement login form" under "add auth system"). Use dep when two tickets are independent work items where one must finish before the other starts. When in doubt, use dep — it's the more common relationship.
Only add
dep when there is a real ordering constraint (e.g., ticket B requires code that ticket A will create). If two tickets could be worked on in parallel, do not link them with deps.
Tagging rule
Only add
--tags ready-for-development when the ticket is fully refined: description, acceptance criteria, and file hints are all present. Vague or wishlist tickets should have no tag — they live in the backlog until refined.
Refining is done via Mode 4 — do not hand-edit YAML frontmatter.
Size rule
A single ticket should be completable in one agent session (~30 min of agent work). If a task is larger, split it into multiple tickets with dependencies.
Modes
Mode 1: single ticket
User says: "create a ticket for X" or "add a ticket for X"
- Clarify scope if ambiguous
- Explore the codebase to find relevant files
- Create the ticket
- Show the created ticket to the user
If the request is vague ("add a ticket for that auth thing I mentioned"), create a backlog ticket: brief title, minimal description, no tag. The user or a later refine session will fill in the details.
If the request is specific enough to write proper acceptance criteria and file hints, create a fully-formed ticket with
--tags ready-for-development.
Mode 2: decompose
User says: "break this goal into tickets" or "create tickets for refactoring X"
- Understand the full goal
- Explore the codebase to understand scope and relevant files. If
exists at project root, runlat.md/
to discover relevant context from the knowledge graphlat search "<goal description>" - Seed context file if it doesn't exist (see "Context seeding" below)
- Break into small, independently completable tickets
- Create tickets in dependency order (create the prerequisite tickets first so you have their IDs)
- Set dependencies:
(downstream depends on upstream)tk dep <downstream-id> <upstream-id> - Validate the dep graph:
andtk dep cycle
— if there are cycles or no ready tickets, fix before proceedingtk ready - Show all created tickets and their dependency chain
Mode 3: seed from plan
User says: "seed tickets from this plan" or provides a plan file path
- Read the plan file
- Explore the codebase to understand scope and relevant files. If
exists at project root, runlat.md/
to discover relevant context from the knowledge graphlat search "<goal description>" - Seed context file if it doesn't exist (see "Context seeding" below)
- Identify discrete work items
- Create tickets for each, preserving the plan's ordering via dependencies
- Show the created tickets
Mode 4: refine
User says: "refine ticket X" or "refine backlog tickets"
Turn a vague backlog ticket into a workable one. If the ticket is too large, split it first (Mode 2).
- Read the ticket:
tk show <id> - If the ticket already has
tag, tell the user it's already refined and stopready-for-development - Explore the codebase to find relevant files, understand scope, and gather context
- If the ticket is unclear, ask the user for clarification before proceeding
- Rewrite the ticket file with:
- Clear title (imperative, scoped)
- Description with what to do, why, and file hints
- Numbered acceptance criteria (each independently verifiable)
- Correct type if the current one is wrong
- Add the
tag to the frontmatterready-for-development - Show the refined ticket to the user
Batch refine: if the user says "refine backlog tickets" or "refine all", repeat for each untagged open ticket. Stop on the first one that needs user clarification — don't guess.
To rewrite a ticket file, use the
write tool on the file path shown by tk show. Preserve the existing YAML frontmatter fields (id, created, deps, links, parent) and only update: title, description, acceptance criteria, type, tags.
Context seeding
Modes 2 and 3 create a context file to avoid redundant discovery in each ticket-worker session.
Prerequisite: if a
lat.md/ directory exists at the project root, skip context seeding entirely. The lat search command (run in step 2) provides living, queryable context that makes a static context file redundant.
After exploring the codebase (step 2 in both modes), check if
plans/.ticket-context.md exists. If it does, skip — the context is already seeded. If not, create it:
mkdir -p plans
Write
plans/.ticket-context.md with three sections:
## Verification commands - Build: <command> - Test: <command> - Lint: <command> ## Key directories - path/ — description ## Conventions - Description of relevant patterns
Fill in each section from what you discovered during exploration:
- Verification commands: look in
(scripts, tasks, git-hooks),devenv.nix
(scripts.test, scripts.lint, scripts.build),package.json
(test, lint, check, build targets),Makefile
(checks)flake.nix - Key directories: list directories relevant to the batch of tickets being created
- Conventions: notable patterns found during exploration (error handling style, file structure, naming)
The context file is project-local and reusable across all tickets in the batch.
Self-validation (always run after Mode 2/3)
After creating all tickets, run these checks:
— confirm all tickets are opentk list- For each ticket:
— verify:tk show <id>- Description has file hints
- Acceptance criteria are numbered and independently verifiable
- If any are vague, refine immediately (Mode 4)
— no cycles allowed. If found, fix.tk dep cycle
— at least one ticket must be unblockedtk ready -T ready-for-development- If no tickets are ready, fix dependency ordering
Report: number of tickets created, any issues found and fixed, whether tickets are ready for work.
Workflow
- Read the user's request
- Explore the codebase to understand relevant files and context
- Draft ticket(s) mentally — title, description with file hints, acceptance criteria
- Create tickets via
with all fields populated. Use real newlines intk create
and-d
arguments — never pass literal--acceptance
characters\n - If multiple tickets: set dependencies via
tk dep <id> <dep-id> - Run self-validation (see above)
- Present created tickets for review