From linear-pack
Guides creating first Linear issue, listing teams, querying issues, and exploring states using @linear/sdk and GraphQL API. For initial setup, connection testing, and basic CRUD.
npx claudepluginhub jeremylongshore/claude-code-plugins-plus-skills --plugin linear-packThis skill is limited to using the following tools:
Create your first issue, query teams, and explore the Linear data model using the `@linear/sdk`. Linear's API is GraphQL-based -- the SDK wraps it with typed models, lazy-loaded relations, and pagination helpers.
Manages Linear issue lifecycle: create, update, transition states, build hierarchies, add labels/comments via SDK and GraphQL. For TypeScript/Node.js Linear users.
Interact with Linear project management via GraphQL API: query assigned/open issues by priority/identifier, fetch details, update status, create tickets, manage workflows. Use for Linear tickets, sprints, project tracking.
Manages Linear issues, projects, and teams using MCP tools, linear CLI, and GraphQL API. Create, update, query issues; handle labels, status, references, and backlogs.
Share bugs, ideas, or general feedback.
Create your first issue, query teams, and explore the Linear data model using the @linear/sdk. Linear's API is GraphQL-based -- the SDK wraps it with typed models, lazy-loaded relations, and pagination helpers.
@linear/sdk installed (npm install @linear/sdk)LINEAR_API_KEY environment variable set (starts with lin_api_)import { LinearClient } from "@linear/sdk";
const client = new LinearClient({ apiKey: process.env.LINEAR_API_KEY! });
// Get current authenticated user
const me = await client.viewer;
console.log(`Hello, ${me.name}! (${me.email})`);
// Get your organization
const org = await me.organization;
console.log(`Workspace: ${org.name}`);
Every issue in Linear belongs to a team. Teams have a short key (e.g., "ENG") used in identifiers like ENG-123.
const teams = await client.teams();
console.log("Your teams:");
for (const team of teams.nodes) {
console.log(` ${team.key} — ${team.name} (${team.id})`);
}
const team = teams.nodes[0];
const result = await client.createIssue({
teamId: team.id,
title: "Hello from Linear SDK!",
description: "This issue was created using the `@linear/sdk` TypeScript SDK.",
priority: 3, // 0=None, 1=Urgent, 2=High, 3=Medium, 4=Low
});
if (result.success) {
const issue = await result.issue;
console.log(`Created: ${issue?.identifier} — ${issue?.title}`);
console.log(`URL: ${issue?.url}`);
}
// Get recent issues from a team
const issues = await client.issues({
filter: {
team: { key: { eq: team.key } },
state: { type: { nin: ["completed", "canceled"] } },
},
first: 10,
});
console.log(`\nOpen issues in ${team.key}:`);
for (const issue of issues.nodes) {
const state = await issue.state;
console.log(` ${issue.identifier}: ${issue.title} [${state?.name}]`);
}
Each team has customizable workflow states organized by type: triage, backlog, unstarted, started, completed, canceled.
const states = await team.states();
console.log(`\nWorkflow states for ${team.key}:`);
for (const state of states.nodes) {
console.log(` ${state.name} (type: ${state.type}, position: ${state.position})`);
}
// Search for a specific issue by its human-readable identifier
const searchResults = await client.issueSearch("ENG-1");
const found = searchResults.nodes[0];
if (found) {
console.log(`\nFound: ${found.identifier}`);
console.log(` Title: ${found.title}`);
console.log(` Priority: ${found.priority}`);
console.log(` Created: ${found.createdAt}`);
const assignee = await found.assignee;
console.log(` Assignee: ${assignee?.name ?? "Unassigned"}`);
}
The SDK exposes the underlying GraphQL client for custom queries.
const response = await client.client.rawRequest(`
query TeamDashboard($teamKey: String!) {
teams(filter: { key: { eq: $teamKey } }) {
nodes {
name
key
issues(first: 5, orderBy: updatedAt) {
nodes {
identifier
title
priority
state { name type }
assignee { name }
}
}
}
}
}
`, { teamKey: "ENG" });
console.log(JSON.stringify(response.data, null, 2));
| Error | Cause | Solution |
|---|---|---|
Authentication required | Invalid API key | Regenerate at Settings > Account > API |
Entity not found | Invalid ID or no access | Use client.teams() first to get valid IDs |
Validation error | Missing required field | teamId and title are required for createIssue |
Cannot read properties of null | Accessing nullable relation | Use optional chaining: (await issue.assignee)?.name |
import { LinearClient } from "@linear/sdk";
async function main() {
const client = new LinearClient({ apiKey: process.env.LINEAR_API_KEY! });
const me = await client.viewer;
console.log(`Connected as ${me.name}\n`);
const teams = await client.teams();
const team = teams.nodes[0];
// Create issue
const result = await client.createIssue({
teamId: team.id,
title: "Hello from Linear SDK!",
description: "Testing the API integration.",
priority: 3,
});
if (result.success) {
const issue = await result.issue;
console.log(`Created: ${issue?.identifier} — ${issue?.url}`);
// Read it back
const fetched = await client.issue(issue!.id);
console.log(`Verified: ${fetched.title}`);
// Clean up
await fetched.delete();
console.log("Deleted test issue.");
}
}
main().catch(console.error);