plugins/lisa-agy/skills/linear-build-intake/SKILL.md
Symmetric counterpart to lisa:jira-build-intake on the Linear side. Scans a Linear team for Issues carrying the configured `ready` build label, claims the first eligible Issue by relabeling to the configured `claimed` label, runs the implementation/build flow via lisa:linear-agent, relabels to the configured `done` label on completion, then exits. Enforces the claim-time arm of the `leaf-only-lifecycle` rule: a parent/container with open child work (or a childless Epic) that still carries a stale build-ready label is skipped or safe-blocked with a lifecycle-repair comment, never claimed. The `ready` label is the human-flipped signal that an Issue is truly ready for development — mirroring how Notion PRDs work Draft → Ready → (us) In Review → Blocked|Ticketed.
npx skillsauth add codyswanngt/lisa linear-build-intakeInstall 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.
$ARGUMENTS is one of:
ENG) — scans that team for ready Issues.linear — falls back to linear.teamKey from .lisa.config.json.Run one build-intake cycle. The first eligible ready Issue is claimed, built via the lisa:linear-agent flow, relabeled to the configured done label on completion, then the cycle exits. Remaining ready Issues stay queued for later scheduler invocations.
This skill is the destination of the lisa:tracker-build-intake shim when tracker = "linear".
Build-queue label names are read from .lisa.config.json linear.labels.build.*, falling back to defaults documented in the config-resolution rule. Bash pattern:
# Read role with default fallback. Local overrides global per-key.
read_role() {
local role="$1" default="$2"
local local_v global_v
local_v=$(jq -r ".linear.labels.build.${role} // empty" .lisa.config.local.json 2>/dev/null)
global_v=$(jq -r ".linear.labels.build.${role} // empty" .lisa.config.json 2>/dev/null)
echo "${local_v:-${global_v:-$default}}"
}
READY=$(read_role ready "status:ready")
CLAIMED=$(read_role claimed "status:in-progress")
REVIEW=$(read_role review "status:code-review")
For env-keyed done, resolve the env first, then look up done[<env>]:
target_env=staging) wins.deploy.branches (reverse lookup).done is a string in config, use it directly regardless of env.done is a map and env cannot be resolved, fail loudly — do not pick arbitrarily.TARGET_ENV="${target_env:-}"
if [ -z "$TARGET_ENV" ] && [ -n "$PR_BASE_BRANCH" ]; then
TARGET_ENV=$(jq -r --arg b "$PR_BASE_BRANCH" \
'.deploy.branches // {} | to_entries[] | select(.value == $b) | .key' \
.lisa.config.json 2>/dev/null | head -1)
fi
DONE_TYPE=$(jq -r '.linear.labels.build.done | type' .lisa.config.json 2>/dev/null)
if [ "$DONE_TYPE" = "string" ]; then
DONE=$(jq -r '.linear.labels.build.done' .lisa.config.json)
elif [ "$DONE_TYPE" = "object" ]; then
[ -z "$TARGET_ENV" ] && { echo "ERROR: linear.labels.build.done is env-keyed but env not resolvable"; exit 1; }
DONE=$(jq -r --arg e "$TARGET_ENV" '.linear.labels.build.done[$e] // empty' .lisa.config.json)
[ -z "$DONE" ] && { echo "ERROR: linear.labels.build.done has no entry for env '$TARGET_ENV'"; exit 1; }
else
case "$TARGET_ENV" in
dev) DONE="status:on-dev" ;;
staging) DONE="status:on-stg" ;;
production) DONE="status:done" ;;
*) echo "ERROR: cannot resolve done label without env"; exit 1 ;;
esac
fi
In prose below, the role names refer to the resolved labels: e.g. "the ready label" means whatever linear.labels.build.ready resolves to (default: status:ready).
Linear's per-team workflow state names vary (Todo / Backlog / Up Next / etc.). Labels are workspace-scoped or team-scoped and stable across teams, so we drive the build queue off labels rather than chasing renamed native states. The native state field is informational until terminal completion; at the true terminal done value, the leaf-only-lifecycle rule requires native closure by moving the Issue to a configured Done / Completed workflow state when one exists.
Reads linear.workspace, linear.teamKey, and linear.labels.build.* from .lisa.config.json (with .local override).
Do NOT ask the caller whether to proceed. Once invoked with a team key, run the cycle to completion — claim and dispatch the first eligible Issue through lisa:linear-agent, transition a successful build to $DONE, write the summary, and exit. The caller (a human or a cron) has already authorized the run by invoking the skill.
Specifically forbidden:
status:blocked by lisa:linear-agent's pre-flight gate. The pre-flight status:blocked outcome is a valid terminal state of the per-Issue lifecycle.The only legitimate reasons to stop early:
ready label does not exist on the team's labels). Surface and exit with an Adoption hint."No Linear Issues labeled $READY. Nothing to do."Linear build queue uses these issue-level labels:
ready → claimed → review → done(env-keyed) (downstream)
(human/PM) (us claim) (us PR ready) (us build done)
(Defaults: status:ready / status:in-progress / status:code-review / status:on-dev/status:on-stg/status:done.)
This skill ONLY transitions $READY → $CLAIMED on claim, and $CLAIMED → $DONE on completion. It never touches status:done-as-terminal, $REVIEW (owned by lisa:linear-agent / lisa:linear-evidence), or status:blocked (owned by lisa:linear-agent's pre-flight gate).
Pre-flight check: at start of each cycle, confirm $READY, $CLAIMED, and the relevant $DONE variants exist on the team via mcp__linear-server__list_issue_labels. If $READY is missing, stop and report adoption needed. The other labels can be created on demand.
$ARGUMENTS:
linear → fall back to linear.teamKey from config.mcp__linear-server__list_teams({query: <teamKey>}).Query: mcp__linear-server__list_issues({team: <teamId>, label: "$READY"}).
Capture each Issue's: identifier, title, type label, priority, assignee, project, labels, description summary.
If empty, report "No Linear Issues labeled $READY. Nothing to do." and exit. Common idle case.
A Linear team can oversee multiple repos (frontend / backend / infrastructure). This skill claims only Issues for the repo it is running in. Run this gate before the leaf-only gate (3a) and the claim (3b), per the repo-scope-split rule's "Claim-time repo scoping" section (cite it by slug; do not restate its decision table).
config-resolution "Repo scoping" (.repo → .github.repo → git remote get-url origin basename). If unresolvable, stop and report.repo:<current> label. Keep the Phase 2 scan broad so unlabeled Issues are still seen, determined, and stamped.repo-scope-split):
repo:<other> → skip (leave it ready for that repo's own intake); next candidate.repo:<name> via mcp__linear-server__save_issue (resolve/create the label via list_issue_labels/create_issue_label) so later cycles filter cheaply; re-apply with the now-known repo.repo-scope-split work-time procedure into single-repo siblings, each created build-ready (build_ready: true) and stamped with its own repo:<name>; the current repo's sibling becomes a normal candidate."No ready Issues for repo <current>. Nothing to do.".Build intake claims only independently implementable leaf work units. This enforces the claim-time arm of the vendor-neutral leaf-only-lifecycle rule: a parent/container that still carries a stale build-ready label (e.g. status:ready applied before this rule existed, or hand-applied to a Project-grouped parent Issue) is never claimed — intake skips it or safe-blocks it with a clear lifecycle-repair message. It is the claim-time complement to the write-time labeling in lisa:linear-write-issue and the validate-time S15 gate in lisa:linear-validate-issue; all three cite the same rule so the classification never drifts. Never silently implement a container.
Run this gate before the claim relabel, starting with the oldest/highest-priority ready candidate. Do NOT relabel, comment "Claimed", or invoke lisa:linear-agent for an Issue that fails the gate.
Resolve container vs. leaf — structural first, then nominal. Per leaf-only-lifecycle the classification is structural: an Issue is a container if it has open child work, whatever its declared type; otherwise the type label decides. Resolve child work using the same hierarchy lisa:linear-read-issue uses — Linear's native parentage: an Issue groups sub-issues via parentId, and a Project (the Epic equivalent) groups Issues via projectId. Relations (save_issue_relation — blocks / is blocked by) express dependencies and are not parentage — do not count them as children.
Fetch the Issue's sub-issues via mcp__linear-server__get_issue (which returns the children) or mcp__linear-server__list_issues({parentId: <issueId>}), then count those still open (Linear state.type not in the completed/canceled set):
# Children of <issueId>: native sub-issues via parentId.
# Count children whose Linear state.type is NOT terminal ("completed" / "canceled").
# A parent whose children are all completed is no longer holding open work and
# rolls up via leaf-only-lifecycle's rollup, not here.
OPEN_CHILDREN = count(list_issues({parentId: <issueId>})
where state.type not in {"completed", "canceled"})
For a Project-level parent (an Issue that itself anchors a projectId grouping rather than a parentId tree), resolve membership the same way lisa:linear-read-issue does and treat the parent as a container if any grouped Issue is still open. If sub-issue resolution is unavailable, fall back to the parentage lisa:linear-read-issue derives and treat the Issue as a container if any derived child is open. Note "sub-issues unavailable — parentage derived" so the operator knows how children were resolved.
Classify and act (first match wins). The type comes from the Issue's type: label (type:Epic, type:Story, type:Spike, type:Bug, type:Task, type:Sub-task, type:Improvement):
| Condition | Class | Action |
|---|---|---|
| OPEN_CHILDREN > 0 (open child work, any type) | Container | Skip / safe-block — do NOT claim |
| no open children AND type = Epic (a Linear Project) | Childless Epic/Project (pure rollup container) | Skip / safe-block — do NOT claim |
| no open children AND type ≠ Epic (Bug, Task, Sub-task, Improvement, Story, Spike, or no type: label) | Leaf work unit | Proceed to 3b claim |
The childless-parent exception promotes every childless type except Epic to a claimable leaf: a childless Story is a directly shippable increment and a childless Spike is the investigation unit, so neither is stranded. Only a childless Epic (a Linear Project) stays unclaimed — it is a pure rollup container by design, and a childless one is an incomplete decomposition or a mis-applied role, never an implementable unit.
Safe-block (default action for a flagged container). Leave the build-ready label in place (don't silently strip it — that hides the lifecycle error), post a single lifecycle-repair comment, record the Issue under "Skipped (container)" in the summary, and end the cycle. Do NOT relabel to $CLAIMED. Keep the comment idempotent — skip posting if an identical [claude-build-intake] lifecycle-repair comment already exists on the Issue, so a re-entrant cycle doesn't spam it.
Post via mcp__linear-server__save_comment with:
[claude-build-intake] Not claimed: this Issue carries the build-ready label ($READY) but is a container with open child work (or a childless Epic), which violates the leaf-only-lifecycle rule. Build-ready (status:ready) is leaf-only per leaf-only-lifecycle — an agent claims and implements leaves, never a container. Repair: move $READY off this parent onto its leaf children (or, for a childless Epic, decompose it into leaf children or reclassify it to a leaf type). A parent's lifecycle state rolls up from its children and is never set to ready directly.
This gate never blocks a legitimate flat Task/Bug: those have no open children and a leaf type:, so they fall straight through to the claim in 3b.
Update labels via mcp__linear-server__save_issue: remove $READY, add $CLAIMED. Resolve label IDs via list_issue_labels (create $CLAIMED if missing).
Post a [claude-build-intake] comment via save_comment: "Claimed by Claude. Starting build."
This is the idempotency lock — a re-entrant cycle's label: $READY filter will not see this Issue again.
If the relabel fails (permission, race), record under "Errors" and skip. Do not invoke the build flow on an Issue you didn't successfully claim.
Invoke lisa:linear-agent (per-Issue lifecycle agent) with the Issue identifier. lisa:linear-agent owns:
lisa:linear-read-issue)lisa:linear-verify)lisa:ticket-triage)lisa:linear-synclisa:linear-evidenceWait for the agent to return. Capture its outcome:
done env status.lisa:linear-agent itself relabels to status:blocked and assigns to creator. Let it stand. Record and move on.$CLAIMED. Surface to human; do not auto-transition. Record under "Errors".$CLAIMED. Record with exception summary.A done env state (status:on-dev, status:on-stg, or the terminal value) asserts that the code has actually reached that environment. Never set it for a PR that is merely open: auto-merge can be blocked indefinitely (a required rebase / BEHIND branch, failing checks, an unaddressed review), and the change may never land. Relabeling an Issue status:on-stg on an open PR makes it claim a deploy that never happened. Transition only after confirming the PR merged.
If lisa:linear-agent returned Success:
gh pr view <pr> --json state,mergedAt,mergeStateStatus,url:
state == MERGED) → proceed to resolve and apply $DONE below. Where the env deploy is observable (a deploy workflow run / deployment status keyed to the merged-into branch via deploy.branches), confirm it did not fail before relabeling; a still-running deploy is treated like an open PR (leave at $CLAIMED), a failed deploy is recorded as an Error.mergeStateStatus), leave it at $CLAIMED, and stop. A later lisa:repair-intake cycle drives the open PR to merge — re-syncing a BEHIND branch so the already-enabled auto-merge can land, or surfacing a real blocker — and, once merged, applies this same env transition. Do not comment "Build complete" or change the native state.$CLAIMED.$DONE for this issue's PR base branch using the Workflow resolution algorithm above. If env can't be resolved and done is env-keyed, record an Error and skip this transition — never guess.$DONE is the true terminal done value per the leaf-only-lifecycle rule's Terminal native closure section:
linear.labels.build.done is a string, that string is terminal.linear.labels.build.done is an object, only the production/final environment value is terminal (default: status:done). Intermediate env values such as status:on-dev and status:on-stg are not terminal and must keep the native Issue open.mcp__linear-server__save_issue: remove $CLAIMED (or $REVIEW if lisa:linear-evidence already moved it forward), add $DONE.$DONE is terminal, move the native Linear Issue state to the configured Done / Completed state. Resolve that state from project configuration if present; otherwise inspect the team workflow for a terminal state with state.type = "completed" and a name such as Done or Completed. If no terminal state can be resolved, record an Error and leave the labels as the source of truth — do not invent a state name.[claude-build-intake] comment: "Build complete. PR <URL> merged. Transitioned to $DONE." Include whether native closure was applied, already satisfied, skipped for an intermediate env, or unavailable for setup reasons.For any non-Success outcome, do NOT transition. The Issue sits where the agent left it — humans take it from there.
Stop immediately after the first claimed, skipped, blocked, held, or errored Issue. Later scheduler invocations process the remaining ready Issues.
## linear-build-intake summary
Team: <teamKey>
Cycle started: <ISO timestamp>
Cycle completed: <ISO timestamp>
Issues processed: <n>
- $DONE (build complete, PR merged): <n>
- <ID> <title> → PR <URL>
- PR open — awaiting merge (left at $CLAIMED for repair-intake): <n>
- <ID> <title> → PR <URL> (mergeStateStatus: <state>)
- Skipped (container — leaf-only-lifecycle): <n>
- <ID> <title> — build-ready on a parent with open child work; lifecycle-repair comment posted
- status:blocked (pre-flight verify failed): <n>
- <ID> <title> — see Issue comments
- Held (triage found ambiguities): <n>
- <ID> <title> — see Issue comments
- Errors: <n>
- <ID> <title> — <reason>
Total PRs opened: <n>
leaf-only-lifecycle rule's claim-time arm). The safe-block comment is idempotent — a re-entrant cycle does not re-post it.$CLAIMED set BEFORE agent invocation — no double-pickup.$READY, $CLAIMED, $DONE, plus terminal-only native state completion required by leaf-only-lifecycle. Every other label change (and non-terminal native state change) is owned by the agent or lisa:linear-evidence.$DONE label is applied, move the Linear Issue to a native completed state only when $DONE is the true terminal done value; intermediate env labels stay open / active.status:* label is present. Two simultaneously breaks the build queue.$DONE. If done is a map and env is ambiguous, fail loudly.Before this skill can run against a Linear team, the team must adopt the build-queue label convention. Using the defaults:
status:ready, status:in-progress, status:code-review, status:on-dev, status:done, status:blocked on the team (or workspace). If your project overrides any linear.labels.build.* role name in config, substitute the actual label names you configured.$READY label to Issues that are ready for development.$CLAIMED, $REVIEW, $DONE for Lisa — humans should not set them manually except to recover from an error.If the team hasn't adopted these labels, the first run exits with an adoption hint.
leaf-only-lifecycle rule, never claim a container — an Issue with open child work, or a childless Epic — even if it carries the build-ready label. Skip or safe-block it (Phase 3a); never silently implement a container.$CLAIMED transition is the signature of cycle ownership.lisa:linear-agent to do build work directly. The agent owns the per-Issue lifecycle.$DONE. Downstream labels are owned by QA / product / a future verification-intake skill.status:on-dev, status:on-stg, or configured equivalents). Native completion happens only at the terminal done value.lisa:linear-agent's pre-flight verify will catch it and relabel to status:blocked — don't try to fix the Issue from here.lisa:linear-agent (label it doesn't claim, missing PR URL on success, etc.), record as Error and surface — never assume.$DONE resolution. If done is a map and env is ambiguous, fail loudly.documentation
Onboard a user to the project via its LLM Wiki. Interviews the user about themselves in relation to the project, captures that to project-scoped memory only, then gives a guided tour of what the project is and sample questions they can ask. Use when someone is new to the project or asks to be onboarded. Read-mostly — it does not open PRs or write PII into the wiki.
documentation
Migrate an existing, hand-rolled wiki implementation onto the lisa-wiki kernel — phased and compatibility-first, with a strict no-loss guarantee. Use when adopting lisa-wiki in a repo that already has its own wiki/, ingest skills, docs, or roles. Renaming things into the canonical shape is fine; losing functionality or data is not. Ends by running /doctor.
development
Health-check the LLM Wiki. Reports orphan pages, contradictions, stale claims, broken internal links, missing index/log coverage, structure-manifest violations, and secret/tenant leaks. Use periodically or before hardening a wiki. Read-only — it reports findings, it does not fix them.
testing
Ingest source material into the LLM Wiki. With an argument (URL, file path, or prompt) it ingests that one source; with no argument it runs a full ingest across every enabled non-external-write source. Routes to the right connector, then runs the ordered pipeline (source note → synthesis → index → log → verify → state → commit/PR). Use whenever new knowledge should enter the wiki.