linear-cost-tuningClaude Skill
Optimize Linear API usage and manage costs effectively.
| name | linear-cost-tuning |
| description | Optimize Linear API usage and manage costs effectively. Use when reducing API calls, managing rate limits efficiently, or optimizing integration costs. Trigger with phrases like "linear cost", "reduce linear API calls", "linear efficiency", "linear API usage", "optimize linear costs". |
| allowed-tools | Read, Write, Edit, Grep |
| version | 1.0.0 |
| license | MIT |
| author | Jeremy Longshore <jeremy@intentsolutions.io> |
Linear Cost Tuning
Overview
Optimize Linear API usage to maximize efficiency and minimize costs.
Prerequisites
- Working Linear integration
- Monitoring in place
- Understanding of usage patterns
Cost Factors
API Request Costs
| Factor | Impact | Optimization Strategy |
|---|---|---|
| Request count | Direct rate limit | Batch operations |
| Query complexity | Complexity limit | Minimal field selection |
| Payload size | Bandwidth/latency | Pagination, filtering |
| Webhook volume | Processing costs | Event filtering |
Instructions
Step 1: Audit Current Usage
// lib/usage-tracker.ts interface UsageStats { requests: number; complexity: number; bytesTransferred: number; period: { start: Date; end: Date }; } class UsageTracker { private stats: UsageStats = { requests: 0, complexity: 0, bytesTransferred: 0, period: { start: new Date(), end: new Date() }, }; recordRequest(complexity: number, bytes: number): void { this.stats.requests++; this.stats.complexity += complexity; this.stats.bytesTransferred += bytes; this.stats.period.end = new Date(); } getStats(): UsageStats { return { ...this.stats }; } getDaily(): { avgRequestsPerHour: number; avgComplexityPerRequest: number; projectedMonthlyRequests: number; } { const hours = (this.stats.period.end.getTime() - this.stats.period.start.getTime()) / (1000 * 60 * 60); return { avgRequestsPerHour: this.stats.requests / Math.max(hours, 1), avgComplexityPerRequest: this.stats.complexity / Math.max(this.stats.requests, 1), projectedMonthlyRequests: (this.stats.requests / Math.max(hours, 1)) * 24 * 30, }; } reset(): void { this.stats = { requests: 0, complexity: 0, bytesTransferred: 0, period: { start: new Date(), end: new Date() }, }; } } export const usageTracker = new UsageTracker();
Step 2: Reduce Request Volume
Polling vs Webhooks:
// BAD: Polling every minute setInterval(async () => { const issues = await client.issues({ first: 100 }); await syncIssues(issues.nodes); }, 60000); // GOOD: Use webhooks for real-time updates // See linear-webhooks-events skill app.post("/webhooks/linear", async (req, res) => { const event = req.body; await handleEvent(event); res.sendStatus(200); });
Conditional Fetching:
// lib/conditional-fetch.ts interface ETagCache { data: any; etag: string; timestamp: Date; } const etagCache = new Map<string, ETagCache>(); async function fetchWithETag(key: string, fetcher: () => Promise<any>) { const cached = etagCache.get(key); // Only fetch if cache is stale (> 5 minutes) if (cached && Date.now() - cached.timestamp.getTime() < 5 * 60 * 1000) { return cached.data; } const data = await fetcher(); etagCache.set(key, { data, etag: JSON.stringify(data).slice(0, 50), // Simple hash timestamp: new Date(), }); return data; }
Step 3: Optimize Query Complexity
Calculate Complexity:
// Linear complexity estimation // - Each field costs 1 // - Each connection costs 1 + (first * child_complexity) // - Nested connections multiply // BAD: High complexity query (~500 complexity) const expensiveQuery = ` query { issues(first: 50) { nodes { id title assignee { name } labels { nodes { name } } comments(first: 10) { nodes { body user { name } } } } } } `; // GOOD: Low complexity query (~100 complexity) const cheapQuery = ` query { issues(first: 50) { nodes { id identifier title priority } } } `;
Step 4: Implement Request Coalescing
// lib/coalesce.ts class RequestCoalescer { private pending = new Map<string, Promise<any>>(); async execute<T>(key: string, fn: () => Promise<T>): Promise<T> { // If same request is already in flight, reuse it const existing = this.pending.get(key); if (existing) { return existing; } const promise = fn().finally(() => { this.pending.delete(key); }); this.pending.set(key, promise); return promise; } } const coalescer = new RequestCoalescer(); // Multiple simultaneous calls reuse the same request const [teams1, teams2, teams3] = await Promise.all([ coalescer.execute("teams", () => client.teams()), coalescer.execute("teams", () => client.teams()), // Reuses first request coalescer.execute("teams", () => client.teams()), // Reuses first request ]);
Step 5: Webhook Event Filtering
// Only process relevant events async function shouldProcessEvent(event: any): boolean { // Skip events from bots if (event.data?.actor?.isBot) return false; // Only process certain issue states if (event.type === "Issue" && event.action === "update") { const importantFields = ["state", "priority", "assignee"]; const changedFields = Object.keys(event.updatedFrom || {}); if (!changedFields.some(f => importantFields.includes(f))) { return false; // Skip trivial updates } } // Only process issues from specific teams const allowedTeams = ["ENG", "PROD"]; if (event.data?.team?.key && !allowedTeams.includes(event.data.team.key)) { return false; } return true; }
Step 6: Lazy Loading Pattern
// lib/lazy-client.ts class LazyLinearClient { private client: LinearClient; private teamsCache: any[] | null = null; private statesCache = new Map<string, any[]>(); constructor(apiKey: string) { this.client = new LinearClient({ apiKey }); } async getTeams() { if (!this.teamsCache) { const teams = await this.client.teams(); this.teamsCache = teams.nodes; } return this.teamsCache; } async getStatesForTeam(teamKey: string) { if (!this.statesCache.has(teamKey)) { const teams = await this.client.teams({ filter: { key: { eq: teamKey } }, }); const states = await teams.nodes[0].states(); this.statesCache.set(teamKey, states.nodes); } return this.statesCache.get(teamKey)!; } // Invalidate on known changes invalidateTeams() { this.teamsCache = null; this.statesCache.clear(); } }
Cost Reduction Checklist
- Replace polling with webhooks
- Implement request caching
- Use request coalescing
- Filter webhook events
- Minimize query complexity
- Batch related operations
- Use lazy loading for static data
- Monitor and track usage
Monitoring Dashboard
// Example metrics to track const metrics = { // Request metrics totalRequests: counter("linear_requests_total"), requestDuration: histogram("linear_request_duration_seconds"), complexityCost: histogram("linear_complexity_cost"), // Cache metrics cacheHits: counter("linear_cache_hits_total"), cacheMisses: counter("linear_cache_misses_total"), // Webhook metrics webhooksReceived: counter("linear_webhooks_received_total"), webhooksFiltered: counter("linear_webhooks_filtered_total"), };
Resources
Next Steps
Learn production architecture with linear-reference-architecture.
Similar Claude Skills & Agent Workflows
git-commit
Generate well-formatted git commit messages following conventional commit standards
code-review
Comprehensive code review assistant that analyzes code quality, security, and best practices
dsql
Build with Aurora DSQL - manage schemas, execute queries, and handle migrations with DSQL-specific requirements.
backend-dev-guidelines
Comprehensive backend development guide for Langfuse's Next.js 14/tRPC/Express/TypeScript monorepo.
Material Component Dev
FlowGram 物料组件开发指南 - 用于在 form-materials 包中创建新的物料组件
Create Node
用于在 FlowGram demo-free-layout 中创建新的自定义节点,支持简单节点(自动表单)和复杂节点(自定义 UI)