@stackmemoryai/stackmemory 0.3.6 → 0.3.8
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/agents/core/agent-task-manager.js +5 -5
- package/dist/agents/core/agent-task-manager.js.map +2 -2
- package/dist/agents/verifiers/base-verifier.js +2 -2
- package/dist/agents/verifiers/base-verifier.js.map +2 -2
- package/dist/agents/verifiers/formatter-verifier.js.map +2 -2
- package/dist/agents/verifiers/llm-judge.js.map +2 -2
- package/dist/cli/claude-sm.js +13 -13
- package/dist/cli/claude-sm.js.map +2 -2
- package/dist/cli/codex-sm.js +13 -13
- package/dist/cli/codex-sm.js.map +2 -2
- package/dist/cli/commands/agent.js.map +2 -2
- package/dist/cli/commands/chromadb.js +261 -46
- package/dist/cli/commands/chromadb.js.map +2 -2
- package/dist/cli/commands/clear.js +10 -3
- package/dist/cli/commands/clear.js.map +2 -2
- package/dist/cli/commands/config.js +43 -33
- package/dist/cli/commands/config.js.map +2 -2
- package/dist/cli/commands/context.js +13 -2
- package/dist/cli/commands/context.js.map +2 -2
- package/dist/cli/commands/dashboard.js +41 -13
- package/dist/cli/commands/dashboard.js.map +2 -2
- package/dist/cli/commands/gc.js +251 -0
- package/dist/cli/commands/gc.js.map +7 -0
- package/dist/cli/commands/handoff.js +12 -1
- package/dist/cli/commands/handoff.js.map +2 -2
- package/dist/cli/commands/infinite-storage.js +92 -40
- package/dist/cli/commands/infinite-storage.js.map +2 -2
- package/dist/cli/commands/linear-create.js +49 -10
- package/dist/cli/commands/linear-create.js.map +2 -2
- package/dist/cli/commands/linear-list.js +45 -11
- package/dist/cli/commands/linear-list.js.map +2 -2
- package/dist/cli/commands/linear-migrate.js +29 -5
- package/dist/cli/commands/linear-migrate.js.map +2 -2
- package/dist/cli/commands/linear-test.js +26 -7
- package/dist/cli/commands/linear-test.js.map +2 -2
- package/dist/cli/commands/linear-unified.js +350 -0
- package/dist/cli/commands/linear-unified.js.map +7 -0
- package/dist/cli/commands/linear.js +17 -6
- package/dist/cli/commands/linear.js.map +2 -2
- package/dist/cli/commands/monitor.js.map +2 -2
- package/dist/cli/commands/onboard.js +35 -8
- package/dist/cli/commands/onboard.js.map +2 -2
- package/dist/cli/commands/quality.js +2 -7
- package/dist/cli/commands/quality.js.map +2 -2
- package/dist/cli/commands/search.js.map +2 -2
- package/dist/cli/commands/session.js +23 -6
- package/dist/cli/commands/session.js.map +2 -2
- package/dist/cli/commands/skills.js +84 -28
- package/dist/cli/commands/skills.js.map +2 -2
- package/dist/cli/commands/storage.js +119 -38
- package/dist/cli/commands/storage.js.map +2 -2
- package/dist/cli/commands/tasks.js.map +2 -2
- package/dist/cli/commands/tui.js +13 -2
- package/dist/cli/commands/tui.js.map +2 -2
- package/dist/cli/commands/webhook.js +71 -21
- package/dist/cli/commands/webhook.js.map +2 -2
- package/dist/cli/commands/workflow.js +11 -7
- package/dist/cli/commands/workflow.js.map +2 -2
- package/dist/cli/commands/worktree.js +34 -13
- package/dist/cli/commands/worktree.js.map +2 -2
- package/dist/cli/index.js +7 -5
- package/dist/cli/index.js.map +2 -2
- package/dist/core/config/config-manager.js.map +2 -2
- package/dist/core/config/types.js.map +1 -1
- package/dist/core/context/auto-context.js +10 -6
- package/dist/core/context/auto-context.js.map +2 -2
- package/dist/core/context/compaction-handler.js.map +2 -2
- package/dist/core/context/context-bridge.js.map +2 -2
- package/dist/core/context/dual-stack-manager.js.map +2 -2
- package/dist/core/context/frame-database.js +13 -3
- package/dist/core/context/frame-database.js.map +2 -2
- package/dist/core/context/frame-digest.js +7 -5
- package/dist/core/context/frame-digest.js.map +2 -2
- package/dist/core/context/frame-handoff-manager.js.map +2 -2
- package/dist/core/context/frame-manager.js +12 -1
- package/dist/core/context/frame-manager.js.map +2 -2
- package/dist/core/context/frame-stack.js +16 -5
- package/dist/core/context/frame-stack.js.map +2 -2
- package/dist/core/context/incremental-gc.js +286 -0
- package/dist/core/context/incremental-gc.js.map +7 -0
- package/dist/core/context/index.js.map +1 -1
- package/dist/core/context/permission-manager.js +12 -1
- package/dist/core/context/permission-manager.js.map +2 -2
- package/dist/core/context/refactored-frame-manager.js +12 -3
- package/dist/core/context/refactored-frame-manager.js.map +2 -2
- package/dist/core/context/shared-context-layer.js +16 -3
- package/dist/core/context/shared-context-layer.js.map +2 -2
- package/dist/core/context/stack-merge-resolver.js.map +2 -2
- package/dist/core/context/validation.js.map +2 -2
- package/dist/core/database/batch-operations.js +112 -86
- package/dist/core/database/batch-operations.js.map +2 -2
- package/dist/core/database/connection-pool.js.map +2 -2
- package/dist/core/database/migration-manager.js.map +2 -2
- package/dist/core/database/paradedb-adapter.js.map +2 -2
- package/dist/core/database/query-cache.js +19 -9
- package/dist/core/database/query-cache.js.map +2 -2
- package/dist/core/database/query-router.js.map +2 -2
- package/dist/core/database/sqlite-adapter.js +1 -1
- package/dist/core/database/sqlite-adapter.js.map +2 -2
- package/dist/core/digest/enhanced-hybrid-digest.js +8 -2
- package/dist/core/digest/enhanced-hybrid-digest.js.map +2 -2
- package/dist/core/errors/recovery.js +9 -2
- package/dist/core/errors/recovery.js.map +2 -2
- package/dist/core/frame/workflow-templates-stub.js.map +1 -1
- package/dist/core/frame/workflow-templates.js +40 -1
- package/dist/core/frame/workflow-templates.js.map +2 -2
- package/dist/core/merge/resolution-engine.js.map +2 -2
- package/dist/core/monitoring/error-handler.js.map +2 -2
- package/dist/core/monitoring/logger.js +19 -3
- package/dist/core/monitoring/logger.js.map +2 -2
- package/dist/core/monitoring/metrics.js +13 -2
- package/dist/core/monitoring/metrics.js.map +2 -2
- package/dist/core/monitoring/progress-tracker.js +12 -1
- package/dist/core/monitoring/progress-tracker.js.map +2 -2
- package/dist/core/monitoring/session-monitor.js.map +2 -2
- package/dist/core/performance/context-cache.js.map +2 -2
- package/dist/core/performance/lazy-context-loader.js +24 -20
- package/dist/core/performance/lazy-context-loader.js.map +2 -2
- package/dist/core/performance/monitor.js.map +2 -2
- package/dist/core/performance/optimized-frame-context.js +27 -12
- package/dist/core/performance/optimized-frame-context.js.map +2 -2
- package/dist/core/performance/performance-benchmark.js +10 -6
- package/dist/core/performance/performance-benchmark.js.map +2 -2
- package/dist/core/performance/performance-profiler.js +63 -15
- package/dist/core/performance/performance-profiler.js.map +2 -2
- package/dist/core/performance/streaming-jsonl-parser.js +5 -1
- package/dist/core/performance/streaming-jsonl-parser.js.map +2 -2
- package/dist/core/persistence/postgres-adapter.js.map +2 -2
- package/dist/core/projects/project-manager.js +14 -20
- package/dist/core/projects/project-manager.js.map +2 -2
- package/dist/core/retrieval/context-retriever.js.map +2 -2
- package/dist/core/retrieval/graph-retrieval.js.map +2 -2
- package/dist/core/retrieval/llm-context-retrieval.js.map +2 -2
- package/dist/core/retrieval/retrieval-benchmarks.js.map +2 -2
- package/dist/core/retrieval/summary-generator.js.map +2 -2
- package/dist/core/session/clear-survival-stub.js +5 -1
- package/dist/core/session/clear-survival-stub.js.map +2 -2
- package/dist/core/session/clear-survival.js +35 -0
- package/dist/core/session/clear-survival.js.map +2 -2
- package/dist/core/session/handoff-generator.js.map +2 -2
- package/dist/core/session/index.js.map +1 -1
- package/dist/core/session/session-manager.js +16 -5
- package/dist/core/session/session-manager.js.map +2 -2
- package/dist/core/skills/skill-storage.js +13 -2
- package/dist/core/skills/skill-storage.js.map +2 -2
- package/dist/core/storage/chromadb-adapter.js +6 -2
- package/dist/core/storage/chromadb-adapter.js.map +2 -2
- package/dist/core/storage/chromadb-simple.js +17 -5
- package/dist/core/storage/chromadb-simple.js.map +2 -2
- package/dist/core/storage/infinite-storage.js +109 -46
- package/dist/core/storage/infinite-storage.js.map +2 -2
- package/dist/core/storage/railway-optimized-storage.js +67 -30
- package/dist/core/storage/railway-optimized-storage.js.map +2 -2
- package/dist/core/storage/remote-storage.js +53 -24
- package/dist/core/storage/remote-storage.js.map +2 -2
- package/dist/core/trace/cli-trace-wrapper.js +25 -7
- package/dist/core/trace/cli-trace-wrapper.js.map +2 -2
- package/dist/core/trace/db-trace-wrapper.js +96 -68
- package/dist/core/trace/db-trace-wrapper.js.map +2 -2
- package/dist/core/trace/debug-trace.js +44 -16
- package/dist/core/trace/debug-trace.js.map +2 -2
- package/dist/core/trace/index.js +50 -35
- package/dist/core/trace/index.js.map +2 -2
- package/dist/core/trace/linear-api-wrapper.js +10 -5
- package/dist/core/trace/linear-api-wrapper.js.map +2 -2
- package/dist/core/trace/trace-demo.js +26 -11
- package/dist/core/trace/trace-demo.js.map +2 -2
- package/dist/core/trace/trace-detector.js +9 -2
- package/dist/core/trace/trace-detector.js.map +2 -2
- package/dist/core/trace/trace-store.js.map +2 -2
- package/dist/core/trace/types.js.map +1 -1
- package/dist/core/utils/compression.js.map +1 -1
- package/dist/core/utils/update-checker.js.map +2 -2
- package/dist/core/worktree/worktree-manager.js +18 -7
- package/dist/core/worktree/worktree-manager.js.map +2 -2
- package/dist/features/analytics/api/analytics-api.js.map +2 -2
- package/dist/features/analytics/core/analytics-service.js +12 -1
- package/dist/features/analytics/core/analytics-service.js.map +2 -2
- package/dist/features/analytics/queries/metrics-queries.js +1 -1
- package/dist/features/analytics/queries/metrics-queries.js.map +2 -2
- package/dist/features/tasks/pebbles-task-store.js.map +2 -2
- package/dist/features/tui/components/analytics-panel.js +36 -15
- package/dist/features/tui/components/analytics-panel.js.map +2 -2
- package/dist/features/tui/components/pr-tracker.js +19 -7
- package/dist/features/tui/components/pr-tracker.js.map +2 -2
- package/dist/features/tui/components/session-monitor.js +22 -9
- package/dist/features/tui/components/session-monitor.js.map +2 -2
- package/dist/features/tui/components/subagent-fleet.js +20 -13
- package/dist/features/tui/components/subagent-fleet.js.map +2 -2
- package/dist/features/tui/components/task-board.js +666 -2
- package/dist/features/tui/components/task-board.js.map +2 -2
- package/dist/features/tui/index.js +16 -5
- package/dist/features/tui/index.js.map +2 -2
- package/dist/features/tui/services/data-service.js +30 -15
- package/dist/features/tui/services/data-service.js.map +2 -2
- package/dist/features/tui/services/linear-task-reader.js +3 -1
- package/dist/features/tui/services/linear-task-reader.js.map +2 -2
- package/dist/features/tui/services/websocket-client.js +16 -3
- package/dist/features/tui/services/websocket-client.js.map +2 -2
- package/dist/features/tui/terminal-compat.js +33 -18
- package/dist/features/tui/terminal-compat.js.map +2 -2
- package/dist/features/web/client/stores/task-store.js.map +2 -2
- package/dist/features/web/server/index.js +31 -12
- package/dist/features/web/server/index.js.map +2 -2
- package/dist/integrations/claude-code/enhanced-pre-clear-hooks.js.map +2 -2
- package/dist/integrations/claude-code/lifecycle-hooks.js.map +2 -2
- package/dist/integrations/claude-code/post-task-hooks.js.map +2 -2
- package/dist/integrations/linear/auth.js +17 -6
- package/dist/integrations/linear/auth.js.map +2 -2
- package/dist/integrations/linear/auto-sync.js.map +2 -2
- package/dist/integrations/linear/client.js.map +2 -2
- package/dist/integrations/linear/config.js.map +2 -2
- package/dist/integrations/linear/migration.js.map +2 -2
- package/dist/integrations/linear/oauth-server.js +13 -2
- package/dist/integrations/linear/oauth-server.js.map +2 -2
- package/dist/integrations/linear/rest-client.js.map +2 -2
- package/dist/integrations/linear/sync-enhanced.js +202 -0
- package/dist/integrations/linear/sync-enhanced.js.map +7 -0
- package/dist/integrations/linear/sync-manager.js.map +2 -2
- package/dist/integrations/linear/sync-service.js +24 -14
- package/dist/integrations/linear/sync-service.js.map +2 -2
- package/dist/integrations/linear/sync.js +196 -3
- package/dist/integrations/linear/sync.js.map +2 -2
- package/dist/integrations/linear/unified-sync.js +560 -0
- package/dist/integrations/linear/unified-sync.js.map +7 -0
- package/dist/integrations/linear/webhook-handler.js +12 -1
- package/dist/integrations/linear/webhook-handler.js.map +2 -2
- package/dist/integrations/linear/webhook-server.js +29 -19
- package/dist/integrations/linear/webhook-server.js.map +2 -2
- package/dist/integrations/linear/webhook.js +12 -1
- package/dist/integrations/linear/webhook.js.map +2 -2
- package/dist/integrations/mcp/handlers/context-handlers.js.map +2 -2
- package/dist/integrations/mcp/handlers/linear-handlers.js.map +2 -2
- package/dist/integrations/mcp/handlers/skill-handlers.js +13 -2
- package/dist/integrations/mcp/handlers/skill-handlers.js.map +2 -2
- package/dist/integrations/mcp/handlers/task-handlers.js.map +2 -2
- package/dist/integrations/mcp/handlers/trace-handlers.js.map +2 -2
- package/dist/integrations/mcp/middleware/tool-scoring.js.map +2 -2
- package/dist/integrations/mcp/refactored-server.js +15 -4
- package/dist/integrations/mcp/refactored-server.js.map +2 -2
- package/dist/integrations/mcp/server.js +12 -1
- package/dist/integrations/mcp/server.js.map +2 -2
- package/dist/integrations/mcp/tool-definitions.js.map +2 -2
- package/dist/integrations/pg-aiguide/embedding-provider.js +13 -2
- package/dist/integrations/pg-aiguide/embedding-provider.js.map +2 -2
- package/dist/integrations/pg-aiguide/semantic-search.js.map +2 -2
- package/dist/mcp/stackmemory-mcp-server.js +1 -1
- package/dist/mcp/stackmemory-mcp-server.js.map +2 -2
- package/dist/middleware/exponential-rate-limiter.js.map +2 -2
- package/dist/servers/production/auth-middleware.js +13 -2
- package/dist/servers/production/auth-middleware.js.map +2 -2
- package/dist/servers/railway/index.js +22 -11
- package/dist/servers/railway/index.js.map +2 -2
- package/dist/services/config-service.js +6 -7
- package/dist/services/config-service.js.map +2 -2
- package/dist/services/context-service.js +11 -12
- package/dist/services/context-service.js.map +2 -2
- package/dist/skills/claude-skills.js +108 -3
- package/dist/skills/claude-skills.js.map +2 -2
- package/dist/skills/dashboard-launcher.js.map +2 -2
- package/dist/skills/repo-ingestion-skill.js +561 -0
- package/dist/skills/repo-ingestion-skill.js.map +7 -0
- package/dist/utils/env.js +46 -0
- package/dist/utils/env.js.map +7 -0
- package/dist/utils/logger.js +1 -1
- package/dist/utils/logger.js.map +2 -2
- package/package.json +5 -1
|
@@ -2,12 +2,25 @@ import { Command } from "commander";
|
|
|
2
2
|
import chalk from "chalk";
|
|
3
3
|
import ora from "ora";
|
|
4
4
|
import Table from "cli-table3";
|
|
5
|
-
import {
|
|
5
|
+
import {
|
|
6
|
+
InfiniteStorageSystem
|
|
7
|
+
} from "../../core/storage/infinite-storage.js";
|
|
6
8
|
import { FrameManager } from "../../core/context/frame-manager.js";
|
|
7
9
|
import { Logger } from "../../core/monitoring/logger.js";
|
|
8
10
|
import dotenv from "dotenv";
|
|
9
11
|
import path from "path";
|
|
10
12
|
import { fileURLToPath } from "url";
|
|
13
|
+
function getEnv(key, defaultValue) {
|
|
14
|
+
const value = process.env[key];
|
|
15
|
+
if (value === void 0) {
|
|
16
|
+
if (defaultValue !== void 0) return defaultValue;
|
|
17
|
+
throw new Error(`Environment variable ${key} is required`);
|
|
18
|
+
}
|
|
19
|
+
return value;
|
|
20
|
+
}
|
|
21
|
+
function getOptionalEnv(key) {
|
|
22
|
+
return process.env[key];
|
|
23
|
+
}
|
|
11
24
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
|
12
25
|
dotenv.config({
|
|
13
26
|
path: path.join(__dirname, "../../../.env"),
|
|
@@ -22,19 +35,19 @@ function createInfiniteStorageCommand() {
|
|
|
22
35
|
try {
|
|
23
36
|
const config = {
|
|
24
37
|
redis: {
|
|
25
|
-
url: options.redisUrl || process.env
|
|
38
|
+
url: options.redisUrl || process.env["REDIS_URL"] || "redis://localhost:6379",
|
|
26
39
|
ttlSeconds: 3600,
|
|
27
|
-
maxMemoryMB: parseInt(process.env
|
|
40
|
+
maxMemoryMB: parseInt(process.env["REDIS_MAX_MEMORY_MB"] || "512")
|
|
28
41
|
},
|
|
29
42
|
timeseries: {
|
|
30
|
-
connectionString: options.timeseriesUrl || process.env
|
|
43
|
+
connectionString: options.timeseriesUrl || process.env["TIMESERIES_URL"] || "",
|
|
31
44
|
retentionDays: 30
|
|
32
45
|
},
|
|
33
46
|
s3: {
|
|
34
|
-
bucket: options.s3Bucket || process.env
|
|
35
|
-
region: options.s3Region || process.env
|
|
36
|
-
accessKeyId: process.env
|
|
37
|
-
secretAccessKey: process.env
|
|
47
|
+
bucket: options.s3Bucket || process.env["S3_BUCKET"] || "",
|
|
48
|
+
region: options.s3Region || process.env["AWS_REGION"] || "us-east-1",
|
|
49
|
+
accessKeyId: process.env["AWS_ACCESS_KEY_ID"],
|
|
50
|
+
secretAccessKey: process.env["AWS_SECRET_ACCESS_KEY"]
|
|
38
51
|
},
|
|
39
52
|
tiers: []
|
|
40
53
|
};
|
|
@@ -48,23 +61,28 @@ function createInfiniteStorageCommand() {
|
|
|
48
61
|
console.log(" Archive (Glacier): > 30 days, 1h latency");
|
|
49
62
|
const envPath = path.join(__dirname, "../../../.env");
|
|
50
63
|
const updates = [];
|
|
51
|
-
if (!process.env
|
|
64
|
+
if (!process.env["REDIS_URL"] && options.redisUrl) {
|
|
52
65
|
updates.push(`REDIS_URL=${options.redisUrl}`);
|
|
53
66
|
}
|
|
54
|
-
if (!process.env
|
|
67
|
+
if (!process.env["TIMESERIES_URL"] && options.timeseriesUrl) {
|
|
55
68
|
updates.push(`TIMESERIES_URL=${options.timeseriesUrl}`);
|
|
56
69
|
}
|
|
57
|
-
if (!process.env
|
|
70
|
+
if (!process.env["S3_BUCKET"] && options.s3Bucket) {
|
|
58
71
|
updates.push(`S3_BUCKET=${options.s3Bucket}`);
|
|
59
72
|
}
|
|
60
73
|
if (updates.length > 0) {
|
|
61
74
|
const fs = await import("fs");
|
|
62
|
-
fs.appendFileSync(
|
|
75
|
+
fs.appendFileSync(
|
|
76
|
+
envPath,
|
|
77
|
+
"\n# Infinite Storage Configuration\n" + updates.join("\n") + "\n"
|
|
78
|
+
);
|
|
63
79
|
}
|
|
64
80
|
} catch (error) {
|
|
65
81
|
spinner.fail("Failed to initialize storage");
|
|
66
82
|
logger.error("Initialization error", error);
|
|
67
|
-
console.error(
|
|
83
|
+
console.error(
|
|
84
|
+
chalk.red(error instanceof Error ? error.message : "Unknown error")
|
|
85
|
+
);
|
|
68
86
|
}
|
|
69
87
|
});
|
|
70
88
|
storage.command("store").description("Store current frames in infinite storage").option("--project <name>", "Project name").option("--user <id>", "User ID").action(async (options) => {
|
|
@@ -75,7 +93,7 @@ function createInfiniteStorageCommand() {
|
|
|
75
93
|
await storage2.initialize();
|
|
76
94
|
const frameManager = new FrameManager();
|
|
77
95
|
const frames = frameManager.getAllFrames();
|
|
78
|
-
const userId = options.user || process.env
|
|
96
|
+
const userId = options.user || process.env["USER"] || "default";
|
|
79
97
|
for (const frame of frames) {
|
|
80
98
|
await storage2.storeFrame(frame, userId);
|
|
81
99
|
}
|
|
@@ -83,7 +101,9 @@ function createInfiniteStorageCommand() {
|
|
|
83
101
|
} catch (error) {
|
|
84
102
|
spinner.fail("Failed to store frames");
|
|
85
103
|
logger.error("Store error", error);
|
|
86
|
-
console.error(
|
|
104
|
+
console.error(
|
|
105
|
+
chalk.red(error instanceof Error ? error.message : "Unknown error")
|
|
106
|
+
);
|
|
87
107
|
}
|
|
88
108
|
});
|
|
89
109
|
storage.command("retrieve <frameId>").description("Retrieve a frame from storage").option("--user <id>", "User ID").action(async (frameId, options) => {
|
|
@@ -92,7 +112,7 @@ function createInfiniteStorageCommand() {
|
|
|
92
112
|
const config = getStorageConfig();
|
|
93
113
|
const storage2 = new InfiniteStorageSystem(config);
|
|
94
114
|
await storage2.initialize();
|
|
95
|
-
const userId = options.user || process.env
|
|
115
|
+
const userId = options.user || process.env["USER"] || "default";
|
|
96
116
|
const frame = await storage2.retrieveFrame(frameId, userId);
|
|
97
117
|
spinner.stop();
|
|
98
118
|
if (frame) {
|
|
@@ -104,7 +124,9 @@ function createInfiniteStorageCommand() {
|
|
|
104
124
|
} catch (error) {
|
|
105
125
|
spinner.fail("Failed to retrieve frame");
|
|
106
126
|
logger.error("Retrieve error", error);
|
|
107
|
-
console.error(
|
|
127
|
+
console.error(
|
|
128
|
+
chalk.red(error instanceof Error ? error.message : "Unknown error")
|
|
129
|
+
);
|
|
108
130
|
}
|
|
109
131
|
});
|
|
110
132
|
storage.command("metrics").description("Show storage system metrics").action(async () => {
|
|
@@ -130,20 +152,30 @@ function createInfiniteStorageCommand() {
|
|
|
130
152
|
console.log(table.toString());
|
|
131
153
|
if (Object.keys(metrics.tierDistribution).length > 0) {
|
|
132
154
|
console.log("\nTier Distribution:");
|
|
133
|
-
for (const [tier, count] of Object.entries(
|
|
134
|
-
|
|
155
|
+
for (const [tier, count] of Object.entries(
|
|
156
|
+
metrics.tierDistribution
|
|
157
|
+
)) {
|
|
158
|
+
const percentage = (count / metrics.totalObjects * 100).toFixed(
|
|
159
|
+
1
|
|
160
|
+
);
|
|
135
161
|
console.log(` ${tier}: ${count} objects (${percentage}%)`);
|
|
136
162
|
}
|
|
137
163
|
}
|
|
138
164
|
console.log(chalk.cyan("\n\u{1F3AF} STA-287 Performance Targets:"));
|
|
139
165
|
const p50Target = metrics.p50LatencyMs <= 50;
|
|
140
166
|
const p99Target = metrics.p99LatencyMs <= 500;
|
|
141
|
-
console.log(
|
|
142
|
-
|
|
167
|
+
console.log(
|
|
168
|
+
` P50 \u2264 50ms: ${p50Target ? chalk.green("\u2705 PASS") : chalk.red("\u274C FAIL")} (${metrics.p50LatencyMs}ms)`
|
|
169
|
+
);
|
|
170
|
+
console.log(
|
|
171
|
+
` P99 \u2264 500ms: ${p99Target ? chalk.green("\u2705 PASS") : chalk.red("\u274C FAIL")} (${metrics.p99LatencyMs}ms)`
|
|
172
|
+
);
|
|
143
173
|
} catch (error) {
|
|
144
174
|
spinner.fail("Failed to get metrics");
|
|
145
175
|
logger.error("Metrics error", error);
|
|
146
|
-
console.error(
|
|
176
|
+
console.error(
|
|
177
|
+
chalk.red(error instanceof Error ? error.message : "Unknown error")
|
|
178
|
+
);
|
|
147
179
|
}
|
|
148
180
|
});
|
|
149
181
|
storage.command("migrate").description("Manually trigger tier migration").action(async () => {
|
|
@@ -157,7 +189,9 @@ function createInfiniteStorageCommand() {
|
|
|
157
189
|
} catch (error) {
|
|
158
190
|
spinner.fail("Migration failed");
|
|
159
191
|
logger.error("Migration error", error);
|
|
160
|
-
console.error(
|
|
192
|
+
console.error(
|
|
193
|
+
chalk.red(error instanceof Error ? error.message : "Unknown error")
|
|
194
|
+
);
|
|
161
195
|
}
|
|
162
196
|
});
|
|
163
197
|
storage.command("status").description("Check storage system status").action(async () => {
|
|
@@ -183,26 +217,44 @@ function createInfiniteStorageCommand() {
|
|
|
183
217
|
if (config.timeseries?.connectionString) {
|
|
184
218
|
try {
|
|
185
219
|
const { Pool } = await import("pg");
|
|
186
|
-
const pool = new Pool({
|
|
220
|
+
const pool = new Pool({
|
|
221
|
+
connectionString: config.timeseries.connectionString
|
|
222
|
+
});
|
|
187
223
|
await pool.query("SELECT 1");
|
|
188
224
|
await pool.end();
|
|
189
|
-
console.log(
|
|
225
|
+
console.log(
|
|
226
|
+
"TimeSeries DB (Warm Tier): " + chalk.green("\u2705 Connected")
|
|
227
|
+
);
|
|
190
228
|
} catch {
|
|
191
|
-
console.log(
|
|
229
|
+
console.log(
|
|
230
|
+
"TimeSeries DB (Warm Tier): " + chalk.red("\u274C Not connected")
|
|
231
|
+
);
|
|
192
232
|
}
|
|
193
233
|
} else {
|
|
194
|
-
console.log(
|
|
234
|
+
console.log(
|
|
235
|
+
"TimeSeries DB (Warm Tier): " + chalk.yellow("\u26A0\uFE0F Not configured")
|
|
236
|
+
);
|
|
195
237
|
}
|
|
196
238
|
if (config.s3?.bucket) {
|
|
197
|
-
console.log(
|
|
239
|
+
console.log(
|
|
240
|
+
`S3 (Cold/Archive Tier): ${chalk.green("\u2705")} Bucket: ${config.s3.bucket}`
|
|
241
|
+
);
|
|
198
242
|
} else {
|
|
199
|
-
console.log(
|
|
243
|
+
console.log(
|
|
244
|
+
"S3 (Cold/Archive Tier): " + chalk.yellow("\u26A0\uFE0F Not configured")
|
|
245
|
+
);
|
|
200
246
|
}
|
|
201
|
-
console.log(
|
|
247
|
+
console.log(
|
|
248
|
+
"\n" + chalk.gray(
|
|
249
|
+
"Configure missing tiers with: stackmemory infinite-storage init"
|
|
250
|
+
)
|
|
251
|
+
);
|
|
202
252
|
} catch (error) {
|
|
203
253
|
spinner.fail("Failed to check status");
|
|
204
254
|
logger.error("Status error", error);
|
|
205
|
-
console.error(
|
|
255
|
+
console.error(
|
|
256
|
+
chalk.red(error instanceof Error ? error.message : "Unknown error")
|
|
257
|
+
);
|
|
206
258
|
}
|
|
207
259
|
});
|
|
208
260
|
return storage;
|
|
@@ -210,19 +262,19 @@ function createInfiniteStorageCommand() {
|
|
|
210
262
|
function getStorageConfig() {
|
|
211
263
|
return {
|
|
212
264
|
redis: {
|
|
213
|
-
url: process.env
|
|
214
|
-
ttlSeconds: parseInt(process.env
|
|
215
|
-
maxMemoryMB: parseInt(process.env
|
|
265
|
+
url: process.env["REDIS_URL"] || "redis://localhost:6379",
|
|
266
|
+
ttlSeconds: parseInt(process.env["REDIS_TTL"] || "3600"),
|
|
267
|
+
maxMemoryMB: parseInt(process.env["REDIS_MAX_MEMORY_MB"] || "512")
|
|
216
268
|
},
|
|
217
269
|
timeseries: {
|
|
218
|
-
connectionString: process.env
|
|
219
|
-
retentionDays: parseInt(process.env
|
|
270
|
+
connectionString: process.env["TIMESERIES_URL"] || "",
|
|
271
|
+
retentionDays: parseInt(process.env["TIMESERIES_RETENTION_DAYS"] || "30")
|
|
220
272
|
},
|
|
221
273
|
s3: {
|
|
222
|
-
bucket: process.env
|
|
223
|
-
region: process.env
|
|
224
|
-
accessKeyId: process.env
|
|
225
|
-
secretAccessKey: process.env
|
|
274
|
+
bucket: process.env["S3_BUCKET"] || "",
|
|
275
|
+
region: process.env["AWS_REGION"] || "us-east-1",
|
|
276
|
+
accessKeyId: process.env["AWS_ACCESS_KEY_ID"],
|
|
277
|
+
secretAccessKey: process.env["AWS_SECRET_ACCESS_KEY"]
|
|
226
278
|
},
|
|
227
279
|
tiers: []
|
|
228
280
|
};
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/cli/commands/infinite-storage.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * CLI commands for Infinite Storage System management\n */\n\nimport { Command } from 'commander';\nimport chalk from 'chalk';\nimport ora from 'ora';\nimport Table from 'cli-table3';\nimport { InfiniteStorageSystem, StorageConfig } from '../../core/storage/infinite-storage.js';\nimport { FrameManager } from '../../core/context/frame-manager.js';\nimport { Logger } from '../../core/monitoring/logger.js';\nimport dotenv from 'dotenv';\nimport path from 'path';\nimport { fileURLToPath } from 'url';\n\nconst __dirname = path.dirname(fileURLToPath(import.meta.url));\n\n// Load environment variables\ndotenv.config({ \n path: path.join(__dirname, '../../../.env'),\n override: true,\n silent: true\n});\n\nconst logger = new Logger('InfiniteStorage-CLI');\n\nexport function createInfiniteStorageCommand(): Command {\n const storage = new Command('infinite-storage')\n .description('Manage infinite storage system with tiered storage')\n .alias('storage');\n\n // Initialize storage system\n storage\n .command('init')\n .description('Initialize infinite storage system')\n .option('--redis-url <url>', 'Redis connection URL')\n .option('--timeseries-url <url>', 'TimeSeries DB connection URL')\n .option('--s3-bucket <bucket>', 'S3 bucket name')\n .option('--s3-region <region>', 'S3 region', 'us-east-1')\n .action(async (options) => {\n const spinner = ora('Initializing infinite storage...').start();\n\n try {\n const config: StorageConfig = {\n redis: {\n url: options.redisUrl || process.env.REDIS_URL || 'redis://localhost:6379',\n ttlSeconds: 3600,\n maxMemoryMB: parseInt(process.env.REDIS_MAX_MEMORY_MB || '512'),\n },\n timeseries: {\n connectionString: options.timeseriesUrl || process.env.TIMESERIES_URL || '',\n retentionDays: 30,\n },\n s3: {\n bucket: options.s3Bucket || process.env.S3_BUCKET || '',\n region: options.s3Region || process.env.AWS_REGION || 'us-east-1',\n accessKeyId: process.env.AWS_ACCESS_KEY_ID,\n secretAccessKey: process.env.AWS_SECRET_ACCESS_KEY,\n },\n tiers: [],\n };\n\n const storage = new InfiniteStorageSystem(config);\n await storage.initialize();\n\n spinner.succeed('Infinite storage initialized');\n \n console.log(chalk.green('\\n\u2705 Storage Tiers Configured:'));\n console.log(' Hot (Redis): < 1 hour, 5ms latency');\n console.log(' Warm (TimeSeries): 1h - 7 days, 50ms latency');\n console.log(' Cold (S3): 7 - 30 days, 100ms latency');\n console.log(' Archive (Glacier): > 30 days, 1h latency');\n \n // Save config to env\n const envPath = path.join(__dirname, '../../../.env');\n const updates: string[] = [];\n \n if (!process.env.REDIS_URL && options.redisUrl) {\n updates.push(`REDIS_URL=${options.redisUrl}`);\n }\n if (!process.env.TIMESERIES_URL && options.timeseriesUrl) {\n updates.push(`TIMESERIES_URL=${options.timeseriesUrl}`);\n }\n if (!process.env.S3_BUCKET && options.s3Bucket) {\n updates.push(`S3_BUCKET=${options.s3Bucket}`);\n }\n \n if (updates.length > 0) {\n const fs = await import('fs');\n fs.appendFileSync(envPath, '\\n# Infinite Storage Configuration\\n' + updates.join('\\n') + '\\n');\n }\n } catch (error) {\n spinner.fail('Failed to initialize storage');\n logger.error('Initialization error', error);\n console.error(chalk.red(error instanceof Error ? error.message : 'Unknown error'));\n }\n });\n\n // Store frames\n storage\n .command('store')\n .description('Store current frames in infinite storage')\n .option('--project <name>', 'Project name')\n .option('--user <id>', 'User ID')\n .action(async (options) => {\n const spinner = ora('Storing frames...').start();\n\n try {\n const config = getStorageConfig();\n const storage = new InfiniteStorageSystem(config);\n await storage.initialize();\n\n const frameManager = new FrameManager();\n const frames = frameManager.getAllFrames();\n const userId = options.user || process.env.USER || 'default';\n\n for (const frame of frames) {\n await storage.storeFrame(frame, userId);\n }\n\n spinner.succeed(`Stored ${frames.length} frames`);\n } catch (error) {\n spinner.fail('Failed to store frames');\n logger.error('Store error', error);\n console.error(chalk.red(error instanceof Error ? error.message : 'Unknown error'));\n }\n });\n\n // Retrieve frame\n storage\n .command('retrieve <frameId>')\n .description('Retrieve a frame from storage')\n .option('--user <id>', 'User ID')\n .action(async (frameId, options) => {\n const spinner = ora('Retrieving frame...').start();\n\n try {\n const config = getStorageConfig();\n const storage = new InfiniteStorageSystem(config);\n await storage.initialize();\n\n const userId = options.user || process.env.USER || 'default';\n const frame = await storage.retrieveFrame(frameId, userId);\n\n spinner.stop();\n\n if (frame) {\n console.log(chalk.green('\\n\u2705 Frame Retrieved:'));\n console.log(JSON.stringify(frame, null, 2));\n } else {\n console.log(chalk.yellow('Frame not found'));\n }\n } catch (error) {\n spinner.fail('Failed to retrieve frame');\n logger.error('Retrieve error', error);\n console.error(chalk.red(error instanceof Error ? error.message : 'Unknown error'));\n }\n });\n\n // Show metrics\n storage\n .command('metrics')\n .description('Show storage system metrics')\n .action(async () => {\n const spinner = ora('Fetching metrics...').start();\n\n try {\n const config = getStorageConfig();\n const storage = new InfiniteStorageSystem(config);\n await storage.initialize();\n\n const metrics = await storage.getMetrics();\n\n spinner.stop();\n\n console.log(chalk.cyan('\\n\uD83D\uDCCA Infinite Storage Metrics\\n'));\n \n const table = new Table({\n head: ['Metric', 'Value'],\n colWidths: [30, 40],\n });\n\n table.push(\n ['Total Objects', metrics.totalObjects.toString()],\n ['Storage Size', formatBytes(metrics.storageBytes)],\n ['Avg Latency', `${metrics.avgLatencyMs.toFixed(2)}ms`],\n ['P50 Latency', `${metrics.p50LatencyMs}ms`],\n ['P99 Latency', `${metrics.p99LatencyMs}ms`],\n );\n\n console.log(table.toString());\n\n if (Object.keys(metrics.tierDistribution).length > 0) {\n console.log('\\nTier Distribution:');\n for (const [tier, count] of Object.entries(metrics.tierDistribution)) {\n const percentage = ((count / metrics.totalObjects) * 100).toFixed(1);\n console.log(` ${tier}: ${count} objects (${percentage}%)`);\n }\n }\n\n // Check if meeting STA-287 targets\n console.log(chalk.cyan('\\n\uD83C\uDFAF STA-287 Performance Targets:'));\n const p50Target = metrics.p50LatencyMs <= 50;\n const p99Target = metrics.p99LatencyMs <= 500;\n \n console.log(` P50 \u2264 50ms: ${p50Target ? chalk.green('\u2705 PASS') : chalk.red('\u274C FAIL')} (${metrics.p50LatencyMs}ms)`);\n console.log(` P99 \u2264 500ms: ${p99Target ? chalk.green('\u2705 PASS') : chalk.red('\u274C FAIL')} (${metrics.p99LatencyMs}ms)`);\n } catch (error) {\n spinner.fail('Failed to get metrics');\n logger.error('Metrics error', error);\n console.error(chalk.red(error instanceof Error ? error.message : 'Unknown error'));\n }\n });\n\n // Migrate data\n storage\n .command('migrate')\n .description('Manually trigger tier migration')\n .action(async () => {\n const spinner = ora('Running migration...').start();\n\n try {\n const config = getStorageConfig();\n const storage = new InfiniteStorageSystem(config);\n await storage.initialize();\n\n // Trigger migration (this would normally run automatically)\n // @ts-ignore - accessing private method for manual trigger\n await storage.migrateAgedData();\n\n spinner.succeed('Migration completed');\n } catch (error) {\n spinner.fail('Migration failed');\n logger.error('Migration error', error);\n console.error(chalk.red(error instanceof Error ? error.message : 'Unknown error'));\n }\n });\n\n // Status command\n storage\n .command('status')\n .description('Check storage system status')\n .action(async () => {\n const spinner = ora('Checking status...').start();\n\n try {\n const config = getStorageConfig();\n \n spinner.stop();\n \n console.log(chalk.cyan('\\n\uD83D\uDCE6 Storage System Status\\n'));\n \n // Check Redis\n if (config.redis?.url) {\n try {\n const { createClient } = await import('redis');\n const client = createClient({ url: config.redis.url });\n await client.connect();\n await client.ping();\n await client.quit();\n console.log('Redis (Hot Tier): ' + chalk.green('\u2705 Connected'));\n } catch {\n console.log('Redis (Hot Tier): ' + chalk.red('\u274C Not connected'));\n }\n } else {\n console.log('Redis (Hot Tier): ' + chalk.yellow('\u26A0\uFE0F Not configured'));\n }\n\n // Check TimeSeries DB\n if (config.timeseries?.connectionString) {\n try {\n const { Pool } = await import('pg');\n const pool = new Pool({ connectionString: config.timeseries.connectionString });\n await pool.query('SELECT 1');\n await pool.end();\n console.log('TimeSeries DB (Warm Tier): ' + chalk.green('\u2705 Connected'));\n } catch {\n console.log('TimeSeries DB (Warm Tier): ' + chalk.red('\u274C Not connected'));\n }\n } else {\n console.log('TimeSeries DB (Warm Tier): ' + chalk.yellow('\u26A0\uFE0F Not configured'));\n }\n\n // Check S3\n if (config.s3?.bucket) {\n console.log(`S3 (Cold/Archive Tier): ${chalk.green('\u2705')} Bucket: ${config.s3.bucket}`);\n } else {\n console.log('S3 (Cold/Archive Tier): ' + chalk.yellow('\u26A0\uFE0F Not configured'));\n }\n\n console.log('\\n' + chalk.gray('Configure missing tiers with: stackmemory infinite-storage init'));\n } catch (error) {\n spinner.fail('Failed to check status');\n logger.error('Status error', error);\n console.error(chalk.red(error instanceof Error ? error.message : 'Unknown error'));\n }\n });\n\n return storage;\n}\n\nfunction getStorageConfig(): StorageConfig {\n return {\n redis: {\n url: process.env.REDIS_URL || 'redis://localhost:6379',\n ttlSeconds: parseInt(process.env.REDIS_TTL || '3600'),\n maxMemoryMB: parseInt(process.env.REDIS_MAX_MEMORY_MB || '512'),\n },\n timeseries: {\n connectionString: process.env.TIMESERIES_URL || '',\n retentionDays: parseInt(process.env.TIMESERIES_RETENTION_DAYS || '30'),\n },\n s3: {\n bucket: process.env.S3_BUCKET || '',\n region: process.env.AWS_REGION || 'us-east-1',\n accessKeyId: process.env.AWS_ACCESS_KEY_ID,\n secretAccessKey: process.env.AWS_SECRET_ACCESS_KEY,\n },\n tiers: [],\n };\n}\n\nfunction formatBytes(bytes: number): string {\n if (bytes === 0) return '0 Bytes';\n const k = 1024;\n const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB'];\n const i = Math.floor(Math.log(bytes) / Math.log(k));\n return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];\n}"],
|
|
5
|
-
"mappings": "AAIA,SAAS,eAAe;AACxB,OAAO,WAAW;AAClB,OAAO,SAAS;AAChB,OAAO,WAAW;AAClB,
|
|
4
|
+
"sourcesContent": ["/**\n * CLI commands for Infinite Storage System management\n */\n\nimport { Command } from 'commander';\nimport chalk from 'chalk';\nimport ora from 'ora';\nimport Table from 'cli-table3';\nimport {\n InfiniteStorageSystem,\n StorageConfig,\n} from '../../core/storage/infinite-storage.js';\nimport { FrameManager } from '../../core/context/frame-manager.js';\nimport { Logger } from '../../core/monitoring/logger.js';\nimport dotenv from 'dotenv';\nimport path from 'path';\nimport { fileURLToPath } from 'url';\n// Type-safe environment variable access\nfunction getEnv(key: string, defaultValue?: string): string {\n const value = process.env[key];\n if (value === undefined) {\n if (defaultValue !== undefined) return defaultValue;\n throw new Error(`Environment variable ${key} is required`);\n }\n return value;\n}\n\nfunction getOptionalEnv(key: string): string | undefined {\n return process.env[key];\n}\n\nconst __dirname = path.dirname(fileURLToPath(import.meta.url));\n\n// Load environment variables\ndotenv.config({\n path: path.join(__dirname, '../../../.env'),\n override: true,\n silent: true,\n});\n\nconst logger = new Logger('InfiniteStorage-CLI');\n\nexport function createInfiniteStorageCommand(): Command {\n const storage = new Command('infinite-storage')\n .description('Manage infinite storage system with tiered storage')\n .alias('storage');\n\n // Initialize storage system\n storage\n .command('init')\n .description('Initialize infinite storage system')\n .option('--redis-url <url>', 'Redis connection URL')\n .option('--timeseries-url <url>', 'TimeSeries DB connection URL')\n .option('--s3-bucket <bucket>', 'S3 bucket name')\n .option('--s3-region <region>', 'S3 region', 'us-east-1')\n .action(async (options) => {\n const spinner = ora('Initializing infinite storage...').start();\n\n try {\n const config: StorageConfig = {\n redis: {\n url:\n options.redisUrl ||\n process.env['REDIS_URL'] ||\n 'redis://localhost:6379',\n ttlSeconds: 3600,\n maxMemoryMB: parseInt(process.env['REDIS_MAX_MEMORY_MB'] || '512'),\n },\n timeseries: {\n connectionString:\n options.timeseriesUrl || process.env['TIMESERIES_URL'] || '',\n retentionDays: 30,\n },\n s3: {\n bucket: options.s3Bucket || process.env['S3_BUCKET'] || '',\n region:\n options.s3Region || process.env['AWS_REGION'] || 'us-east-1',\n accessKeyId: process.env['AWS_ACCESS_KEY_ID'],\n secretAccessKey: process.env['AWS_SECRET_ACCESS_KEY'],\n },\n tiers: [],\n };\n\n const storage = new InfiniteStorageSystem(config);\n await storage.initialize();\n\n spinner.succeed('Infinite storage initialized');\n\n console.log(chalk.green('\\n\u2705 Storage Tiers Configured:'));\n console.log(' Hot (Redis): < 1 hour, 5ms latency');\n console.log(' Warm (TimeSeries): 1h - 7 days, 50ms latency');\n console.log(' Cold (S3): 7 - 30 days, 100ms latency');\n console.log(' Archive (Glacier): > 30 days, 1h latency');\n\n // Save config to env\n const envPath = path.join(__dirname, '../../../.env');\n const updates: string[] = [];\n\n if (!process.env['REDIS_URL'] && options.redisUrl) {\n updates.push(`REDIS_URL=${options.redisUrl}`);\n }\n if (!process.env['TIMESERIES_URL'] && options.timeseriesUrl) {\n updates.push(`TIMESERIES_URL=${options.timeseriesUrl}`);\n }\n if (!process.env['S3_BUCKET'] && options.s3Bucket) {\n updates.push(`S3_BUCKET=${options.s3Bucket}`);\n }\n\n if (updates.length > 0) {\n const fs = await import('fs');\n fs.appendFileSync(\n envPath,\n '\\n# Infinite Storage Configuration\\n' + updates.join('\\n') + '\\n'\n );\n }\n } catch (error: unknown) {\n spinner.fail('Failed to initialize storage');\n logger.error('Initialization error', error);\n console.error(\n chalk.red(error instanceof Error ? error.message : 'Unknown error')\n );\n }\n });\n\n // Store frames\n storage\n .command('store')\n .description('Store current frames in infinite storage')\n .option('--project <name>', 'Project name')\n .option('--user <id>', 'User ID')\n .action(async (options) => {\n const spinner = ora('Storing frames...').start();\n\n try {\n const config = getStorageConfig();\n const storage = new InfiniteStorageSystem(config);\n await storage.initialize();\n\n const frameManager = new FrameManager();\n const frames = frameManager.getAllFrames();\n const userId = options.user || process.env['USER'] || 'default';\n\n for (const frame of frames) {\n await storage.storeFrame(frame, userId);\n }\n\n spinner.succeed(`Stored ${frames.length} frames`);\n } catch (error: unknown) {\n spinner.fail('Failed to store frames');\n logger.error('Store error', error);\n console.error(\n chalk.red(error instanceof Error ? error.message : 'Unknown error')\n );\n }\n });\n\n // Retrieve frame\n storage\n .command('retrieve <frameId>')\n .description('Retrieve a frame from storage')\n .option('--user <id>', 'User ID')\n .action(async (frameId, options) => {\n const spinner = ora('Retrieving frame...').start();\n\n try {\n const config = getStorageConfig();\n const storage = new InfiniteStorageSystem(config);\n await storage.initialize();\n\n const userId = options.user || process.env['USER'] || 'default';\n const frame = await storage.retrieveFrame(frameId, userId);\n\n spinner.stop();\n\n if (frame) {\n console.log(chalk.green('\\n\u2705 Frame Retrieved:'));\n console.log(JSON.stringify(frame, null, 2));\n } else {\n console.log(chalk.yellow('Frame not found'));\n }\n } catch (error: unknown) {\n spinner.fail('Failed to retrieve frame');\n logger.error('Retrieve error', error);\n console.error(\n chalk.red(error instanceof Error ? error.message : 'Unknown error')\n );\n }\n });\n\n // Show metrics\n storage\n .command('metrics')\n .description('Show storage system metrics')\n .action(async () => {\n const spinner = ora('Fetching metrics...').start();\n\n try {\n const config = getStorageConfig();\n const storage = new InfiniteStorageSystem(config);\n await storage.initialize();\n\n const metrics = await storage.getMetrics();\n\n spinner.stop();\n\n console.log(chalk.cyan('\\n\uD83D\uDCCA Infinite Storage Metrics\\n'));\n\n const table = new Table({\n head: ['Metric', 'Value'],\n colWidths: [30, 40],\n });\n\n table.push(\n ['Total Objects', metrics.totalObjects.toString()],\n ['Storage Size', formatBytes(metrics.storageBytes)],\n ['Avg Latency', `${metrics.avgLatencyMs.toFixed(2)}ms`],\n ['P50 Latency', `${metrics.p50LatencyMs}ms`],\n ['P99 Latency', `${metrics.p99LatencyMs}ms`]\n );\n\n console.log(table.toString());\n\n if (Object.keys(metrics.tierDistribution).length > 0) {\n console.log('\\nTier Distribution:');\n for (const [tier, count] of Object.entries(\n metrics.tierDistribution\n )) {\n const percentage = ((count / metrics.totalObjects) * 100).toFixed(\n 1\n );\n console.log(` ${tier}: ${count} objects (${percentage}%)`);\n }\n }\n\n // Check if meeting STA-287 targets\n console.log(chalk.cyan('\\n\uD83C\uDFAF STA-287 Performance Targets:'));\n const p50Target = metrics.p50LatencyMs <= 50;\n const p99Target = metrics.p99LatencyMs <= 500;\n\n console.log(\n ` P50 \u2264 50ms: ${p50Target ? chalk.green('\u2705 PASS') : chalk.red('\u274C FAIL')} (${metrics.p50LatencyMs}ms)`\n );\n console.log(\n ` P99 \u2264 500ms: ${p99Target ? chalk.green('\u2705 PASS') : chalk.red('\u274C FAIL')} (${metrics.p99LatencyMs}ms)`\n );\n } catch (error: unknown) {\n spinner.fail('Failed to get metrics');\n logger.error('Metrics error', error);\n console.error(\n chalk.red(error instanceof Error ? error.message : 'Unknown error')\n );\n }\n });\n\n // Migrate data\n storage\n .command('migrate')\n .description('Manually trigger tier migration')\n .action(async () => {\n const spinner = ora('Running migration...').start();\n\n try {\n const config = getStorageConfig();\n const storage = new InfiniteStorageSystem(config);\n await storage.initialize();\n\n // Trigger migration (this would normally run automatically)\n // @ts-ignore - accessing private method for manual trigger\n await storage.migrateAgedData();\n\n spinner.succeed('Migration completed');\n } catch (error: unknown) {\n spinner.fail('Migration failed');\n logger.error('Migration error', error);\n console.error(\n chalk.red(error instanceof Error ? error.message : 'Unknown error')\n );\n }\n });\n\n // Status command\n storage\n .command('status')\n .description('Check storage system status')\n .action(async () => {\n const spinner = ora('Checking status...').start();\n\n try {\n const config = getStorageConfig();\n\n spinner.stop();\n\n console.log(chalk.cyan('\\n\uD83D\uDCE6 Storage System Status\\n'));\n\n // Check Redis\n if (config.redis?.url) {\n try {\n const { createClient } = await import('redis');\n const client = createClient({ url: config.redis.url });\n await client.connect();\n await client.ping();\n await client.quit();\n console.log('Redis (Hot Tier): ' + chalk.green('\u2705 Connected'));\n } catch {\n console.log('Redis (Hot Tier): ' + chalk.red('\u274C Not connected'));\n }\n } else {\n console.log('Redis (Hot Tier): ' + chalk.yellow('\u26A0\uFE0F Not configured'));\n }\n\n // Check TimeSeries DB\n if (config.timeseries?.connectionString) {\n try {\n const { Pool } = await import('pg');\n const pool = new Pool({\n connectionString: config.timeseries.connectionString,\n });\n await pool.query('SELECT 1');\n await pool.end();\n console.log(\n 'TimeSeries DB (Warm Tier): ' + chalk.green('\u2705 Connected')\n );\n } catch {\n console.log(\n 'TimeSeries DB (Warm Tier): ' + chalk.red('\u274C Not connected')\n );\n }\n } else {\n console.log(\n 'TimeSeries DB (Warm Tier): ' + chalk.yellow('\u26A0\uFE0F Not configured')\n );\n }\n\n // Check S3\n if (config.s3?.bucket) {\n console.log(\n `S3 (Cold/Archive Tier): ${chalk.green('\u2705')} Bucket: ${config.s3.bucket}`\n );\n } else {\n console.log(\n 'S3 (Cold/Archive Tier): ' + chalk.yellow('\u26A0\uFE0F Not configured')\n );\n }\n\n console.log(\n '\\n' +\n chalk.gray(\n 'Configure missing tiers with: stackmemory infinite-storage init'\n )\n );\n } catch (error: unknown) {\n spinner.fail('Failed to check status');\n logger.error('Status error', error);\n console.error(\n chalk.red(error instanceof Error ? error.message : 'Unknown error')\n );\n }\n });\n\n return storage;\n}\n\nfunction getStorageConfig(): StorageConfig {\n return {\n redis: {\n url: process.env['REDIS_URL'] || 'redis://localhost:6379',\n ttlSeconds: parseInt(process.env['REDIS_TTL'] || '3600'),\n maxMemoryMB: parseInt(process.env['REDIS_MAX_MEMORY_MB'] || '512'),\n },\n timeseries: {\n connectionString: process.env['TIMESERIES_URL'] || '',\n retentionDays: parseInt(process.env['TIMESERIES_RETENTION_DAYS'] || '30'),\n },\n s3: {\n bucket: process.env['S3_BUCKET'] || '',\n region: process.env['AWS_REGION'] || 'us-east-1',\n accessKeyId: process.env['AWS_ACCESS_KEY_ID'],\n secretAccessKey: process.env['AWS_SECRET_ACCESS_KEY'],\n },\n tiers: [],\n };\n}\n\nfunction formatBytes(bytes: number): string {\n if (bytes === 0) return '0 Bytes';\n const k = 1024;\n const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB'];\n const i = Math.floor(Math.log(bytes) / Math.log(k));\n return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": ["storage"]
|
|
7
7
|
}
|
|
@@ -1,15 +1,40 @@
|
|
|
1
1
|
import chalk from "chalk";
|
|
2
2
|
import { LinearRestClient } from "../../integrations/linear/rest-client.js";
|
|
3
|
+
function getEnv(key, defaultValue) {
|
|
4
|
+
const value = process.env[key];
|
|
5
|
+
if (value === void 0) {
|
|
6
|
+
if (defaultValue !== void 0) return defaultValue;
|
|
7
|
+
throw new Error(`Environment variable ${key} is required`);
|
|
8
|
+
}
|
|
9
|
+
return value;
|
|
10
|
+
}
|
|
11
|
+
function getOptionalEnv(key) {
|
|
12
|
+
return process.env[key];
|
|
13
|
+
}
|
|
3
14
|
function registerLinearCreateCommand(parent) {
|
|
4
|
-
parent.command("linear:create").description("Create a new Linear task in the correct project").option("--api-key <key>", "Linear API key for target workspace").option("--title <title>", "Task title (required)").option("--description <desc>", "Task description").option(
|
|
15
|
+
parent.command("linear:create").description("Create a new Linear task in the correct project").option("--api-key <key>", "Linear API key for target workspace").option("--title <title>", "Task title (required)").option("--description <desc>", "Task description").option(
|
|
16
|
+
"--priority <level>",
|
|
17
|
+
"Priority: urgent(1), high(2), medium(3), low(4)",
|
|
18
|
+
"3"
|
|
19
|
+
).option(
|
|
20
|
+
"--state <state>",
|
|
21
|
+
"Initial state: backlog, todo, started",
|
|
22
|
+
"backlog"
|
|
23
|
+
).action(async (options) => {
|
|
5
24
|
try {
|
|
6
|
-
const apiKey = options.apiKey || process.env
|
|
25
|
+
const apiKey = options.apiKey || process.env["LINEAR_NEW_API_KEY"];
|
|
7
26
|
if (!apiKey) {
|
|
8
|
-
console.error(
|
|
27
|
+
console.error(
|
|
28
|
+
chalk.red(
|
|
29
|
+
"\u274C API key required. Use --api-key or set LINEAR_NEW_API_KEY"
|
|
30
|
+
)
|
|
31
|
+
);
|
|
9
32
|
return;
|
|
10
33
|
}
|
|
11
34
|
if (!options.title) {
|
|
12
|
-
console.error(
|
|
35
|
+
console.error(
|
|
36
|
+
chalk.red('\u274C Task title required. Use --title "Your task title"')
|
|
37
|
+
);
|
|
13
38
|
return;
|
|
14
39
|
}
|
|
15
40
|
const client = new LinearRestClient(apiKey);
|
|
@@ -57,14 +82,21 @@ function registerLinearCreateCommand(parent) {
|
|
|
57
82
|
console.log(chalk.gray(` Description: ${options.description}`));
|
|
58
83
|
}
|
|
59
84
|
} catch (error) {
|
|
60
|
-
console.error(
|
|
85
|
+
console.error(
|
|
86
|
+
chalk.red("\u274C Task creation failed:"),
|
|
87
|
+
error.message
|
|
88
|
+
);
|
|
61
89
|
}
|
|
62
90
|
});
|
|
63
91
|
parent.command("linear:quick").description("Quick task creation with prompts").option("--api-key <key>", "Linear API key for target workspace").action(async (options) => {
|
|
64
92
|
try {
|
|
65
|
-
const apiKey = options.apiKey || process.env
|
|
93
|
+
const apiKey = options.apiKey || process.env["LINEAR_NEW_API_KEY"];
|
|
66
94
|
if (!apiKey) {
|
|
67
|
-
console.error(
|
|
95
|
+
console.error(
|
|
96
|
+
chalk.red(
|
|
97
|
+
"\u274C API key required. Use --api-key or set LINEAR_NEW_API_KEY"
|
|
98
|
+
)
|
|
99
|
+
);
|
|
68
100
|
return;
|
|
69
101
|
}
|
|
70
102
|
const readline = await import("readline");
|
|
@@ -85,8 +117,12 @@ function registerLinearCreateCommand(parent) {
|
|
|
85
117
|
rl.close();
|
|
86
118
|
return;
|
|
87
119
|
}
|
|
88
|
-
const description = await question(
|
|
89
|
-
|
|
120
|
+
const description = await question(
|
|
121
|
+
chalk.yellow("Description (optional): ")
|
|
122
|
+
);
|
|
123
|
+
const priorityInput = await question(
|
|
124
|
+
chalk.yellow("Priority (1=urgent, 2=high, 3=medium, 4=low) [3]: ")
|
|
125
|
+
);
|
|
90
126
|
const priority = priorityInput.trim() || "3";
|
|
91
127
|
rl.close();
|
|
92
128
|
console.log(chalk.yellow("\n\u{1F504} Creating task..."));
|
|
@@ -122,7 +158,10 @@ function registerLinearCreateCommand(parent) {
|
|
|
122
158
|
console.log(chalk.blue(`\u{1F4CB} ${task.identifier}: ${task.title}`));
|
|
123
159
|
console.log(chalk.gray(` URL: ${task.url}`));
|
|
124
160
|
} catch (error) {
|
|
125
|
-
console.error(
|
|
161
|
+
console.error(
|
|
162
|
+
chalk.red("\u274C Task creation failed:"),
|
|
163
|
+
error.message
|
|
164
|
+
);
|
|
126
165
|
}
|
|
127
166
|
});
|
|
128
167
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/cli/commands/linear-create.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * Linear Task Creation Command\n */\n\nimport { Command } from 'commander';\nimport chalk from 'chalk';\nimport { LinearRestClient } from '../../integrations/linear/rest-client.js';\n\nexport function registerLinearCreateCommand(parent: Command) {\n parent\n .command('linear:create')\n .description('Create a new Linear task in the correct project')\n .option('--api-key <key>', 'Linear API key for target workspace')\n .option('--title <title>', 'Task title (required)')\n .option('--description <desc>', 'Task description')\n .option('--priority <level>'
|
|
5
|
-
"mappings": "AAKA,OAAO,WAAW;AAClB,SAAS,wBAAwB;
|
|
4
|
+
"sourcesContent": ["/**\n * Linear Task Creation Command\n */\n\nimport { Command } from 'commander';\nimport chalk from 'chalk';\nimport { LinearRestClient } from '../../integrations/linear/rest-client.js';\n// Type-safe environment variable access\nfunction getEnv(key: string, defaultValue?: string): string {\n const value = process.env[key];\n if (value === undefined) {\n if (defaultValue !== undefined) return defaultValue;\n throw new Error(`Environment variable ${key} is required`);\n }\n return value;\n}\n\nfunction getOptionalEnv(key: string): string | undefined {\n return process.env[key];\n}\n\nexport function registerLinearCreateCommand(parent: Command) {\n parent\n .command('linear:create')\n .description('Create a new Linear task in the correct project')\n .option('--api-key <key>', 'Linear API key for target workspace')\n .option('--title <title>', 'Task title (required)')\n .option('--description <desc>', 'Task description')\n .option(\n '--priority <level>',\n 'Priority: urgent(1), high(2), medium(3), low(4)',\n '3'\n )\n .option(\n '--state <state>',\n 'Initial state: backlog, todo, started',\n 'backlog'\n )\n .action(async (options) => {\n try {\n const apiKey = options.apiKey || process.env['LINEAR_NEW_API_KEY'];\n\n if (!apiKey) {\n console.error(\n chalk.red(\n '\u274C API key required. Use --api-key or set LINEAR_NEW_API_KEY'\n )\n );\n return;\n }\n\n if (!options.title) {\n console.error(\n chalk.red('\u274C Task title required. Use --title \"Your task title\"')\n );\n return;\n }\n\n const client = new LinearRestClient(apiKey);\n\n console.log(chalk.yellow('\uD83D\uDD04 Creating new Linear task...'));\n\n // Get team info\n const team = await client.getTeam();\n console.log(chalk.cyan(`\uD83C\uDFAF Target team: ${team.name} (${team.key})`));\n\n // Create the task\n const createQuery = `\n mutation CreateIssue($input: IssueCreateInput!) {\n issueCreate(input: $input) {\n success\n issue {\n id\n identifier\n title\n description\n state {\n id\n name\n type\n }\n priority\n url\n createdAt\n }\n }\n }\n `;\n\n const taskInput = {\n title: options.title,\n description: options.description || '',\n teamId: team.id,\n priority: parseInt(options.priority),\n };\n\n const response = await client.makeRequest<{\n data: {\n issueCreate: {\n success: boolean;\n issue: any;\n };\n };\n }>(createQuery, { input: taskInput });\n\n if (!response.data?.issueCreate?.success) {\n throw new Error('Failed to create task in Linear');\n }\n\n const task = response.data.issueCreate.issue;\n\n console.log(chalk.green('\\n\u2705 Task created successfully!'));\n console.log(chalk.blue(`\uD83D\uDCCB ${task.identifier}: ${task.title}`));\n console.log(chalk.gray(` State: ${task.state.name}`));\n console.log(chalk.gray(` Priority: ${task.priority}`));\n console.log(chalk.gray(` URL: ${task.url}`));\n\n if (options.description) {\n console.log(chalk.gray(` Description: ${options.description}`));\n }\n } catch (error: unknown) {\n console.error(\n chalk.red('\u274C Task creation failed:'),\n (error as Error).message\n );\n }\n });\n\n parent\n .command('linear:quick')\n .description('Quick task creation with prompts')\n .option('--api-key <key>', 'Linear API key for target workspace')\n .action(async (options) => {\n try {\n const apiKey = options.apiKey || process.env['LINEAR_NEW_API_KEY'];\n\n if (!apiKey) {\n console.error(\n chalk.red(\n '\u274C API key required. Use --api-key or set LINEAR_NEW_API_KEY'\n )\n );\n return;\n }\n\n const readline = await import('readline');\n const rl = readline.createInterface({\n input: process.stdin,\n output: process.stdout,\n });\n\n const question = (prompt: string): Promise<string> => {\n return new Promise((resolve) => {\n rl.question(prompt, resolve);\n });\n };\n\n console.log(chalk.blue('\uD83D\uDCDD Quick Linear Task Creation'));\n console.log(chalk.gray('Press Ctrl+C to cancel at any time\\n'));\n\n const title = await question(chalk.yellow('Task title: '));\n if (!title.trim()) {\n console.log(chalk.red('\u274C Title is required'));\n rl.close();\n return;\n }\n\n const description = await question(\n chalk.yellow('Description (optional): ')\n );\n const priorityInput = await question(\n chalk.yellow('Priority (1=urgent, 2=high, 3=medium, 4=low) [3]: ')\n );\n const priority = priorityInput.trim() || '3';\n\n rl.close();\n\n console.log(chalk.yellow('\\n\uD83D\uDD04 Creating task...'));\n\n const client = new LinearRestClient(apiKey);\n const team = await client.getTeam();\n\n const createQuery = `\n mutation CreateIssue($input: IssueCreateInput!) {\n issueCreate(input: $input) {\n success\n issue {\n id\n identifier\n title\n state { name }\n priority\n url\n }\n }\n }\n `;\n\n const taskInput = {\n title: title.trim(),\n description: description.trim() || undefined,\n teamId: team.id,\n priority: parseInt(priority),\n };\n\n const response = await client.makeRequest<{\n data: {\n issueCreate: {\n success: boolean;\n issue: any;\n };\n };\n }>(createQuery, { input: taskInput });\n\n if (!response.data?.issueCreate?.success) {\n throw new Error('Failed to create task in Linear');\n }\n\n const task = response.data.issueCreate.issue;\n\n console.log(chalk.green('\\n\u2705 Task created successfully!'));\n console.log(chalk.blue(`\uD83D\uDCCB ${task.identifier}: ${task.title}`));\n console.log(chalk.gray(` URL: ${task.url}`));\n } catch (error: unknown) {\n console.error(\n chalk.red('\u274C Task creation failed:'),\n (error as Error).message\n );\n }\n });\n}\n"],
|
|
5
|
+
"mappings": "AAKA,OAAO,WAAW;AAClB,SAAS,wBAAwB;AAEjC,SAAS,OAAO,KAAa,cAA+B;AAC1D,QAAM,QAAQ,QAAQ,IAAI,GAAG;AAC7B,MAAI,UAAU,QAAW;AACvB,QAAI,iBAAiB,OAAW,QAAO;AACvC,UAAM,IAAI,MAAM,wBAAwB,GAAG,cAAc;AAAA,EAC3D;AACA,SAAO;AACT;AAEA,SAAS,eAAe,KAAiC;AACvD,SAAO,QAAQ,IAAI,GAAG;AACxB;AAEO,SAAS,4BAA4B,QAAiB;AAC3D,SACG,QAAQ,eAAe,EACvB,YAAY,iDAAiD,EAC7D,OAAO,mBAAmB,qCAAqC,EAC/D,OAAO,mBAAmB,uBAAuB,EACjD,OAAO,wBAAwB,kBAAkB,EACjD;AAAA,IACC;AAAA,IACA;AAAA,IACA;AAAA,EACF,EACC;AAAA,IACC;AAAA,IACA;AAAA,IACA;AAAA,EACF,EACC,OAAO,OAAO,YAAY;AACzB,QAAI;AACF,YAAM,SAAS,QAAQ,UAAU,QAAQ,IAAI,oBAAoB;AAEjE,UAAI,CAAC,QAAQ;AACX,gBAAQ;AAAA,UACN,MAAM;AAAA,YACJ;AAAA,UACF;AAAA,QACF;AACA;AAAA,MACF;AAEA,UAAI,CAAC,QAAQ,OAAO;AAClB,gBAAQ;AAAA,UACN,MAAM,IAAI,2DAAsD;AAAA,QAClE;AACA;AAAA,MACF;AAEA,YAAM,SAAS,IAAI,iBAAiB,MAAM;AAE1C,cAAQ,IAAI,MAAM,OAAO,uCAAgC,CAAC;AAG1D,YAAM,OAAO,MAAM,OAAO,QAAQ;AAClC,cAAQ,IAAI,MAAM,KAAK,0BAAmB,KAAK,IAAI,KAAK,KAAK,GAAG,GAAG,CAAC;AAGpE,YAAM,cAAc;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAsBpB,YAAM,YAAY;AAAA,QAChB,OAAO,QAAQ;AAAA,QACf,aAAa,QAAQ,eAAe;AAAA,QACpC,QAAQ,KAAK;AAAA,QACb,UAAU,SAAS,QAAQ,QAAQ;AAAA,MACrC;AAEA,YAAM,WAAW,MAAM,OAAO,YAO3B,aAAa,EAAE,OAAO,UAAU,CAAC;AAEpC,UAAI,CAAC,SAAS,MAAM,aAAa,SAAS;AACxC,cAAM,IAAI,MAAM,iCAAiC;AAAA,MACnD;AAEA,YAAM,OAAO,SAAS,KAAK,YAAY;AAEvC,cAAQ,IAAI,MAAM,MAAM,qCAAgC,CAAC;AACzD,cAAQ,IAAI,MAAM,KAAK,aAAM,KAAK,UAAU,KAAK,KAAK,KAAK,EAAE,CAAC;AAC9D,cAAQ,IAAI,MAAM,KAAK,aAAa,KAAK,MAAM,IAAI,EAAE,CAAC;AACtD,cAAQ,IAAI,MAAM,KAAK,gBAAgB,KAAK,QAAQ,EAAE,CAAC;AACvD,cAAQ,IAAI,MAAM,KAAK,WAAW,KAAK,GAAG,EAAE,CAAC;AAE7C,UAAI,QAAQ,aAAa;AACvB,gBAAQ,IAAI,MAAM,KAAK,mBAAmB,QAAQ,WAAW,EAAE,CAAC;AAAA,MAClE;AAAA,IACF,SAAS,OAAgB;AACvB,cAAQ;AAAA,QACN,MAAM,IAAI,8BAAyB;AAAA,QAClC,MAAgB;AAAA,MACnB;AAAA,IACF;AAAA,EACF,CAAC;AAEH,SACG,QAAQ,cAAc,EACtB,YAAY,kCAAkC,EAC9C,OAAO,mBAAmB,qCAAqC,EAC/D,OAAO,OAAO,YAAY;AACzB,QAAI;AACF,YAAM,SAAS,QAAQ,UAAU,QAAQ,IAAI,oBAAoB;AAEjE,UAAI,CAAC,QAAQ;AACX,gBAAQ;AAAA,UACN,MAAM;AAAA,YACJ;AAAA,UACF;AAAA,QACF;AACA;AAAA,MACF;AAEA,YAAM,WAAW,MAAM,OAAO,UAAU;AACxC,YAAM,KAAK,SAAS,gBAAgB;AAAA,QAClC,OAAO,QAAQ;AAAA,QACf,QAAQ,QAAQ;AAAA,MAClB,CAAC;AAED,YAAM,WAAW,CAAC,WAAoC;AACpD,eAAO,IAAI,QAAQ,CAAC,YAAY;AAC9B,aAAG,SAAS,QAAQ,OAAO;AAAA,QAC7B,CAAC;AAAA,MACH;AAEA,cAAQ,IAAI,MAAM,KAAK,sCAA+B,CAAC;AACvD,cAAQ,IAAI,MAAM,KAAK,sCAAsC,CAAC;AAE9D,YAAM,QAAQ,MAAM,SAAS,MAAM,OAAO,cAAc,CAAC;AACzD,UAAI,CAAC,MAAM,KAAK,GAAG;AACjB,gBAAQ,IAAI,MAAM,IAAI,0BAAqB,CAAC;AAC5C,WAAG,MAAM;AACT;AAAA,MACF;AAEA,YAAM,cAAc,MAAM;AAAA,QACxB,MAAM,OAAO,0BAA0B;AAAA,MACzC;AACA,YAAM,gBAAgB,MAAM;AAAA,QAC1B,MAAM,OAAO,oDAAoD;AAAA,MACnE;AACA,YAAM,WAAW,cAAc,KAAK,KAAK;AAEzC,SAAG,MAAM;AAET,cAAQ,IAAI,MAAM,OAAO,8BAAuB,CAAC;AAEjD,YAAM,SAAS,IAAI,iBAAiB,MAAM;AAC1C,YAAM,OAAO,MAAM,OAAO,QAAQ;AAElC,YAAM,cAAc;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAgBpB,YAAM,YAAY;AAAA,QAChB,OAAO,MAAM,KAAK;AAAA,QAClB,aAAa,YAAY,KAAK,KAAK;AAAA,QACnC,QAAQ,KAAK;AAAA,QACb,UAAU,SAAS,QAAQ;AAAA,MAC7B;AAEA,YAAM,WAAW,MAAM,OAAO,YAO3B,aAAa,EAAE,OAAO,UAAU,CAAC;AAEpC,UAAI,CAAC,SAAS,MAAM,aAAa,SAAS;AACxC,cAAM,IAAI,MAAM,iCAAiC;AAAA,MACnD;AAEA,YAAM,OAAO,SAAS,KAAK,YAAY;AAEvC,cAAQ,IAAI,MAAM,MAAM,qCAAgC,CAAC;AACzD,cAAQ,IAAI,MAAM,KAAK,aAAM,KAAK,UAAU,KAAK,KAAK,KAAK,EAAE,CAAC;AAC9D,cAAQ,IAAI,MAAM,KAAK,WAAW,KAAK,GAAG,EAAE,CAAC;AAAA,IAC/C,SAAS,OAAgB;AACvB,cAAQ;AAAA,QACN,MAAM,IAAI,8BAAyB;AAAA,QAClC,MAAgB;AAAA,MACnB;AAAA,IACF;AAAA,EACF,CAAC;AACL;",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -1,11 +1,27 @@
|
|
|
1
1
|
import chalk from "chalk";
|
|
2
2
|
import { LinearRestClient } from "../../integrations/linear/rest-client.js";
|
|
3
|
+
function getEnv(key, defaultValue) {
|
|
4
|
+
const value = process.env[key];
|
|
5
|
+
if (value === void 0) {
|
|
6
|
+
if (defaultValue !== void 0) return defaultValue;
|
|
7
|
+
throw new Error(`Environment variable ${key} is required`);
|
|
8
|
+
}
|
|
9
|
+
return value;
|
|
10
|
+
}
|
|
11
|
+
function getOptionalEnv(key) {
|
|
12
|
+
return process.env[key];
|
|
13
|
+
}
|
|
3
14
|
function registerLinearListCommand(parent) {
|
|
4
|
-
parent.command("linear:list").alias("linear:ls").description("List Linear tasks (fast, memory-cached)").option("--limit <n>", "Number of tasks to show", "20").option(
|
|
15
|
+
parent.command("linear:list").alias("linear:ls").description("List Linear tasks (fast, memory-cached)").option("--limit <n>", "Number of tasks to show", "20").option(
|
|
16
|
+
"--status <status>",
|
|
17
|
+
"Filter by status (backlog, started, completed, etc.)"
|
|
18
|
+
).option("--my", "Show only tasks assigned to me").option("--cache", "Show cache stats only").option("--refresh", "Force refresh cache").option("--count", "Show count by status only").action(async (options) => {
|
|
5
19
|
try {
|
|
6
|
-
const apiKey = process.env
|
|
20
|
+
const apiKey = process.env["LINEAR_API_KEY"];
|
|
7
21
|
if (!apiKey) {
|
|
8
|
-
console.log(
|
|
22
|
+
console.log(
|
|
23
|
+
chalk.yellow("\u26A0 Set LINEAR_API_KEY environment variable")
|
|
24
|
+
);
|
|
9
25
|
return;
|
|
10
26
|
}
|
|
11
27
|
const restClient = new LinearRestClient(apiKey);
|
|
@@ -15,7 +31,9 @@ function registerLinearListCommand(parent) {
|
|
|
15
31
|
console.log(` Size: ${stats.size} tasks`);
|
|
16
32
|
console.log(` Age: ${Math.round(stats.age / 1e3)}s`);
|
|
17
33
|
console.log(` Fresh: ${stats.fresh ? "yes" : "no"}`);
|
|
18
|
-
console.log(
|
|
34
|
+
console.log(
|
|
35
|
+
` Last sync: ${new Date(stats.lastSync).toLocaleString()}`
|
|
36
|
+
);
|
|
19
37
|
return;
|
|
20
38
|
}
|
|
21
39
|
if (options.count) {
|
|
@@ -25,8 +43,12 @@ function registerLinearListCommand(parent) {
|
|
|
25
43
|
console.log(` ${status}: ${count}`);
|
|
26
44
|
});
|
|
27
45
|
const cacheStats2 = restClient.getCacheStats();
|
|
28
|
-
console.log(
|
|
29
|
-
|
|
46
|
+
console.log(
|
|
47
|
+
chalk.gray(
|
|
48
|
+
`
|
|
49
|
+
Cache: ${cacheStats2.size} tasks, age: ${Math.round(cacheStats2.age / 1e3)}s`
|
|
50
|
+
)
|
|
51
|
+
);
|
|
30
52
|
return;
|
|
31
53
|
}
|
|
32
54
|
let tasks;
|
|
@@ -43,18 +65,30 @@ Cache: ${cacheStats2.size} tasks, age: ${Math.round(cacheStats2.age / 1e3)}s`));
|
|
|
43
65
|
}
|
|
44
66
|
const limit = parseInt(options.limit);
|
|
45
67
|
const displayTasks = tasks.slice(0, limit);
|
|
46
|
-
console.log(
|
|
47
|
-
|
|
68
|
+
console.log(
|
|
69
|
+
chalk.cyan(
|
|
70
|
+
`
|
|
71
|
+
\u{1F4CB} Linear Tasks (${displayTasks.length}/${tasks.length}):`
|
|
72
|
+
)
|
|
73
|
+
);
|
|
48
74
|
displayTasks.forEach((task) => {
|
|
49
75
|
const priority = task.priority ? `P${task.priority}` : "";
|
|
50
76
|
const assignee = task.assignee ? ` @${task.assignee.name}` : "";
|
|
51
77
|
const statusColor = task.state.type === "completed" ? chalk.green : task.state.type === "started" ? chalk.yellow : chalk.gray;
|
|
52
78
|
console.log(`${chalk.blue(task.identifier)} ${task.title}`);
|
|
53
|
-
console.log(
|
|
79
|
+
console.log(
|
|
80
|
+
chalk.gray(
|
|
81
|
+
` ${statusColor(task.state.name)} ${priority}${assignee}`
|
|
82
|
+
)
|
|
83
|
+
);
|
|
54
84
|
});
|
|
55
85
|
const cacheStats = restClient.getCacheStats();
|
|
56
|
-
console.log(
|
|
57
|
-
|
|
86
|
+
console.log(
|
|
87
|
+
chalk.gray(
|
|
88
|
+
`
|
|
89
|
+
${displayTasks.length} shown, ${tasks.length} total \u2022 Cache: ${Math.round(cacheStats.age / 1e3)}s old`
|
|
90
|
+
)
|
|
91
|
+
);
|
|
58
92
|
} catch (error) {
|
|
59
93
|
console.error(
|
|
60
94
|
chalk.red("Failed to list tasks:"),
|