skills/linear-grill/SKILL.md
Grill a collaborator's Linear tickets and move every processed ticket to where it belongs. Resolves the board from the repo's .ro-linear.json, reads the collaborator's Backlog / Ready-for-agent issues, then per ticket either posts 3-5 decision-extracting questions (state moves to "Question for <name>") or confirms it build-ready (state moves to "Grilled", the gate /ro:linear-nightshift consumes); shipped-and-confirmed tickets close as Done. The async-collaborator counterpart of /ro:day-shift for people who never touch GitHub. Triggers on "grill linear", "grill eoin's tickets", "linear grill", "add questions to the linear tickets", "/ro:linear-grill".
npx skillsauth add RonanCodes/ronan-skills linear-grillInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Process a collaborator's Linear tickets so every one ends the run in one of three places: Question for <name> (we asked, they answer async in comments), Grilled (build-ready, the gate /ro:linear-nightshift consumes), or Done (shipped and confirmed). The agent moves the tickets; the collaborator only ever answers in comments (dragging a ticket to "Ready for agent" is their priority nudge, not a build-ready signal). Pairs with /ro:linear-nightshift (consumes Grilled) and /ro:linear-update (closes the loop on deploy).
Part of the hybrid one-way sync model (locked 2026-06-11, see ronan-skills#61): Linear is where collaborator stories are born and prioritised; GitHub is the execution layer. This skill is the grilling step that makes a born story buildable.
https://api.linear.app/graphql. Header Authorization: <key> — NO Bearer prefix for personal API keys. Key: LINEAR_API_KEY in ~/.claude/.env.@[displayName](user-uuid) in markdown bodies renders as a real mention and notifies. Get the uuid via { users { nodes { id name displayName } } }.subscriberIds: [<their-uuid>] so they're notified too.team(id){ states { nodes { id name type } } }, match by name (Backlog / Question for <name> / Grilled / In Progress / Done)..ro-linear.json)The current repo maps to its Linear board via a committed .ro-linear.json at the repo root:
{
"teamKey": "NUT",
"collaborator": "econnolly321",
"states": { "questions": "Question for Eoin", "grilled": "Grilled" }
}
If the file exists, skip discovery entirely. If it doesn't: infer the team (repo name vs team names, or the one obvious collaborator team), ask when ambiguous, then WRITE the file and commit it so the next run starts instantly. Not in a repo, or no plausible team: ask.
--team <KEY> overrides the config file; both override workspace inference.--collaborator <name>: config file, else the team's non-Ronan member with created issues (match on users.displayName).--issue <ID> list if given; else ALL team issues in backlog-type or unstarted-type states except Grilled (so: Backlog, Ready for agent, Question for <name>), regardless of creator. Ronan files proposal/menu tickets on the collaborator's board too (e.g. NUT-61, a "which of these would you use?" ticket); a creator filter silently drops their answers to those. Tickets already in Grilled are the nightshift's queue, leave them alone.Read each candidate's comments, latest grill comment first:
The comment shape that works (worked example: NUT-50…54 + NUT-55…58, 2026-06-11):
@[<displayName>](<uuid>) — a few questions so this gets built the way you picture it:.Post via commentCreate. With --dry-run, print the would-be comments instead.
The board is the report; never leave a processed ticket where you found it. Three outcomes, each a comment + issueUpdate of stateId:
questions from the config (e.g. "Question for Eoin"; create it, type backlog, if missing).grilled (create it, type unstarted, if missing). Creating a state that already exists fails with "duplicate workflow state": on that error, look it up with workflowStates(includeArchived: true, filter: {team, name}) instead.Print per ticket: questioned / grilled (build-ready) / done / skipped (open grill). The cycle: the collaborator answers in comments, the NEXT grill run promotes answered tickets to Grilled, and /ro:linear-nightshift drains Grilled.
| Condition | Behaviour |
|---|---|
| No LINEAR_API_KEY | Fail loudly, point at ~/.claude/.env |
| Collaborator not found in workspace | List members, ask |
| Mention renders literally (@[name](id) as text) | Fallback: profile-URL link form + subscriberIds; note it in the report |
/ro:linear-nightshift — consumes the Grilled tickets this skill matured/ro:linear-update — closes the loop on deploy/ro:day-shift — the GitHub-native sibling for Ronan's own backlogdevelopment
Close the loop on a Linear ticket when its work ships - move the status and post a deploy comment with the PR link, what shipped, and a try-it link, mentioning the collaborator. Used as the tail of /ro:linear-nightshift for every merged mirror, or manually after an ad-hoc build. Triggers on "linear update", "update the linear ticket", "mark NUT-x done", "tell eoin it shipped", "/ro:linear-update".
devops
Run a night-shift against a collaborator's Linear board. Pulls the team's Grilled tickets (/ro:linear-grill moves a ticket to Grilled once its questions are answered), VERIFIES the questions were actually answered (unanswered → bounce the ticket to the "Question for <name>" state), mirrors verified tickets to ephemeral GitHub issues with ready-for-agent, then runs the standard /ro:night-shift machinery on GitHub. Tail-calls /ro:linear-update for everything that merged + deployed. Triggers on "linear nightshift", "nightshift linear", "drain the linear board", "run the shift off linear", "/ro:linear-nightshift".
development
--- name: about-page description: Add a standard About page to any web app, what it is, the tech stack, and an FAQ, wired into a footer link with a sticky footer. Built with Spartan + Tailwind (the canonical component layer) and falls back to semantic HTML so it ships reliably. Use whenever building, polishing, or shipping an app, every app should have one. Triggers on "add an about page", "about page", "footer about link", or as a standard step in app build/polish. category: frontend argument-h
data-ai
Wire the common "invite-only with a waitlist" access pattern into a TanStack Start + Better Auth + Drizzle + Resend app — passwordless email-OTP sign-in gated by an email allowlist, a public waitlist capture with confirmation email, a "you're in" email when an address is whitelisted, a super-admin / admin / member role tier, and a super-admin-only admin section to view users, view the waitlist, whitelist emails, and promote admins. Use when the user wants invite-only access, an allowlist/whitelist of emails, a waitlist with emails, gated sign-up, an admin user-management section, role tiers, or "only let approved people in".