linear-incident-runbookClaude Skill
Production incident response procedures for Linear integrations.
| name | linear-incident-runbook |
| description | Production incident response procedures for Linear integrations. Use when handling production issues, diagnosing outages, or responding to Linear-related incidents. Trigger with phrases like "linear incident", "linear outage", "linear production issue", "debug linear production", "linear down". |
| allowed-tools | Read, Write, Edit, Bash(curl:*), Grep |
| version | 1.0.0 |
| license | MIT |
| author | Jeremy Longshore <jeremy@intentsolutions.io> |
Linear Incident Runbook
Overview
Step-by-step procedures for handling production incidents with Linear integrations.
Prerequisites
- Production access credentials
- Monitoring dashboard access
- Communication channels configured
- Escalation paths defined
Incident Classification
| Severity | Description | Response Time | Examples |
|---|---|---|---|
| SEV1 | Complete outage | < 15 minutes | API unreachable, auth broken |
| SEV2 | Major degradation | < 30 minutes | High error rate, slow responses |
| SEV3 | Minor issues | < 2 hours | Some features affected |
| SEV4 | Low impact | < 24 hours | Cosmetic issues, warnings |
Immediate Actions
Step 1: Confirm the Issue
# Check Linear API status curl -s https://status.linear.app/api/v2/status.json | jq '.status' # Quick health check curl -s -H "Authorization: $LINEAR_API_KEY" \ -H "Content-Type: application/json" \ -d '{"query": "{ viewer { name } }"}' \ https://api.linear.app/graphql | jq # Check your application health endpoint curl -s https://yourapp.com/api/health | jq
Step 2: Gather Initial Information
// scripts/incident-info.ts import { LinearClient } from "@linear/sdk"; async function gatherIncidentInfo() { const client = new LinearClient({ apiKey: process.env.LINEAR_API_KEY! }); console.log("=== Linear Incident Information ===\n"); // 1. Test authentication console.log("1. Authentication:"); try { const viewer = await client.viewer; console.log(` Status: OK (${viewer.name})`); } catch (error) { console.log(` Status: FAILED - ${error}`); } // 2. Check teams access console.log("\n2. Team Access:"); try { const teams = await client.teams(); console.log(` Accessible teams: ${teams.nodes.length}`); } catch (error) { console.log(` Status: FAILED - ${error}`); } // 3. Test issue creation (dry run) console.log("\n3. Write Capability:"); try { const teams = await client.teams(); const result = await client.createIssue({ teamId: teams.nodes[0].id, title: "[INCIDENT TEST] Delete immediately", }); if (result.success) { const issue = await result.issue; await issue?.delete(); console.log(" Status: OK (created and deleted test issue)"); } } catch (error) { console.log(` Status: FAILED - ${error}`); } console.log("\n=== End Information ==="); } gatherIncidentInfo();
Runbook: API Authentication Failure
Symptoms
- All API calls returning 401/403
- "Authentication required" errors
- Sudden spike in auth errors
Diagnosis
# Test API key directly curl -I -H "Authorization: $LINEAR_API_KEY" \ https://api.linear.app/graphql # Check for key format issues echo $LINEAR_API_KEY | head -c 8 # Should output: lin_api_ # Verify key in secrets manager vault read secret/data/linear/production # or aws secretsmanager get-secret-value --secret-id linear/production
Resolution Steps
-
Verify API key is loaded correctly
# Check env var is set (don't print actual key) [ -n "$LINEAR_API_KEY" ] && echo "Key is set" || echo "Key is NOT set" -
Check if key was rotated/revoked
- Log into Linear dashboard
- Navigate to Settings > API > Personal API keys
- Verify key exists and is active
-
Generate new API key if needed
- Create new key in Linear dashboard
- Update secrets manager
- Restart affected services
-
Rollback if recent deployment
# Check last deployment git log --oneline -5 # Rollback to previous version git revert HEAD
Runbook: Rate Limiting Issues
Symptoms
- HTTP 429 responses
- "Rate limit exceeded" errors
- Degraded performance
Diagnosis
# Check current rate limit status curl -I -H "Authorization: $LINEAR_API_KEY" \ -H "Content-Type: application/json" \ -d '{"query": "{ viewer { name } }"}' \ https://api.linear.app/graphql 2>&1 | grep -i ratelimit # Check application metrics curl -s http://localhost:9090/api/v1/query?query=linear_rate_limit_remaining | jq
Resolution Steps
-
Identify rate limit cause
# Check request patterns grep "linear" /var/log/app/*.log | grep -E "[0-9]{4}-[0-9]{2}-[0-9]{2}" | wc -l -
Implement emergency throttling
// Emergency rate limiter const EMERGENCY_MODE = true; const MIN_DELAY_MS = 5000; async function emergencyThrottle<T>(fn: () => Promise<T>): Promise<T> { if (EMERGENCY_MODE) { await new Promise(r => setTimeout(r, MIN_DELAY_MS)); } return fn(); } -
Disable non-critical operations
- Stop background sync jobs
- Disable polling (if using)
- Queue non-urgent requests
-
Wait for rate limit reset
- Linear resets every minute
- Monitor X-RateLimit-Reset header
Runbook: Webhook Failures
Symptoms
- Events not being received
- Webhook signature validation failing
- Processing timeouts
Diagnosis
# Check webhook endpoint is reachable curl -I https://yourapp.com/api/webhooks/linear # Check recent webhook logs tail -100 /var/log/webhooks.log | grep linear # Verify webhook secret echo $LINEAR_WEBHOOK_SECRET | wc -c # Should be > 20 characters
Resolution Steps
-
Verify endpoint health
app.get("/api/webhooks/linear/health", (req, res) => { res.json({ status: "ok", timestamp: new Date().toISOString() }); }); -
Check signature verification
// Debug signature verification function debugVerifySignature(payload: string, signature: string): boolean { const secret = process.env.LINEAR_WEBHOOK_SECRET!; const expected = crypto.createHmac("sha256", secret).update(payload).digest("hex"); console.log("Debug: Received signature:", signature); console.log("Debug: Expected signature:", expected); console.log("Debug: Secret length:", secret.length); return signature === expected; } -
Recreate webhook if needed
- Go to Linear Settings > API > Webhooks
- Delete existing webhook
- Create new webhook with same URL
- Update webhook secret in secrets manager
Communication Templates
Initial Incident Announcement
**INCIDENT: Linear Integration Issue** Severity: SEVX Status: Investigating Impact: [Description of user impact] Start Time: [UTC timestamp] We are investigating issues with our Linear integration. Updates will follow.
Status Update
**UPDATE: Linear Integration Issue** Status: [Investigating/Identified/Mitigating/Resolved] Time: [UTC timestamp] Update: [What we know/did] Next Steps: [What we're doing next] ETA: [If known]
Resolution Notice
**RESOLVED: Linear Integration Issue** Duration: [X hours Y minutes] Root Cause: [Brief description] Impact: [What was affected] A full post-mortem will follow within 48 hours.
Post-Incident
Immediate Actions
[ ] Verify all systems are healthy
[ ] Clear any queued/stuck jobs
[ ] Validate data consistency
[ ] Notify stakeholders of resolution
Post-Mortem Checklist
[ ] Timeline of events
[ ] Root cause analysis
[ ] Impact assessment
[ ] What went well
[ ] What could be improved
[ ] Action items with owners
Resources
- Linear Status Page
- Linear API Documentation
- Internal: On-call runbook wiki
- Internal: Escalation contacts
Next Steps
Learn data handling patterns with linear-data-handling.
Similar Claude Skills & Agent Workflows
analyze-ci
Analyze failed GitHub Action jobs for a pull request.
sla-monitor-setup
Configure sla monitor setup operations.
github-project-setup
Configure github project setup operations.
vpc-network-setup
Configure vpc network setup operations.
memorystore-config
Configure memorystore config operations.
gke-cluster-config
Configure gke cluster config operations.