@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
|
@@ -8,20 +8,31 @@ import Database from "better-sqlite3";
|
|
|
8
8
|
import { BrowserMCPIntegration } from "../../features/browser/browser-mcp.js";
|
|
9
9
|
import { join, dirname } from "path";
|
|
10
10
|
import { existsSync, mkdirSync } from "fs";
|
|
11
|
+
function getEnv(key, defaultValue) {
|
|
12
|
+
const value = process.env[key];
|
|
13
|
+
if (value === void 0) {
|
|
14
|
+
if (defaultValue !== void 0) return defaultValue;
|
|
15
|
+
throw new Error(`Environment variable ${key} is required`);
|
|
16
|
+
}
|
|
17
|
+
return value;
|
|
18
|
+
}
|
|
19
|
+
function getOptionalEnv(key) {
|
|
20
|
+
return process.env[key];
|
|
21
|
+
}
|
|
11
22
|
const config = {
|
|
12
|
-
port: parseInt(process.env
|
|
13
|
-
environment: process.env
|
|
14
|
-
corsOrigins: process.env
|
|
23
|
+
port: parseInt(process.env["PORT"] || "3000"),
|
|
24
|
+
environment: process.env["NODE_ENV"] || "development",
|
|
25
|
+
corsOrigins: process.env["CORS_ORIGINS"]?.split(",") || [
|
|
15
26
|
"http://localhost:3000"
|
|
16
27
|
],
|
|
17
|
-
authMode: process.env
|
|
18
|
-
apiKeySecret: process.env
|
|
19
|
-
jwtSecret: process.env
|
|
20
|
-
databaseUrl: process.env
|
|
21
|
-
rateLimitEnabled: process.env
|
|
22
|
-
rateLimitFree: parseInt(process.env
|
|
23
|
-
enableWebSocket: process.env
|
|
24
|
-
enableAnalytics: process.env
|
|
28
|
+
authMode: process.env["AUTH_MODE"] || "api_key",
|
|
29
|
+
apiKeySecret: process.env["API_KEY_SECRET"] || "development-secret",
|
|
30
|
+
jwtSecret: process.env["JWT_SECRET"] || "development-jwt-secret",
|
|
31
|
+
databaseUrl: process.env["DATABASE_URL"] || join(process.cwd(), ".stackmemory", "railway.db"),
|
|
32
|
+
rateLimitEnabled: process.env["RATE_LIMIT_ENABLED"] === "true",
|
|
33
|
+
rateLimitFree: parseInt(process.env["RATE_LIMIT_FREE"] || "100"),
|
|
34
|
+
enableWebSocket: process.env["ENABLE_WEBSOCKET"] !== "false",
|
|
35
|
+
enableAnalytics: process.env["ENABLE_ANALYTICS"] === "true"
|
|
25
36
|
};
|
|
26
37
|
const rateLimiter = /* @__PURE__ */ new Map();
|
|
27
38
|
class RailwayMCPServer {
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/servers/railway/index.ts"],
|
|
4
|
-
"sourcesContent": ["#!/usr/bin/env node\n/**\n * Railway MCP Server Entry Point\n * Simplified production server for Railway deployment\n */\n\nimport express from 'express';\nimport { createServer } from 'http';\nimport { WebSocketServer } from 'ws';\nimport cors from 'cors';\nimport { Server } from '@modelcontextprotocol/sdk/server/index.js';\n// WebSocket transport will be handled differently for Railway\nimport Database from 'better-sqlite3';\nimport { BrowserMCPIntegration } from '../../features/browser/browser-mcp.js';\nimport { join, dirname } from 'path';\nimport { existsSync, mkdirSync } from 'fs';\n\n// Configuration\nconst config = {\n port: parseInt(process.env.PORT || '3000'),\n environment: process.env.NODE_ENV || 'development',\n corsOrigins: process.env.CORS_ORIGINS?.split(',') || [\n 'http://localhost:3000',\n ],\n authMode: process.env.AUTH_MODE || 'api_key',\n apiKeySecret: process.env.API_KEY_SECRET || 'development-secret',\n jwtSecret: process.env.JWT_SECRET || 'development-jwt-secret',\n databaseUrl:\n process.env.DATABASE_URL ||\n join(process.cwd(), '.stackmemory', 'railway.db'),\n rateLimitEnabled: process.env.RATE_LIMIT_ENABLED === 'true',\n rateLimitFree: parseInt(process.env.RATE_LIMIT_FREE || '100'),\n enableWebSocket: process.env.ENABLE_WEBSOCKET !== 'false',\n enableAnalytics: process.env.ENABLE_ANALYTICS === 'true',\n};\n\n// Simple in-memory rate limiter\nconst rateLimiter = new Map<string, { count: number; resetTime: number }>();\n\nclass RailwayMCPServer {\n private app: express.Application;\n private httpServer: any;\n private wss?: WebSocketServer;\n private mcpServer!: Server;\n private db!: Database.Database;\n private connections: Map<string, any> = new Map();\n private browserMCP: BrowserMCPIntegration;\n\n constructor() {\n this.app = express();\n this.httpServer = createServer(this.app);\n this.initializeDatabase();\n this.setupMiddleware();\n this.setupRoutes();\n this.setupMCPServer();\n\n // Initialize Browser MCP for Railway\n this.browserMCP = new BrowserMCPIntegration({\n headless: true, // Always headless in production\n defaultViewport: { width: 1280, height: 720 },\n });\n\n if (config.enableWebSocket) {\n this.setupWebSocket();\n }\n }\n\n private initializeDatabase(): void {\n // Use PostgreSQL in production, SQLite for development\n if (\n config.environment === 'production' &&\n config.databaseUrl.startsWith('postgresql://')\n ) {\n console.log('Using PostgreSQL database');\n // In production, we'd use pg client here\n // For now, we'll use SQLite as fallback\n const dbPath = '/tmp/stackmemory.db';\n this.db = new Database(dbPath);\n } else {\n // Create database directory if it doesn't exist\n const dbDir = dirname(config.databaseUrl);\n if (!existsSync(dbDir)) {\n mkdirSync(dbDir, { recursive: true });\n }\n this.db = new Database(config.databaseUrl);\n }\n\n // Initialize tables\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS contexts (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n project_id TEXT NOT NULL,\n content TEXT NOT NULL,\n type TEXT DEFAULT 'general',\n metadata TEXT DEFAULT '{}',\n created_at DATETIME DEFAULT CURRENT_TIMESTAMP,\n updated_at DATETIME DEFAULT CURRENT_TIMESTAMP\n );\n\n CREATE TABLE IF NOT EXISTS api_keys (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n key_hash TEXT UNIQUE NOT NULL,\n user_id TEXT NOT NULL,\n name TEXT,\n created_at DATETIME DEFAULT CURRENT_TIMESTAMP,\n last_used DATETIME,\n revoked BOOLEAN DEFAULT 0\n );\n\n CREATE INDEX IF NOT EXISTS idx_contexts_project ON contexts(project_id);\n CREATE INDEX IF NOT EXISTS idx_api_keys_hash ON api_keys(key_hash);\n `);\n }\n\n private setupMiddleware(): void {\n // CORS\n this.app.use(\n cors({\n origin: config.corsOrigins,\n credentials: true,\n })\n );\n\n // Body parsing\n this.app.use(express.json({ limit: '10mb' }));\n\n // Request logging\n this.app.use((req, res, next) => {\n console.log(`${new Date().toISOString()} ${req.method} ${req.path}`);\n next();\n });\n\n // Simple authentication middleware\n this.app.use('/api', this.authenticate.bind(this));\n\n // Rate limiting\n if (config.rateLimitEnabled) {\n this.app.use('/api', this.rateLimit.bind(this));\n }\n }\n\n private authenticate(\n req: express.Request,\n res: express.Response,\n next: express.NextFunction\n ): any {\n // Skip auth for health check\n if (req.path === '/health') {\n return next();\n }\n\n const authHeader = req.headers.authorization;\n\n if (config.authMode === 'api_key') {\n // Simple API key authentication\n if (!authHeader?.startsWith('Bearer ')) {\n return res.status(401).json({ error: 'Missing API key' });\n }\n\n const apiKey = authHeader.substring(7);\n\n // In production, validate against database\n // For now, simple check\n if (apiKey.length < 32) {\n return res.status(403).json({ error: 'Invalid API key' });\n }\n\n (req as any).user = { id: 'api-user', tier: 'free' };\n next();\n } else {\n // OAuth/JWT mode would go here\n next();\n }\n }\n\n private rateLimit(\n req: express.Request,\n res: express.Response,\n next: express.NextFunction\n ): any {\n const userId = (req as any).user?.id || req.ip;\n const now = Date.now();\n const windowMs = 15 * 60 * 1000; // 15 minutes\n\n const userLimit = rateLimiter.get(userId);\n\n if (!userLimit || userLimit.resetTime < now) {\n rateLimiter.set(userId, {\n count: 1,\n resetTime: now + windowMs,\n });\n return next();\n }\n\n if (userLimit.count >= config.rateLimitFree) {\n const retryAfter = Math.ceil((userLimit.resetTime - now) / 1000);\n res.setHeader('Retry-After', retryAfter.toString());\n return res.status(429).json({\n error: 'Rate limit exceeded',\n retryAfter,\n });\n }\n\n userLimit.count++;\n next();\n }\n\n private setupRoutes(): void {\n // Health check\n this.app.get('/health', (req, res) => {\n const health = {\n status: 'healthy',\n version: '1.0.0',\n timestamp: new Date().toISOString(),\n uptime: process.uptime(),\n environment: config.environment,\n };\n res.json(health);\n });\n\n // API Routes\n this.app.post('/api/context/save', (req, res) => {\n try {\n const {\n projectId = 'default',\n content,\n type = 'general',\n metadata = {},\n } = req.body;\n\n const stmt = this.db.prepare(`\n INSERT INTO contexts (project_id, content, type, metadata)\n VALUES (?, ?, ?, ?)\n `);\n\n const result = stmt.run(\n projectId,\n content,\n type,\n JSON.stringify(metadata)\n );\n\n res.json({\n success: true,\n id: result.lastInsertRowid,\n });\n } catch (error: any) {\n res.status(500).json({ error: error.message });\n }\n });\n\n this.app.get('/api/context/load', (req, res) => {\n try {\n const { projectId = 'default', limit = 10, offset = 0 } = req.query;\n\n const stmt = this.db.prepare(`\n SELECT * FROM contexts\n WHERE project_id = ?\n ORDER BY created_at DESC\n LIMIT ? OFFSET ?\n `);\n\n const contexts = stmt.all(projectId, limit, offset);\n\n res.json({\n success: true,\n contexts: contexts.map((c: any) => ({\n ...c,\n metadata: JSON.parse(c.metadata || '{}'),\n })),\n });\n } catch (error: any) {\n res.status(500).json({ error: error.message });\n }\n });\n\n // MCP tool execution endpoint\n this.app.post('/api/tools/execute', async (req, res) => {\n try {\n const { tool, params } = req.body;\n\n // Execute MCP tool\n const result = await this.executeMCPTool(tool, params);\n\n res.json({\n success: true,\n result,\n });\n } catch (error: any) {\n res.status(500).json({ error: error.message });\n }\n });\n\n // Analytics endpoint\n if (config.enableAnalytics) {\n this.app.get('/api/analytics', (req, res) => {\n try {\n const { projectId = 'default' } = req.query;\n\n const stats = this.db\n .prepare(\n `\n SELECT \n COUNT(*) as total_contexts,\n COUNT(DISTINCT type) as unique_types,\n MAX(created_at) as last_activity\n FROM contexts\n WHERE project_id = ?\n `\n )\n .get(projectId);\n\n res.json({\n success: true,\n analytics: stats,\n });\n } catch (error: any) {\n res.status(500).json({ error: error.message });\n }\n });\n }\n }\n\n private setupWebSocket(): void {\n this.wss = new WebSocketServer({\n server: this.httpServer,\n path: '/ws',\n });\n\n this.wss.on('connection', (ws, _req) => {\n console.log('WebSocket connection established');\n\n const connectionId = Math.random().toString(36).substring(7);\n this.connections.set(connectionId, ws);\n\n ws.on('message', async (data) => {\n try {\n const message = JSON.parse(data.toString());\n const response = await this.handleWebSocketMessage(message);\n ws.send(JSON.stringify(response));\n } catch (error: any) {\n ws.send(\n JSON.stringify({\n error: error.message,\n })\n );\n }\n });\n\n ws.on('close', () => {\n this.connections.delete(connectionId);\n console.log('WebSocket connection closed');\n });\n });\n }\n\n private async handleWebSocketMessage(message: any): Promise<any> {\n const { type, tool, params } = message;\n\n switch (type) {\n case 'execute':\n return await this.executeMCPTool(tool, params);\n\n case 'ping':\n return { type: 'pong' };\n\n default:\n throw new Error(`Unknown message type: ${type}`);\n }\n }\n\n private async setupMCPServer(): Promise<void> {\n this.mcpServer = new Server(\n {\n name: 'stackmemory-railway',\n version: '1.0.0',\n },\n {\n capabilities: {\n tools: {},\n resources: {},\n },\n }\n );\n\n // Initialize Browser MCP with the server\n await this.browserMCP.initialize(this.mcpServer);\n\n // Register MCP tools\n this.mcpServer.setRequestHandler('tools/list' as any, async () => {\n return {\n tools: [\n {\n name: 'save_context',\n description: 'Save context to StackMemory',\n inputSchema: {\n type: 'object',\n properties: {\n content: { type: 'string' },\n type: { type: 'string' },\n },\n },\n },\n {\n name: 'load_context',\n description: 'Load context from StackMemory',\n inputSchema: {\n type: 'object',\n properties: {\n query: { type: 'string' },\n limit: { type: 'number' },\n },\n },\n },\n ],\n };\n });\n\n this.mcpServer.setRequestHandler(\n 'tools/call' as any,\n async (request: any) => {\n const { name, arguments: args } = request.params;\n return await this.executeMCPTool(name, args);\n }\n );\n }\n\n private async executeMCPTool(tool: string, params: any): Promise<any> {\n switch (tool) {\n case 'save_context': {\n const stmt = this.db.prepare(`\n INSERT INTO contexts (project_id, content, type, metadata)\n VALUES (?, ?, ?, ?)\n `);\n const result = stmt.run(\n params.projectId || 'default',\n params.content,\n params.type || 'general',\n JSON.stringify(params.metadata || {})\n );\n return { id: result.lastInsertRowid, success: true };\n }\n\n case 'load_context': {\n const stmt = this.db.prepare(`\n SELECT * FROM contexts\n WHERE project_id = ? AND content LIKE ?\n ORDER BY created_at DESC\n LIMIT ?\n `);\n const contexts = stmt.all(\n params.projectId || 'default',\n `%${params.query || ''}%`,\n params.limit || 10\n );\n return { contexts, success: true };\n }\n\n default:\n throw new Error(`Unknown tool: ${tool}`);\n }\n }\n\n public start(): void {\n this.httpServer.listen(config.port, '0.0.0.0', () => {\n console.log(`\n\uD83D\uDE82 Railway MCP Server Started\n================================\nEnvironment: ${config.environment}\nPort: ${config.port}\nWebSocket: ${config.enableWebSocket ? 'Enabled' : 'Disabled'}\nAnalytics: ${config.enableAnalytics ? 'Enabled' : 'Disabled'}\nRate Limiting: ${config.rateLimitEnabled ? 'Enabled' : 'Disabled'}\nAuth Mode: ${config.authMode}\n================================\nHealth: http://localhost:${config.port}/health\n `);\n });\n }\n}\n\n// Start server\nconst server = new RailwayMCPServer();\nserver.start();\n\n// Graceful shutdown\nprocess.on('SIGTERM', () => {\n console.log('Shutting down gracefully...');\n process.exit(0);\n});\n\nprocess.on('SIGINT', () => {\n console.log('Shutting down...');\n process.exit(0);\n});\n"],
|
|
5
|
-
"mappings": ";AAMA,OAAO,aAAa;AACpB,SAAS,oBAAoB;AAC7B,SAAS,uBAAuB;AAChC,OAAO,UAAU;AACjB,SAAS,cAAc;AAEvB,OAAO,cAAc;AACrB,SAAS,6BAA6B;AACtC,SAAS,MAAM,eAAe;AAC9B,SAAS,YAAY,iBAAiB;
|
|
4
|
+
"sourcesContent": ["#!/usr/bin/env node\n/**\n * Railway MCP Server Entry Point\n * Simplified production server for Railway deployment\n */\n\nimport express from 'express';\nimport { createServer } from 'http';\nimport { WebSocketServer } from 'ws';\nimport cors from 'cors';\nimport { Server } from '@modelcontextprotocol/sdk/server/index.js';\n// WebSocket transport will be handled differently for Railway\nimport Database from 'better-sqlite3';\nimport { BrowserMCPIntegration } from '../../features/browser/browser-mcp.js';\nimport { join, dirname } from 'path';\nimport { existsSync, mkdirSync } from 'fs';\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\n// Configuration\nconst config = {\n port: parseInt(process.env['PORT'] || '3000'),\n environment: process.env['NODE_ENV'] || 'development',\n corsOrigins: process.env['CORS_ORIGINS']?.split(',') || [\n 'http://localhost:3000',\n ],\n authMode: process.env['AUTH_MODE'] || 'api_key',\n apiKeySecret: process.env['API_KEY_SECRET'] || 'development-secret',\n jwtSecret: process.env['JWT_SECRET'] || 'development-jwt-secret',\n databaseUrl:\n process.env['DATABASE_URL'] ||\n join(process.cwd(), '.stackmemory', 'railway.db'),\n rateLimitEnabled: process.env['RATE_LIMIT_ENABLED'] === 'true',\n rateLimitFree: parseInt(process.env['RATE_LIMIT_FREE'] || '100'),\n enableWebSocket: process.env['ENABLE_WEBSOCKET'] !== 'false',\n enableAnalytics: process.env['ENABLE_ANALYTICS'] === 'true',\n};\n\n// Simple in-memory rate limiter\nconst rateLimiter = new Map<string, { count: number; resetTime: number }>();\n\nclass RailwayMCPServer {\n private app: express.Application;\n private httpServer: any;\n private wss?: WebSocketServer;\n private mcpServer!: Server;\n private db!: Database.Database;\n private connections: Map<string, any> = new Map();\n private browserMCP: BrowserMCPIntegration;\n\n constructor() {\n this.app = express();\n this.httpServer = createServer(this.app);\n this.initializeDatabase();\n this.setupMiddleware();\n this.setupRoutes();\n this.setupMCPServer();\n\n // Initialize Browser MCP for Railway\n this.browserMCP = new BrowserMCPIntegration({\n headless: true, // Always headless in production\n defaultViewport: { width: 1280, height: 720 },\n });\n\n if (config.enableWebSocket) {\n this.setupWebSocket();\n }\n }\n\n private initializeDatabase(): void {\n // Use PostgreSQL in production, SQLite for development\n if (\n config.environment === 'production' &&\n config.databaseUrl.startsWith('postgresql://')\n ) {\n console.log('Using PostgreSQL database');\n // In production, we'd use pg client here\n // For now, we'll use SQLite as fallback\n const dbPath = '/tmp/stackmemory.db';\n this.db = new Database(dbPath);\n } else {\n // Create database directory if it doesn't exist\n const dbDir = dirname(config.databaseUrl);\n if (!existsSync(dbDir)) {\n mkdirSync(dbDir, { recursive: true });\n }\n this.db = new Database(config.databaseUrl);\n }\n\n // Initialize tables\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS contexts (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n project_id TEXT NOT NULL,\n content TEXT NOT NULL,\n type TEXT DEFAULT 'general',\n metadata TEXT DEFAULT '{}',\n created_at DATETIME DEFAULT CURRENT_TIMESTAMP,\n updated_at DATETIME DEFAULT CURRENT_TIMESTAMP\n );\n\n CREATE TABLE IF NOT EXISTS api_keys (\n id INTEGER PRIMARY KEY AUTOINCREMENT,\n key_hash TEXT UNIQUE NOT NULL,\n user_id TEXT NOT NULL,\n name TEXT,\n created_at DATETIME DEFAULT CURRENT_TIMESTAMP,\n last_used DATETIME,\n revoked BOOLEAN DEFAULT 0\n );\n\n CREATE INDEX IF NOT EXISTS idx_contexts_project ON contexts(project_id);\n CREATE INDEX IF NOT EXISTS idx_api_keys_hash ON api_keys(key_hash);\n `);\n }\n\n private setupMiddleware(): void {\n // CORS\n this.app.use(\n cors({\n origin: config.corsOrigins,\n credentials: true,\n })\n );\n\n // Body parsing\n this.app.use(express.json({ limit: '10mb' }));\n\n // Request logging\n this.app.use((req, res, next) => {\n console.log(`${new Date().toISOString()} ${req.method} ${req.path}`);\n next();\n });\n\n // Simple authentication middleware\n this.app.use('/api', this.authenticate.bind(this));\n\n // Rate limiting\n if (config.rateLimitEnabled) {\n this.app.use('/api', this.rateLimit.bind(this));\n }\n }\n\n private authenticate(\n req: express.Request,\n res: express.Response,\n next: express.NextFunction\n ): any {\n // Skip auth for health check\n if (req.path === '/health') {\n return next();\n }\n\n const authHeader = req.headers.authorization;\n\n if (config.authMode === 'api_key') {\n // Simple API key authentication\n if (!authHeader?.startsWith('Bearer ')) {\n return res.status(401).json({ error: 'Missing API key' });\n }\n\n const apiKey = authHeader.substring(7);\n\n // In production, validate against database\n // For now, simple check\n if (apiKey.length < 32) {\n return res.status(403).json({ error: 'Invalid API key' });\n }\n\n (req as any).user = { id: 'api-user', tier: 'free' };\n next();\n } else {\n // OAuth/JWT mode would go here\n next();\n }\n }\n\n private rateLimit(\n req: express.Request,\n res: express.Response,\n next: express.NextFunction\n ): any {\n const userId = (req as any).user?.id || req.ip;\n const now = Date.now();\n const windowMs = 15 * 60 * 1000; // 15 minutes\n\n const userLimit = rateLimiter.get(userId);\n\n if (!userLimit || userLimit.resetTime < now) {\n rateLimiter.set(userId, {\n count: 1,\n resetTime: now + windowMs,\n });\n return next();\n }\n\n if (userLimit.count >= config.rateLimitFree) {\n const retryAfter = Math.ceil((userLimit.resetTime - now) / 1000);\n res.setHeader('Retry-After', retryAfter.toString());\n return res.status(429).json({\n error: 'Rate limit exceeded',\n retryAfter,\n });\n }\n\n userLimit.count++;\n next();\n }\n\n private setupRoutes(): void {\n // Health check\n this.app.get('/health', (req, res) => {\n const health = {\n status: 'healthy',\n version: '1.0.0',\n timestamp: new Date().toISOString(),\n uptime: process.uptime(),\n environment: config.environment,\n };\n res.json(health);\n });\n\n // API Routes\n this.app.post('/api/context/save', (req, res) => {\n try {\n const {\n projectId = 'default',\n content,\n type = 'general',\n metadata = {},\n } = req.body;\n\n const stmt = this.db.prepare(`\n INSERT INTO contexts (project_id, content, type, metadata)\n VALUES (?, ?, ?, ?)\n `);\n\n const result = stmt.run(\n projectId,\n content,\n type,\n JSON.stringify(metadata)\n );\n\n res.json({\n success: true,\n id: result.lastInsertRowid,\n });\n } catch (error: any) {\n res.status(500).json({ error: error.message });\n }\n });\n\n this.app.get('/api/context/load', (req, res) => {\n try {\n const { projectId = 'default', limit = 10, offset = 0 } = req.query;\n\n const stmt = this.db.prepare(`\n SELECT * FROM contexts\n WHERE project_id = ?\n ORDER BY created_at DESC\n LIMIT ? OFFSET ?\n `);\n\n const contexts = stmt.all(projectId, limit, offset);\n\n res.json({\n success: true,\n contexts: contexts.map((c: any) => ({\n ...c,\n metadata: JSON.parse(c.metadata || '{}'),\n })),\n });\n } catch (error: any) {\n res.status(500).json({ error: error.message });\n }\n });\n\n // MCP tool execution endpoint\n this.app.post('/api/tools/execute', async (req, res) => {\n try {\n const { tool, params } = req.body;\n\n // Execute MCP tool\n const result = await this.executeMCPTool(tool, params);\n\n res.json({\n success: true,\n result,\n });\n } catch (error: any) {\n res.status(500).json({ error: error.message });\n }\n });\n\n // Analytics endpoint\n if (config.enableAnalytics) {\n this.app.get('/api/analytics', (req, res) => {\n try {\n const { projectId = 'default' } = req.query;\n\n const stats = this.db\n .prepare(\n `\n SELECT \n COUNT(*) as total_contexts,\n COUNT(DISTINCT type) as unique_types,\n MAX(created_at) as last_activity\n FROM contexts\n WHERE project_id = ?\n `\n )\n .get(projectId);\n\n res.json({\n success: true,\n analytics: stats,\n });\n } catch (error: any) {\n res.status(500).json({ error: error.message });\n }\n });\n }\n }\n\n private setupWebSocket(): void {\n this.wss = new WebSocketServer({\n server: this.httpServer,\n path: '/ws',\n });\n\n this.wss.on('connection', (ws, _req) => {\n console.log('WebSocket connection established');\n\n const connectionId = Math.random().toString(36).substring(7);\n this.connections.set(connectionId, ws);\n\n ws.on('message', async (data) => {\n try {\n const message = JSON.parse(data.toString());\n const response = await this.handleWebSocketMessage(message);\n ws.send(JSON.stringify(response));\n } catch (error: any) {\n ws.send(\n JSON.stringify({\n error: error.message,\n })\n );\n }\n });\n\n ws.on('close', () => {\n this.connections.delete(connectionId);\n console.log('WebSocket connection closed');\n });\n });\n }\n\n private async handleWebSocketMessage(message: any): Promise<any> {\n const { type, tool, params } = message;\n\n switch (type) {\n case 'execute':\n return await this.executeMCPTool(tool, params);\n\n case 'ping':\n return { type: 'pong' };\n\n default:\n throw new Error(`Unknown message type: ${type}`);\n }\n }\n\n private async setupMCPServer(): Promise<void> {\n this.mcpServer = new Server(\n {\n name: 'stackmemory-railway',\n version: '1.0.0',\n },\n {\n capabilities: {\n tools: {},\n resources: {},\n },\n }\n );\n\n // Initialize Browser MCP with the server\n await this.browserMCP.initialize(this.mcpServer);\n\n // Register MCP tools\n this.mcpServer.setRequestHandler('tools/list' as any, async () => {\n return {\n tools: [\n {\n name: 'save_context',\n description: 'Save context to StackMemory',\n inputSchema: {\n type: 'object',\n properties: {\n content: { type: 'string' },\n type: { type: 'string' },\n },\n },\n },\n {\n name: 'load_context',\n description: 'Load context from StackMemory',\n inputSchema: {\n type: 'object',\n properties: {\n query: { type: 'string' },\n limit: { type: 'number' },\n },\n },\n },\n ],\n };\n });\n\n this.mcpServer.setRequestHandler(\n 'tools/call' as any,\n async (request: any) => {\n const { name, arguments: args } = request.params;\n return await this.executeMCPTool(name, args);\n }\n );\n }\n\n private async executeMCPTool(tool: string, params: any): Promise<any> {\n switch (tool) {\n case 'save_context': {\n const stmt = this.db.prepare(`\n INSERT INTO contexts (project_id, content, type, metadata)\n VALUES (?, ?, ?, ?)\n `);\n const result = stmt.run(\n params.projectId || 'default',\n params.content,\n params.type || 'general',\n JSON.stringify(params.metadata || {})\n );\n return { id: result.lastInsertRowid, success: true };\n }\n\n case 'load_context': {\n const stmt = this.db.prepare(`\n SELECT * FROM contexts\n WHERE project_id = ? AND content LIKE ?\n ORDER BY created_at DESC\n LIMIT ?\n `);\n const contexts = stmt.all(\n params.projectId || 'default',\n `%${params.query || ''}%`,\n params.limit || 10\n );\n return { contexts, success: true };\n }\n\n default:\n throw new Error(`Unknown tool: ${tool}`);\n }\n }\n\n public start(): void {\n this.httpServer.listen(config.port, '0.0.0.0', () => {\n console.log(`\n\uD83D\uDE82 Railway MCP Server Started\n================================\nEnvironment: ${config.environment}\nPort: ${config.port}\nWebSocket: ${config.enableWebSocket ? 'Enabled' : 'Disabled'}\nAnalytics: ${config.enableAnalytics ? 'Enabled' : 'Disabled'}\nRate Limiting: ${config.rateLimitEnabled ? 'Enabled' : 'Disabled'}\nAuth Mode: ${config.authMode}\n================================\nHealth: http://localhost:${config.port}/health\n `);\n });\n }\n}\n\n// Start server\nconst server = new RailwayMCPServer();\nserver.start();\n\n// Graceful shutdown\nprocess.on('SIGTERM', () => {\n console.log('Shutting down gracefully...');\n process.exit(0);\n});\n\nprocess.on('SIGINT', () => {\n console.log('Shutting down...');\n process.exit(0);\n});\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -1,12 +1,11 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { logger } from "../core/monitoring/logger.js";
|
|
2
2
|
import { readFileSync, writeFileSync, existsSync } from "fs";
|
|
3
3
|
import { join } from "path";
|
|
4
4
|
class ConfigService {
|
|
5
|
-
logger
|
|
5
|
+
// Using singleton logger from monitoring
|
|
6
6
|
config = {};
|
|
7
7
|
configPath;
|
|
8
8
|
constructor() {
|
|
9
|
-
this.logger = new Logger("ConfigService");
|
|
10
9
|
this.configPath = join(process.cwd(), ".stackmemory", "config.json");
|
|
11
10
|
this.loadConfig();
|
|
12
11
|
}
|
|
@@ -15,18 +14,18 @@ class ConfigService {
|
|
|
15
14
|
if (existsSync(this.configPath)) {
|
|
16
15
|
const content = readFileSync(this.configPath, "utf-8");
|
|
17
16
|
this.config = JSON.parse(content);
|
|
18
|
-
|
|
17
|
+
logger.debug("Loaded configuration", this.config);
|
|
19
18
|
}
|
|
20
19
|
} catch (error) {
|
|
21
|
-
|
|
20
|
+
logger.warn("Failed to load configuration, using defaults", error);
|
|
22
21
|
}
|
|
23
22
|
}
|
|
24
23
|
saveConfig() {
|
|
25
24
|
try {
|
|
26
25
|
writeFileSync(this.configPath, JSON.stringify(this.config, null, 2));
|
|
27
|
-
|
|
26
|
+
logger.debug("Saved configuration");
|
|
28
27
|
} catch (error) {
|
|
29
|
-
|
|
28
|
+
logger.error("Failed to save configuration", error);
|
|
30
29
|
}
|
|
31
30
|
}
|
|
32
31
|
async getConfig() {
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../src/services/config-service.ts"],
|
|
4
|
-
"sourcesContent": ["import {
|
|
5
|
-
"mappings": "AAAA,SAAS,cAAc;AACvB,SAAS,cAAc,eAAe,kBAAkB;AACxD,SAAS,YAAY;AA2Bd,MAAM,cAAc;AAAA
|
|
4
|
+
"sourcesContent": ["import { logger } from '../core/monitoring/logger.js';\nimport { readFileSync, writeFileSync, existsSync } from 'fs';\nimport { join } from 'path';\n\nexport interface StackMemoryConfig {\n version?: string;\n projectId?: string;\n integrations?: {\n linear?: {\n enabled?: boolean;\n apiKey?: string;\n teamId?: string;\n projectId?: string;\n syncInterval?: number;\n webhookSecret?: string;\n };\n };\n webhook?: {\n port?: number;\n host?: string;\n ngrokSubdomain?: string;\n };\n features?: {\n autoSync?: boolean;\n realTimeSync?: boolean;\n conflictResolution?: 'manual' | 'auto' | 'prompt';\n };\n}\n\nexport class ConfigService {\n // Using singleton logger from monitoring\n private config: StackMemoryConfig = {};\n private configPath: string;\n\n constructor() {\n // Use singleton logger\n this.configPath = join(process.cwd(), '.stackmemory', 'config.json');\n this.loadConfig();\n }\n\n private loadConfig(): void {\n try {\n if (existsSync(this.configPath)) {\n const content = readFileSync(this.configPath, 'utf-8');\n this.config = JSON.parse(content);\n logger.debug('Loaded configuration', this.config);\n }\n } catch (error: unknown) {\n logger.warn('Failed to load configuration, using defaults', error);\n }\n }\n\n private saveConfig(): void {\n try {\n writeFileSync(this.configPath, JSON.stringify(this.config, null, 2));\n logger.debug('Saved configuration');\n } catch (error: unknown) {\n logger.error('Failed to save configuration', error);\n }\n }\n\n public async getConfig(): Promise<StackMemoryConfig> {\n return this.config;\n }\n\n public async updateConfig(\n updates: Partial<StackMemoryConfig>\n ): Promise<void> {\n this.config = {\n ...this.config,\n ...updates,\n };\n this.saveConfig();\n }\n\n public async getLinearConfig() {\n return this.config.integrations?.linear || {};\n }\n\n public async updateLinearConfig(\n updates: Record<string, unknown>\n ): Promise<void> {\n if (!this.config.integrations) {\n this.config.integrations = {};\n }\n if (!this.config.integrations.linear) {\n this.config.integrations.linear = {};\n }\n\n this.config.integrations.linear = {\n ...this.config.integrations.linear,\n ...updates,\n };\n\n this.saveConfig();\n }\n}\n"],
|
|
5
|
+
"mappings": "AAAA,SAAS,cAAc;AACvB,SAAS,cAAc,eAAe,kBAAkB;AACxD,SAAS,YAAY;AA2Bd,MAAM,cAAc;AAAA;AAAA,EAEjB,SAA4B,CAAC;AAAA,EAC7B;AAAA,EAER,cAAc;AAEZ,SAAK,aAAa,KAAK,QAAQ,IAAI,GAAG,gBAAgB,aAAa;AACnE,SAAK,WAAW;AAAA,EAClB;AAAA,EAEQ,aAAmB;AACzB,QAAI;AACF,UAAI,WAAW,KAAK,UAAU,GAAG;AAC/B,cAAM,UAAU,aAAa,KAAK,YAAY,OAAO;AACrD,aAAK,SAAS,KAAK,MAAM,OAAO;AAChC,eAAO,MAAM,wBAAwB,KAAK,MAAM;AAAA,MAClD;AAAA,IACF,SAAS,OAAgB;AACvB,aAAO,KAAK,gDAAgD,KAAK;AAAA,IACnE;AAAA,EACF;AAAA,EAEQ,aAAmB;AACzB,QAAI;AACF,oBAAc,KAAK,YAAY,KAAK,UAAU,KAAK,QAAQ,MAAM,CAAC,CAAC;AACnE,aAAO,MAAM,qBAAqB;AAAA,IACpC,SAAS,OAAgB;AACvB,aAAO,MAAM,gCAAgC,KAAK;AAAA,IACpD;AAAA,EACF;AAAA,EAEA,MAAa,YAAwC;AACnD,WAAO,KAAK;AAAA,EACd;AAAA,EAEA,MAAa,aACX,SACe;AACf,SAAK,SAAS;AAAA,MACZ,GAAG,KAAK;AAAA,MACR,GAAG;AAAA,IACL;AACA,SAAK,WAAW;AAAA,EAClB;AAAA,EAEA,MAAa,kBAAkB;AAC7B,WAAO,KAAK,OAAO,cAAc,UAAU,CAAC;AAAA,EAC9C;AAAA,EAEA,MAAa,mBACX,SACe;AACf,QAAI,CAAC,KAAK,OAAO,cAAc;AAC7B,WAAK,OAAO,eAAe,CAAC;AAAA,IAC9B;AACA,QAAI,CAAC,KAAK,OAAO,aAAa,QAAQ;AACpC,WAAK,OAAO,aAAa,SAAS,CAAC;AAAA,IACrC;AAEA,SAAK,OAAO,aAAa,SAAS;AAAA,MAChC,GAAG,KAAK,OAAO,aAAa;AAAA,MAC5B,GAAG;AAAA,IACL;AAEA,SAAK,WAAW;AAAA,EAClB;AACF;",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -1,13 +1,12 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { logger } from "../core/monitoring/logger.js";
|
|
2
2
|
import Database from "better-sqlite3";
|
|
3
3
|
import { join } from "path";
|
|
4
4
|
import { existsSync } from "fs";
|
|
5
5
|
class ContextService {
|
|
6
|
-
logger
|
|
6
|
+
// Using singleton logger from monitoring
|
|
7
7
|
db = null;
|
|
8
8
|
tasks = /* @__PURE__ */ new Map();
|
|
9
9
|
constructor() {
|
|
10
|
-
this.logger = new Logger("ContextService");
|
|
11
10
|
this.initializeDatabase();
|
|
12
11
|
}
|
|
13
12
|
initializeDatabase() {
|
|
@@ -18,7 +17,7 @@ class ContextService {
|
|
|
18
17
|
this.loadTasksFromDatabase();
|
|
19
18
|
}
|
|
20
19
|
} catch (error) {
|
|
21
|
-
|
|
20
|
+
logger.warn(
|
|
22
21
|
"Could not connect to database, using in-memory storage",
|
|
23
22
|
error
|
|
24
23
|
);
|
|
@@ -62,9 +61,9 @@ class ContextService {
|
|
|
62
61
|
};
|
|
63
62
|
this.tasks.set(task.id, task);
|
|
64
63
|
}
|
|
65
|
-
|
|
64
|
+
logger.info(`Loaded ${rows.length} tasks from database`);
|
|
66
65
|
} catch (error) {
|
|
67
|
-
|
|
66
|
+
logger.error("Failed to load tasks from database", error);
|
|
68
67
|
}
|
|
69
68
|
}
|
|
70
69
|
async getTask(id) {
|
|
@@ -120,10 +119,10 @@ class ContextService {
|
|
|
120
119
|
task.updatedAt.getTime()
|
|
121
120
|
);
|
|
122
121
|
} catch (error) {
|
|
123
|
-
|
|
122
|
+
logger.error("Failed to persist task to database", error);
|
|
124
123
|
}
|
|
125
124
|
}
|
|
126
|
-
|
|
125
|
+
logger.debug(`Created task: ${task.id} - ${task.title}`);
|
|
127
126
|
return task;
|
|
128
127
|
}
|
|
129
128
|
async updateTask(id, updates) {
|
|
@@ -160,10 +159,10 @@ class ContextService {
|
|
|
160
159
|
id
|
|
161
160
|
);
|
|
162
161
|
} catch (error) {
|
|
163
|
-
|
|
162
|
+
logger.error("Failed to update task in database", error);
|
|
164
163
|
}
|
|
165
164
|
}
|
|
166
|
-
|
|
165
|
+
logger.debug(`Updated task: ${id} - ${updatedTask.title}`);
|
|
167
166
|
return updatedTask;
|
|
168
167
|
}
|
|
169
168
|
async deleteTask(id) {
|
|
@@ -174,10 +173,10 @@ class ContextService {
|
|
|
174
173
|
const stmt = this.db.prepare("DELETE FROM tasks WHERE id = ?");
|
|
175
174
|
stmt.run(id);
|
|
176
175
|
} catch (error) {
|
|
177
|
-
|
|
176
|
+
logger.error("Failed to delete task from database", error);
|
|
178
177
|
}
|
|
179
178
|
}
|
|
180
|
-
|
|
179
|
+
logger.debug(`Deleted task: ${id}`);
|
|
181
180
|
}
|
|
182
181
|
return deleted;
|
|
183
182
|
}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../src/services/context-service.ts"],
|
|
4
|
-
"sourcesContent": ["import { Task } from '../types/task.js';\n// TaskStatus and TaskPriority will be used in future implementations\n// import { TaskStatus, TaskPriority } from '../types/task.js';\nimport {
|
|
5
|
-
"mappings": "AAGA,SAAS,cAAc;AACvB,OAAO,cAAc;AACrB,SAAS,YAAY;AACrB,SAAS,kBAAkB;AAiBpB,MAAM,eAAe;AAAA
|
|
4
|
+
"sourcesContent": ["import { Task } from '../types/task.js';\n// TaskStatus and TaskPriority will be used in future implementations\n// import { TaskStatus, TaskPriority } from '../types/task.js';\nimport { logger } from '../core/monitoring/logger.js';\nimport Database from 'better-sqlite3';\nimport { join } from 'path';\nimport { existsSync } from 'fs';\n\ninterface TaskRow {\n id: string;\n title: string;\n description?: string;\n status?: string;\n priority?: string;\n tags?: string;\n external_id?: string;\n external_identifier?: string;\n external_url?: string;\n metadata?: string;\n created_at: number;\n updated_at: number;\n}\n\nexport class ContextService {\n // Using singleton logger from monitoring\n private db: Database.Database | null = null;\n private tasks: Map<string, Task> = new Map();\n\n constructor() {\n // Use singleton logger\n this.initializeDatabase();\n }\n\n private initializeDatabase(): void {\n try {\n const dbPath = join(process.cwd(), '.stackmemory', 'context.db');\n if (existsSync(dbPath)) {\n this.db = new Database(dbPath);\n this.loadTasksFromDatabase();\n }\n } catch (error: unknown) {\n logger.warn(\n 'Could not connect to database, using in-memory storage',\n error\n );\n }\n }\n\n private loadTasksFromDatabase(): void {\n if (!this.db) return;\n\n try {\n // Create tasks table if it doesn't exist\n this.db.exec(`\n CREATE TABLE IF NOT EXISTS tasks (\n id TEXT PRIMARY KEY,\n title TEXT NOT NULL,\n description TEXT,\n status TEXT DEFAULT 'todo',\n priority TEXT,\n tags TEXT,\n external_id TEXT,\n external_identifier TEXT,\n external_url TEXT,\n metadata TEXT,\n created_at INTEGER,\n updated_at INTEGER\n )\n `);\n\n // Load all tasks from database\n const stmt = this.db.prepare('SELECT * FROM tasks');\n const rows = stmt.all() as TaskRow[];\n\n for (const row of rows) {\n const task: Task = {\n id: row.id,\n title: row.title,\n description: row.description || '',\n status: row.status || 'todo',\n priority: row.priority || undefined,\n tags: row.tags ? JSON.parse(row.tags) : [],\n externalId: row.external_id || undefined,\n externalIdentifier: row.external_identifier || undefined,\n externalUrl: row.external_url || undefined,\n metadata: row.metadata ? JSON.parse(row.metadata) : undefined,\n createdAt: new Date(row.created_at),\n updatedAt: new Date(row.updated_at),\n };\n this.tasks.set(task.id, task);\n }\n\n logger.info(`Loaded ${rows.length} tasks from database`);\n } catch (error: unknown) {\n logger.error('Failed to load tasks from database', error);\n }\n }\n\n public async getTask(id: string): Promise<Task | null> {\n return this.tasks.get(id) || null;\n }\n\n public async getTaskByExternalId(externalId: string): Promise<Task | null> {\n for (const task of this.tasks.values()) {\n if (task.externalId === externalId) {\n return task;\n }\n }\n return null;\n }\n\n public async getAllTasks(): Promise<Task[]> {\n return Array.from(this.tasks.values());\n }\n\n public async createTask(taskData: Partial<Task>): Promise<Task> {\n const task: Task = {\n id: this.generateId(),\n title: taskData.title || 'Untitled Task',\n description: taskData.description || '',\n status: taskData.status || 'todo',\n priority: taskData.priority,\n tags: taskData.tags || [],\n externalId: taskData.externalId,\n externalIdentifier: taskData.externalIdentifier,\n externalUrl: taskData.externalUrl,\n metadata: taskData.metadata,\n createdAt: taskData.createdAt || new Date(),\n updatedAt: taskData.updatedAt || new Date(),\n };\n\n this.tasks.set(task.id, task);\n\n // Persist to database if available\n if (this.db) {\n try {\n const stmt = this.db.prepare(`\n INSERT INTO tasks (id, title, description, status, priority, tags, \n external_id, external_identifier, external_url, \n metadata, created_at, updated_at)\n VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)\n `);\n stmt.run(\n task.id,\n task.title,\n task.description,\n task.status,\n task.priority || null,\n JSON.stringify(task.tags),\n task.externalId || null,\n task.externalIdentifier || null,\n task.externalUrl || null,\n task.metadata ? JSON.stringify(task.metadata) : null,\n task.createdAt.getTime(),\n task.updatedAt.getTime()\n );\n } catch (error: unknown) {\n logger.error('Failed to persist task to database', error);\n }\n }\n\n logger.debug(`Created task: ${task.id} - ${task.title}`);\n return task;\n }\n\n public async updateTask(\n id: string,\n updates: Partial<Task>\n ): Promise<Task | null> {\n const task = this.tasks.get(id);\n if (!task) {\n return null;\n }\n\n const updatedTask = {\n ...task,\n ...updates,\n updatedAt: new Date(),\n };\n\n this.tasks.set(id, updatedTask);\n\n // Update in database if available\n if (this.db) {\n try {\n const stmt = this.db.prepare(`\n UPDATE tasks SET title = ?, description = ?, status = ?, \n priority = ?, tags = ?, external_id = ?, \n external_identifier = ?, external_url = ?, \n metadata = ?, updated_at = ?\n WHERE id = ?\n `);\n stmt.run(\n updatedTask.title,\n updatedTask.description,\n updatedTask.status,\n updatedTask.priority || null,\n JSON.stringify(updatedTask.tags),\n updatedTask.externalId || null,\n updatedTask.externalIdentifier || null,\n updatedTask.externalUrl || null,\n updatedTask.metadata ? JSON.stringify(updatedTask.metadata) : null,\n updatedTask.updatedAt.getTime(),\n id\n );\n } catch (error: unknown) {\n logger.error('Failed to update task in database', error);\n }\n }\n\n logger.debug(`Updated task: ${id} - ${updatedTask.title}`);\n return updatedTask;\n }\n\n public async deleteTask(id: string): Promise<boolean> {\n const deleted = this.tasks.delete(id);\n\n if (deleted) {\n // Delete from database if available\n if (this.db) {\n try {\n const stmt = this.db.prepare('DELETE FROM tasks WHERE id = ?');\n stmt.run(id);\n } catch (error: unknown) {\n logger.error('Failed to delete task from database', error);\n }\n }\n logger.debug(`Deleted task: ${id}`);\n }\n return deleted;\n }\n\n private generateId(): string {\n return 'task_' + Date.now() + '_' + Math.random().toString(36).substr(2, 9);\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": []
|
|
7
7
|
}
|
|
@@ -1,4 +1,7 @@
|
|
|
1
1
|
import { logger } from "../core/monitoring/logger.js";
|
|
2
|
+
import {
|
|
3
|
+
RepoIngestionSkill
|
|
4
|
+
} from "./repo-ingestion-skill.js";
|
|
2
5
|
import * as fs from "fs";
|
|
3
6
|
import * as path from "path";
|
|
4
7
|
import * as os from "os";
|
|
@@ -538,7 +541,10 @@ class ArchaeologistSkill {
|
|
|
538
541
|
if (!timeline.has(date)) {
|
|
539
542
|
timeline.set(date, []);
|
|
540
543
|
}
|
|
541
|
-
timeline.get(date)
|
|
544
|
+
const dateItems = timeline.get(date);
|
|
545
|
+
if (dateItems) {
|
|
546
|
+
dateItems.push(result);
|
|
547
|
+
}
|
|
542
548
|
});
|
|
543
549
|
return Array.from(timeline.entries()).map(([date, items]) => ({
|
|
544
550
|
date,
|
|
@@ -590,7 +596,6 @@ class ArchaeologistSkill {
|
|
|
590
596
|
}
|
|
591
597
|
}
|
|
592
598
|
class ClaudeSkillsManager {
|
|
593
|
-
// DashboardLauncherSkill
|
|
594
599
|
constructor(context) {
|
|
595
600
|
this.context = context;
|
|
596
601
|
this.handoffSkill = new HandoffSkill(context);
|
|
@@ -602,11 +607,28 @@ class ClaudeSkillsManager {
|
|
|
602
607
|
logger.warn("Dashboard auto-launch failed:", error);
|
|
603
608
|
});
|
|
604
609
|
});
|
|
610
|
+
const chromaConfig = {
|
|
611
|
+
apiKey: process.env["CHROMADB_API_KEY"] || "",
|
|
612
|
+
tenant: process.env["CHROMADB_TENANT"] || "",
|
|
613
|
+
database: process.env["CHROMADB_DATABASE"] || "stackmemory",
|
|
614
|
+
collectionName: process.env["CHROMADB_COLLECTION"] || "stackmemory_repos"
|
|
615
|
+
};
|
|
616
|
+
if (chromaConfig.apiKey && chromaConfig.tenant) {
|
|
617
|
+
this.repoIngestionSkill = new RepoIngestionSkill(
|
|
618
|
+
chromaConfig,
|
|
619
|
+
context.userId,
|
|
620
|
+
process.env["CHROMADB_TEAM_ID"]
|
|
621
|
+
);
|
|
622
|
+
this.repoIngestionSkill.initialize().catch((error) => {
|
|
623
|
+
logger.warn("Repo ingestion skill initialization failed:", error);
|
|
624
|
+
});
|
|
625
|
+
}
|
|
605
626
|
}
|
|
606
627
|
handoffSkill;
|
|
607
628
|
checkpointSkill;
|
|
608
629
|
archaeologistSkill;
|
|
609
630
|
dashboardLauncher;
|
|
631
|
+
repoIngestionSkill = null;
|
|
610
632
|
async executeSkill(skillName, args, options) {
|
|
611
633
|
switch (skillName) {
|
|
612
634
|
case "handoff":
|
|
@@ -630,6 +652,64 @@ class ClaudeSkillsManager {
|
|
|
630
652
|
}
|
|
631
653
|
case "dig":
|
|
632
654
|
return this.archaeologistSkill.dig(args[0], options);
|
|
655
|
+
case "repo":
|
|
656
|
+
case "ingest":
|
|
657
|
+
if (!this.repoIngestionSkill) {
|
|
658
|
+
return {
|
|
659
|
+
success: false,
|
|
660
|
+
message: "Repo ingestion skill not initialized. Please configure ChromaDB."
|
|
661
|
+
};
|
|
662
|
+
}
|
|
663
|
+
const repoCommand = args[0];
|
|
664
|
+
switch (repoCommand) {
|
|
665
|
+
case "ingest":
|
|
666
|
+
const repoPath = args[1] || process.cwd();
|
|
667
|
+
const repoName = args[2] || path.basename(repoPath);
|
|
668
|
+
return await this.repoIngestionSkill.ingestRepository(
|
|
669
|
+
repoPath,
|
|
670
|
+
repoName,
|
|
671
|
+
options
|
|
672
|
+
);
|
|
673
|
+
case "update":
|
|
674
|
+
const updatePath = args[1] || process.cwd();
|
|
675
|
+
const updateName = args[2] || path.basename(updatePath);
|
|
676
|
+
return await this.repoIngestionSkill.updateRepository(
|
|
677
|
+
updatePath,
|
|
678
|
+
updateName,
|
|
679
|
+
options
|
|
680
|
+
);
|
|
681
|
+
case "search":
|
|
682
|
+
const query = args[1];
|
|
683
|
+
if (!query) {
|
|
684
|
+
return {
|
|
685
|
+
success: false,
|
|
686
|
+
message: "Search query required"
|
|
687
|
+
};
|
|
688
|
+
}
|
|
689
|
+
const results = await this.repoIngestionSkill.searchCode(query, {
|
|
690
|
+
repoName: options?.repoName,
|
|
691
|
+
language: options?.language,
|
|
692
|
+
limit: options?.limit,
|
|
693
|
+
includeContext: options?.includeContext
|
|
694
|
+
});
|
|
695
|
+
return {
|
|
696
|
+
success: true,
|
|
697
|
+
message: `Found ${results.length} results`,
|
|
698
|
+
data: results
|
|
699
|
+
};
|
|
700
|
+
case "stats":
|
|
701
|
+
const stats = await this.repoIngestionSkill.getRepoStats(args[1]);
|
|
702
|
+
return {
|
|
703
|
+
success: true,
|
|
704
|
+
message: "Repository statistics",
|
|
705
|
+
data: stats
|
|
706
|
+
};
|
|
707
|
+
default:
|
|
708
|
+
return {
|
|
709
|
+
success: false,
|
|
710
|
+
message: `Unknown repo command: ${repoCommand}. Use: ingest, update, search, or stats`
|
|
711
|
+
};
|
|
712
|
+
}
|
|
633
713
|
case "dashboard":
|
|
634
714
|
const dashboardCmd = args[0];
|
|
635
715
|
if (!this.dashboardLauncher) {
|
|
@@ -668,7 +748,11 @@ class ClaudeSkillsManager {
|
|
|
668
748
|
}
|
|
669
749
|
}
|
|
670
750
|
getAvailableSkills() {
|
|
671
|
-
|
|
751
|
+
const skills = ["handoff", "checkpoint", "dig", "dashboard"];
|
|
752
|
+
if (this.repoIngestionSkill) {
|
|
753
|
+
skills.push("repo");
|
|
754
|
+
}
|
|
755
|
+
return skills;
|
|
672
756
|
}
|
|
673
757
|
getSkillHelp(skillName) {
|
|
674
758
|
switch (skillName) {
|
|
@@ -697,6 +781,27 @@ Launch the StackMemory web dashboard for real-time monitoring
|
|
|
697
781
|
- launch: Start the web dashboard and open in browser (default)
|
|
698
782
|
- stop: Stop the dashboard server
|
|
699
783
|
Auto-launches on new sessions when configured
|
|
784
|
+
`;
|
|
785
|
+
case "repo":
|
|
786
|
+
return `
|
|
787
|
+
/repo ingest [path] [name] [--incremental] [--include-tests] [--include-docs]
|
|
788
|
+
/repo update [path] [name] [--force-update]
|
|
789
|
+
/repo search "query" [--repo-name name] [--language lang] [--limit n]
|
|
790
|
+
/repo stats [repo-name]
|
|
791
|
+
|
|
792
|
+
Ingest and search code repositories in ChromaDB:
|
|
793
|
+
- ingest: Index a new repository (defaults to current directory)
|
|
794
|
+
- update: Update an existing repository with changes
|
|
795
|
+
- search: Semantic search across ingested code
|
|
796
|
+
- stats: View statistics about ingested repositories
|
|
797
|
+
|
|
798
|
+
Options:
|
|
799
|
+
- --incremental: Only process changed files
|
|
800
|
+
- --include-tests: Include test files in indexing
|
|
801
|
+
- --include-docs: Include documentation files
|
|
802
|
+
- --force-update: Force re-indexing of all files
|
|
803
|
+
- --language: Filter search by programming language
|
|
804
|
+
- --limit: Maximum search results (default: 20)
|
|
700
805
|
`;
|
|
701
806
|
default:
|
|
702
807
|
return `Unknown skill: ${skillName}`;
|