@stackmemoryai/stackmemory 0.3.7 → 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/cli/claude-sm.js +0 -11
- package/dist/cli/claude-sm.js.map +2 -2
- package/dist/cli/codex-sm.js +0 -11
- package/dist/cli/codex-sm.js.map +2 -2
- package/dist/cli/commands/chromadb.js +64 -34
- package/dist/cli/commands/chromadb.js.map +2 -2
- package/dist/cli/commands/clear.js +9 -13
- 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.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 +69 -20
- package/dist/cli/commands/gc.js.map +2 -2
- package/dist/cli/commands/handoff.js.map +2 -2
- package/dist/cli/commands/infinite-storage.js +60 -19
- package/dist/cli/commands/infinite-storage.js.map +2 -2
- package/dist/cli/commands/linear-create.js +36 -8
- package/dist/cli/commands/linear-create.js.map +2 -2
- package/dist/cli/commands/linear-list.js +33 -10
- package/dist/cli/commands/linear-list.js.map +2 -2
- package/dist/cli/commands/linear-migrate.js +17 -4
- package/dist/cli/commands/linear-migrate.js.map +2 -2
- package/dist/cli/commands/linear-test.js +14 -6
- package/dist/cli/commands/linear-test.js.map +2 -2
- package/dist/cli/commands/linear-unified.js +123 -35
- package/dist/cli/commands/linear-unified.js.map +2 -2
- 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/session.js +23 -6
- package/dist/cli/commands/session.js.map +2 -2
- package/dist/cli/commands/skills.js +72 -27
- package/dist/cli/commands/skills.js.map +2 -2
- package/dist/cli/commands/storage.js +108 -38
- package/dist/cli/commands/storage.js.map +2 -2
- package/dist/cli/commands/tui.js.map +2 -2
- package/dist/cli/commands/webhook.js +57 -18
- package/dist/cli/commands/webhook.js.map +2 -2
- package/dist/cli/commands/workflow.js +8 -15
- 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 +0 -11
- package/dist/cli/index.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/context-bridge.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-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 +10 -3
- package/dist/core/context/incremental-gc.js.map +2 -2
- package/dist/core/context/index.js.map +1 -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 +4 -2
- package/dist/core/context/shared-context-layer.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/query-cache.js +19 -9
- package/dist/core/database/query-cache.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/monitoring/logger.js +6 -1
- package/dist/core/monitoring/logger.js.map +2 -2
- package/dist/core/monitoring/metrics.js.map +2 -2
- package/dist/core/monitoring/progress-tracker.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/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 +51 -14
- 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/projects/project-manager.js +14 -20
- package/dist/core/projects/project-manager.js.map +2 -2
- package/dist/core/retrieval/context-retriever.js.map +1 -1
- package/dist/core/retrieval/llm-context-retrieval.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/index.js.map +1 -1
- package/dist/core/session/session-manager.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 +48 -22
- package/dist/core/storage/railway-optimized-storage.js.map +2 -2
- package/dist/core/storage/remote-storage.js +41 -23
- package/dist/core/storage/remote-storage.js.map +2 -2
- package/dist/core/trace/cli-trace-wrapper.js +9 -2
- 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 +25 -8
- package/dist/core/trace/debug-trace.js.map +2 -2
- package/dist/core/trace/index.js +6 -2
- 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 +14 -10
- 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/types.js.map +1 -1
- package/dist/core/utils/compression.js.map +1 -1
- package/dist/core/utils/update-checker.js.map +1 -1
- package/dist/core/worktree/worktree-manager.js +18 -7
- package/dist/core/worktree/worktree-manager.js.map +2 -2
- 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 +1 -1
- 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 +26 -10
- package/dist/features/tui/components/task-board.js.map +2 -2
- package/dist/features/tui/index.js.map +2 -2
- package/dist/features/tui/services/data-service.js +6 -2
- 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 +3 -1
- package/dist/features/tui/services/websocket-client.js.map +2 -2
- package/dist/features/tui/terminal-compat.js +6 -2
- 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 +18 -10
- package/dist/features/web/server/index.js.map +2 -2
- package/dist/integrations/linear/sync-service.js +12 -13
- package/dist/integrations/linear/sync-service.js.map +2 -2
- package/dist/integrations/linear/sync.js +174 -12
- package/dist/integrations/linear/sync.js.map +2 -2
- package/dist/integrations/linear/unified-sync.js +1 -1
- package/dist/integrations/linear/unified-sync.js.map +1 -1
- package/dist/integrations/linear/webhook-server.js +15 -16
- package/dist/integrations/linear/webhook-server.js.map +2 -2
- package/dist/mcp/stackmemory-mcp-server.js +0 -11
- package/dist/mcp/stackmemory-mcp-server.js.map +2 -2
- package/dist/servers/production/auth-middleware.js.map +2 -2
- 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 +4 -2
- 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.map +2 -2
- package/dist/utils/env.js +46 -0
- package/dist/utils/env.js.map +7 -0
- package/dist/utils/logger.js +0 -11
- package/dist/utils/logger.js.map +2 -2
- package/package.json +1 -1
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"version": 3,
|
|
3
3
|
"sources": ["../../../src/servers/production/auth-middleware.ts"],
|
|
4
|
-
"sourcesContent": ["/**\n * Production Authentication Middleware for Runway MCP Server\n * Implements JWT validation with Auth0, refresh tokens, and rate limiting\n */\n\nimport jwt from 'jsonwebtoken';\nimport jwksRsa from 'jwks-rsa';\nimport { Request, Response, NextFunction } from 'express';\nimport { RateLimiterRedis, RateLimiterRes } from 'rate-limiter-flexible';\nimport Redis from 'ioredis';\nimport BetterSqlite3 from 'better-sqlite3';\nimport { logger } from '../../core/monitoring/logger.js';\nimport { metrics } from '../../core/monitoring/metrics.js';\nimport { getUserModel, UserModel, User } from '../../models/user.model.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\n\nexport interface AuthUser {\n id: string;\n email: string;\n sub: string;\n name?: string;\n picture?: string;\n tier: 'free' | 'pro' | 'enterprise';\n organizations?: string[];\n permissions: string[];\n metadata?: Record<string, any>;\n}\n\nexport interface AuthRequest extends Request {\n user?: AuthUser;\n rateLimitInfo?: RateLimiterRes;\n}\n\nexport class AuthMiddleware {\n private jwksClient: jwksRsa.JwksClient;\n private redis: Redis;\n private rateLimiters!: Map<string, RateLimiterRedis>;\n private blacklistedTokens: Set<string> = new Set();\n private userModel: UserModel;\n private db: BetterSqlite3.Database;\n private mockUser?: AuthUser;\n private mockUserInitializing = false;\n\n constructor(\n private config: {\n auth0Domain: string;\n auth0Audience: string;\n redisUrl: string;\n jwtSecret?: string;\n bypassAuth?: boolean; // For testing\n dbPath?: string; // Path to SQLite database\n }\n ) {\n this.redis = new Redis(config.redisUrl);\n\n // Initialize database\n const dbPath =\n config.dbPath || process.env['STACKMEMORY_DB'] || '.stackmemory/auth.db';\n this.db = new BetterSqlite3(dbPath);\n this.userModel = getUserModel(this.db);\n\n this.jwksClient = jwksRsa({\n jwksUri: `https://${config.auth0Domain}/.well-known/jwks.json`,\n cache: true,\n cacheMaxAge: 600000, // 10 minutes\n rateLimit: true,\n jwksRequestsPerMinute: 5,\n });\n\n this.initializeRateLimiters();\n this.setupTokenBlacklistSync();\n }\n\n private initializeRateLimiters(): void {\n // Different rate limits for different tiers\n this.rateLimiters = new Map([\n [\n 'free',\n new RateLimiterRedis({\n storeClient: this.redis,\n keyPrefix: 'rl:free',\n points: 100, // requests\n duration: 900, // per 15 minutes\n blockDuration: 900, // block for 15 minutes\n }),\n ],\n [\n 'pro',\n new RateLimiterRedis({\n storeClient: this.redis,\n keyPrefix: 'rl:pro',\n points: 1000,\n duration: 900,\n blockDuration: 300,\n }),\n ],\n [\n 'enterprise',\n new RateLimiterRedis({\n storeClient: this.redis,\n keyPrefix: 'rl:enterprise',\n points: 10000,\n duration: 900,\n blockDuration: 60,\n }),\n ],\n ]);\n\n // Special rate limiter for auth endpoints\n this.rateLimiters.set(\n 'auth',\n new RateLimiterRedis({\n storeClient: this.redis,\n keyPrefix: 'rl:auth',\n points: 10, // Only 10 auth attempts\n duration: 900,\n blockDuration: 3600, // Block for 1 hour on excessive auth attempts\n })\n );\n }\n\n private setupTokenBlacklistSync(): void {\n // Subscribe to token revocation events\n const subscriber = new Redis(this.config.redisUrl);\n subscriber.subscribe('token:revoked');\n\n subscriber.on('message', (channel, token) => {\n if (channel === 'token:revoked') {\n this.blacklistedTokens.add(token);\n // Clean up old tokens periodically\n if (this.blacklistedTokens.size > 10000) {\n this.blacklistedTokens.clear();\n }\n }\n });\n }\n\n private async getSigningKey(kid: string): Promise<string> {\n return new Promise((resolve, reject) => {\n this.jwksClient.getSigningKey(kid, (err, key) => {\n if (err) {\n reject(err);\n } else {\n const signingKey = key?.getPublicKey();\n if (!signingKey) {\n reject(new Error('No signing key found'));\n } else {\n resolve(signingKey);\n }\n }\n });\n });\n }\n\n /**\n * Main authentication middleware\n */\n public authenticate = async (\n req: AuthRequest,\n res: Response,\n next: NextFunction\n ): Promise<any> => {\n const startTime = Date.now();\n\n try {\n // Bypass auth for health checks\n if (req.path === '/health' || req.path === '/metrics') {\n return next();\n }\n\n // Development bypass\n if (this.config.bypassAuth && process.env['NODE_ENV'] === 'development') {\n req.user = this.getMockUser();\n return next();\n }\n\n // Extract token or API key\n const token = this.extractToken(req);\n const apiKey = this.extractApiKey(req);\n\n if (!token && !apiKey) {\n metrics.increment('auth.missing_credentials');\n return res.status(401).json({\n error: 'Authentication required',\n code: 'MISSING_CREDENTIALS',\n });\n }\n\n // API Key authentication\n if (apiKey) {\n const user = await this.userModel.validateApiKey(apiKey);\n if (!user) {\n metrics.increment('auth.invalid_api_key');\n return res.status(401).json({\n error: 'Invalid API key',\n code: 'INVALID_API_KEY',\n });\n }\n\n // Convert to AuthUser format\n req.user = {\n id: user.id,\n sub: user.sub,\n email: user.email,\n name: user.name,\n picture: user.avatar,\n tier: user.tier,\n permissions: user.permissions,\n organizations: user.organizations.map((org) => org.id),\n metadata: { ...user.metadata, authMethod: 'api_key' },\n };\n\n metrics.increment('auth.api_key_success');\n await metrics.timing('auth.api_key_duration', Date.now() - startTime);\n return next();\n }\n\n // Check blacklist for JWT tokens\n if (token && this.blacklistedTokens.has(token)) {\n metrics.increment('auth.blacklisted_token');\n return res.status(401).json({\n error: 'Token has been revoked',\n code: 'TOKEN_REVOKED',\n });\n }\n\n // Ensure token exists for JWT processing\n if (!token) {\n // This should not happen as we checked earlier, but TypeScript needs this\n return res.status(401).json({\n error: 'No token provided',\n code: 'NO_TOKEN',\n });\n }\n\n // Decode and verify token\n const decoded = jwt.decode(token, { complete: true }) as any;\n if (!decoded) {\n metrics.increment('auth.invalid_token');\n return res.status(401).json({\n error: 'Invalid token format',\n code: 'INVALID_TOKEN',\n });\n }\n\n // Get signing key and verify\n const signingKey = await this.getSigningKey(decoded.header.kid);\n const verified = jwt.verify(token, signingKey, {\n algorithms: ['RS256'],\n audience: this.config.auth0Audience,\n issuer: `https://${this.config.auth0Domain}/`,\n }) as any;\n\n // Load user from database or cache\n const user = await this.loadUser(verified.sub, verified);\n if (!user) {\n metrics.increment('auth.user_not_found');\n return res.status(403).json({\n error: 'User not found',\n code: 'USER_NOT_FOUND',\n });\n }\n\n // Check user suspension\n if (user.metadata?.suspended) {\n metrics.increment('auth.user_suspended');\n return res.status(403).json({\n error: 'Account suspended',\n code: 'ACCOUNT_SUSPENDED',\n });\n }\n\n // Apply rate limiting\n const rateLimiter =\n this.rateLimiters.get(user.tier) || this.rateLimiters.get('free')!;\n try {\n const rateLimitRes = await rateLimiter.consume(user.id);\n req.rateLimitInfo = rateLimitRes;\n\n // Add rate limit headers\n res.setHeader('X-RateLimit-Limit', rateLimiter.points.toString());\n res.setHeader(\n 'X-RateLimit-Remaining',\n rateLimitRes.remainingPoints.toString()\n );\n res.setHeader(\n 'X-RateLimit-Reset',\n new Date(Date.now() + rateLimitRes.msBeforeNext).toISOString()\n );\n } catch (rateLimitError: any) {\n metrics.increment('auth.rate_limited');\n res.setHeader(\n 'Retry-After',\n Math.round(rateLimitError.msBeforeNext / 1000).toString()\n );\n return res.status(429).json({\n error: 'Too many requests',\n code: 'RATE_LIMITED',\n retryAfter: rateLimitError.msBeforeNext,\n });\n }\n\n // Attach user to request\n req.user = user;\n\n // Track metrics\n metrics.increment('auth.success', { tier: user.tier });\n metrics.timing('auth.duration', Date.now() - startTime);\n\n logger.info('Authentication successful', {\n userId: user.id,\n tier: user.tier,\n path: req.path,\n });\n\n next();\n } catch (error: any) {\n metrics.increment('auth.error');\n logger.error('Authentication error', error);\n\n if (error.name === 'TokenExpiredError') {\n return res.status(401).json({\n error: 'Token expired',\n code: 'TOKEN_EXPIRED',\n });\n }\n\n if (error.name === 'JsonWebTokenError') {\n return res.status(401).json({\n error: 'Invalid token',\n code: 'INVALID_TOKEN',\n });\n }\n\n res.status(500).json({\n error: 'Authentication failed',\n code: 'AUTH_ERROR',\n });\n }\n };\n\n /**\n * WebSocket authentication handler\n */\n public authenticateWebSocket = async (\n token: string\n ): Promise<AuthUser | null> => {\n try {\n const decoded = jwt.decode(token, { complete: true }) as any;\n if (!decoded || this.blacklistedTokens.has(token)) {\n return null;\n }\n\n const signingKey = await this.getSigningKey(decoded.header.kid);\n const verified = jwt.verify(token, signingKey, {\n algorithms: ['RS256'],\n audience: this.config.auth0Audience,\n issuer: `https://${this.config.auth0Domain}/`,\n }) as any;\n\n return await this.loadUser(verified.sub, verified);\n } catch (error: unknown) {\n logger.error(\n 'WebSocket authentication failed',\n error instanceof Error ? error : undefined\n );\n return null;\n }\n };\n\n /**\n * Permission checking middleware\n */\n public requirePermission = (permission: string) => {\n return (req: AuthRequest, res: Response, next: NextFunction) => {\n if (!req.user) {\n return res.status(401).json({\n error: 'Authentication required',\n code: 'NOT_AUTHENTICATED',\n });\n }\n\n if (!req.user.permissions.includes(permission)) {\n metrics.increment('auth.permission_denied', { permission });\n return res.status(403).json({\n error: 'Insufficient permissions',\n code: 'PERMISSION_DENIED',\n required: permission,\n });\n }\n\n return next();\n };\n };\n\n /**\n * Organization access middleware\n */\n public requireOrganization = (\n req: AuthRequest,\n res: Response,\n next: NextFunction\n ) => {\n const orgId = req.params.orgId || req.query.orgId;\n\n if (!req.user || !orgId) {\n return res.status(401).json({\n error: 'Authentication required',\n code: 'NOT_AUTHENTICATED',\n });\n }\n\n if (!req.user.organizations?.includes(orgId as string)) {\n return res.status(403).json({\n error: 'Organization access denied',\n code: 'ORG_ACCESS_DENIED',\n });\n }\n\n return next();\n };\n\n private extractApiKey(req: Request): string | null {\n // Check Authorization header for API key\n const authHeader = req.headers.authorization;\n if (authHeader?.startsWith('Bearer sk-')) {\n return authHeader.substring(7);\n }\n\n // Check X-API-Key header\n const apiKeyHeader = req.headers['x-api-key'] as string;\n if (apiKeyHeader?.startsWith('sk-')) {\n return apiKeyHeader;\n }\n\n // Query parameter support removed for security reasons\n // API keys should only be sent via headers to prevent:\n // - URL logging exposure\n // - Browser history leakage\n // - Referer header transmission\n\n return null;\n }\n\n private extractToken(req: Request): string | null {\n const authHeader = req.headers.authorization;\n if (\n authHeader?.startsWith('Bearer ') &&\n !authHeader.startsWith('Bearer sk-')\n ) {\n return authHeader.substring(7);\n }\n\n // Also check cookie for web clients\n return req.cookies?.access_token || null;\n }\n\n private async loadUser(\n sub: string,\n tokenPayload?: any\n ): Promise<AuthUser | null> {\n // Try cache first\n const cached = await this.redis.get(`user:${sub}`);\n if (cached) {\n const cachedUser = JSON.parse(cached);\n // Update last login time in background\n this.userModel\n .updateLastLogin(cachedUser.id)\n .catch((err) => logger.error('Failed to update last login', err));\n return cachedUser;\n }\n\n // Load from database\n let dbUser = await this.userModel.findUserBySub(sub);\n\n // If user doesn't exist, create from token payload\n if (!dbUser && tokenPayload) {\n dbUser = await this.userModel.createUser({\n sub,\n email: tokenPayload.email || `${sub}@auth.local`,\n name: tokenPayload.name,\n avatar: tokenPayload.picture,\n tier: this.determineTier(tokenPayload),\n permissions: this.determinePermissions(tokenPayload),\n organizations: this.extractOrganizations(tokenPayload),\n metadata: {\n auth0: tokenPayload,\n signupSource: 'auth0',\n createdVia: 'auth-middleware',\n },\n });\n logger.info('Auto-created user from auth token', {\n sub,\n email: dbUser.email,\n });\n }\n\n if (!dbUser) {\n return null;\n }\n\n // Update last login\n await this.userModel.updateLastLogin(dbUser.id);\n\n // Convert to AuthUser format\n const user: AuthUser = {\n id: dbUser.id,\n sub: dbUser.sub,\n email: dbUser.email,\n name: dbUser.name,\n picture: dbUser.avatar,\n tier: dbUser.tier,\n permissions: dbUser.permissions,\n organizations: dbUser.organizations.map((org) => org.id),\n metadata: dbUser.metadata,\n };\n\n // Cache for 5 minutes\n await this.redis.setex(`user:${sub}`, 300, JSON.stringify(user));\n\n return user;\n }\n\n private determineTier(tokenPayload: any): 'free' | 'pro' | 'enterprise' {\n // Check custom claims or metadata\n if (tokenPayload['https://stackmemory.ai/tier']) {\n return tokenPayload['https://stackmemory.ai/tier'];\n }\n\n // Check for subscription info\n if (tokenPayload.subscription?.plan) {\n const plan = tokenPayload.subscription.plan.toLowerCase();\n if (plan.includes('enterprise')) return 'enterprise';\n if (plan.includes('pro') || plan.includes('premium')) return 'pro';\n }\n\n // Default to free\n return 'free';\n }\n\n private determinePermissions(tokenPayload: any): string[] {\n const permissions: string[] = ['read', 'write'];\n\n // Check custom permissions claim\n if (tokenPayload['https://stackmemory.ai/permissions']) {\n return tokenPayload['https://stackmemory.ai/permissions'];\n }\n\n // Check standard permissions\n if (tokenPayload.permissions && Array.isArray(tokenPayload.permissions)) {\n return tokenPayload.permissions;\n }\n\n // Check roles\n if (tokenPayload.roles && Array.isArray(tokenPayload.roles)) {\n if (tokenPayload.roles.includes('admin')) {\n permissions.push('admin', 'delete');\n }\n if (tokenPayload.roles.includes('moderator')) {\n permissions.push('moderate');\n }\n }\n\n return permissions;\n }\n\n private extractOrganizations(\n tokenPayload: any\n ): Array<{ id: string; name: string; role: string }> {\n const orgs: Array<{ id: string; name: string; role: string }> = [];\n\n // Check custom organization claim\n if (tokenPayload['https://stackmemory.ai/organizations']) {\n return tokenPayload['https://stackmemory.ai/organizations'];\n }\n\n // Check Auth0 organizations\n if (tokenPayload.org_id) {\n orgs.push({\n id: tokenPayload.org_id,\n name: tokenPayload.org_name || tokenPayload.org_id,\n role: tokenPayload.org_role || 'member',\n });\n }\n\n return orgs;\n }\n\n private async initializeMockUser(): Promise<AuthUser> {\n const mockSub = 'dev-sub';\n\n // Check if user exists in database\n let dbUser = await this.userModel.findUserBySub(mockSub);\n\n if (!dbUser) {\n // Create mock user in database\n dbUser = await this.userModel.createUser({\n sub: mockSub,\n email: 'dev@stackmemory.local',\n name: 'Development User',\n tier: 'enterprise',\n permissions: ['read', 'write', 'admin', 'delete'],\n organizations: [\n {\n id: 'dev-org',\n name: 'Development Organization',\n role: 'admin',\n },\n ],\n metadata: {\n isDevelopmentUser: true,\n createdAt: new Date().toISOString(),\n },\n });\n logger.info('Created development mock user');\n }\n\n return {\n id: dbUser.id,\n sub: dbUser.sub,\n email: dbUser.email,\n name: dbUser.name,\n picture: dbUser.avatar,\n tier: dbUser.tier,\n permissions: dbUser.permissions,\n organizations: dbUser.organizations.map((org) => org.id),\n metadata: dbUser.metadata,\n };\n }\n\n private getMockUser(): AuthUser {\n // Return cached mock user if available\n if (this.mockUser) {\n return this.mockUser;\n }\n\n // Initialize mock user synchronously to prevent race conditions\n // This runs during constructor or first use\n if (!this.mockUserInitializing) {\n this.mockUserInitializing = true;\n\n // Initialize asynchronously but return a temporary user immediately\n this.initializeMockUser()\n .then((user) => {\n this.mockUser = user;\n this.mockUserInitializing = false;\n logger.info('Mock user initialized and cached');\n })\n .catch((err) => {\n logger.error('Failed to initialize mock user', err);\n this.mockUserInitializing = false;\n });\n }\n\n // Return temporary mock user while initialization is in progress\n return {\n id: 'temp-dev-user-id',\n sub: 'dev-sub',\n email: 'dev@stackmemory.local',\n name: 'Development User',\n tier: 'enterprise',\n permissions: ['read', 'write', 'admin', 'delete'],\n organizations: ['dev-org'],\n metadata: { temporary: true },\n };\n }\n\n /**\n * Revoke a token (add to blacklist)\n */\n public async revokeToken(token: string): Promise<void> {\n this.blacklistedTokens.add(token);\n await this.redis.publish('token:revoked', token);\n\n // Also store in Redis with TTL matching token expiry\n const decoded = jwt.decode(token) as any;\n if (decoded?.exp) {\n const ttl = decoded.exp - Math.floor(Date.now() / 1000);\n if (ttl > 0) {\n await this.redis.setex(`blacklist:${token}`, ttl, '1');\n }\n }\n }\n\n /**\n * Cleanup resources\n */\n public async close(): Promise<void> {\n await this.redis.quit();\n }\n}\n"],
|
|
5
|
-
"mappings": "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",
|
|
4
|
+
"sourcesContent": ["/**\n * Production Authentication Middleware for Runway MCP Server\n * Implements JWT validation with Auth0, refresh tokens, and rate limiting\n */\n\nimport jwt from 'jsonwebtoken';\nimport jwksRsa from 'jwks-rsa';\nimport { Request, Response, NextFunction } from 'express';\nimport { RateLimiterRedis, RateLimiterRes } from 'rate-limiter-flexible';\nimport Redis from 'ioredis';\nimport BetterSqlite3 from 'better-sqlite3';\nimport { logger } from '../../core/monitoring/logger.js';\nimport { metrics } from '../../core/monitoring/metrics.js';\nimport { getUserModel, UserModel, User } from '../../models/user.model.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 interface AuthUser {\n id: string;\n email: string;\n sub: string;\n name?: string;\n picture?: string;\n tier: 'free' | 'pro' | 'enterprise';\n organizations?: string[];\n permissions: string[];\n metadata?: Record<string, any>;\n}\n\nexport interface AuthRequest extends Request {\n user?: AuthUser;\n rateLimitInfo?: RateLimiterRes;\n}\n\nexport class AuthMiddleware {\n private jwksClient: jwksRsa.JwksClient;\n private redis: Redis;\n private rateLimiters!: Map<string, RateLimiterRedis>;\n private blacklistedTokens: Set<string> = new Set();\n private userModel: UserModel;\n private db: BetterSqlite3.Database;\n private mockUser?: AuthUser;\n private mockUserInitializing = false;\n\n constructor(\n private config: {\n auth0Domain: string;\n auth0Audience: string;\n redisUrl: string;\n jwtSecret?: string;\n bypassAuth?: boolean; // For testing\n dbPath?: string; // Path to SQLite database\n }\n ) {\n this.redis = new Redis(config.redisUrl);\n\n // Initialize database\n const dbPath =\n config.dbPath || process.env['STACKMEMORY_DB'] || '.stackmemory/auth.db';\n this.db = new BetterSqlite3(dbPath);\n this.userModel = getUserModel(this.db);\n\n this.jwksClient = jwksRsa({\n jwksUri: `https://${config.auth0Domain}/.well-known/jwks.json`,\n cache: true,\n cacheMaxAge: 600000, // 10 minutes\n rateLimit: true,\n jwksRequestsPerMinute: 5,\n });\n\n this.initializeRateLimiters();\n this.setupTokenBlacklistSync();\n }\n\n private initializeRateLimiters(): void {\n // Different rate limits for different tiers\n this.rateLimiters = new Map([\n [\n 'free',\n new RateLimiterRedis({\n storeClient: this.redis,\n keyPrefix: 'rl:free',\n points: 100, // requests\n duration: 900, // per 15 minutes\n blockDuration: 900, // block for 15 minutes\n }),\n ],\n [\n 'pro',\n new RateLimiterRedis({\n storeClient: this.redis,\n keyPrefix: 'rl:pro',\n points: 1000,\n duration: 900,\n blockDuration: 300,\n }),\n ],\n [\n 'enterprise',\n new RateLimiterRedis({\n storeClient: this.redis,\n keyPrefix: 'rl:enterprise',\n points: 10000,\n duration: 900,\n blockDuration: 60,\n }),\n ],\n ]);\n\n // Special rate limiter for auth endpoints\n this.rateLimiters.set(\n 'auth',\n new RateLimiterRedis({\n storeClient: this.redis,\n keyPrefix: 'rl:auth',\n points: 10, // Only 10 auth attempts\n duration: 900,\n blockDuration: 3600, // Block for 1 hour on excessive auth attempts\n })\n );\n }\n\n private setupTokenBlacklistSync(): void {\n // Subscribe to token revocation events\n const subscriber = new Redis(this.config.redisUrl);\n subscriber.subscribe('token:revoked');\n\n subscriber.on('message', (channel, token) => {\n if (channel === 'token:revoked') {\n this.blacklistedTokens.add(token);\n // Clean up old tokens periodically\n if (this.blacklistedTokens.size > 10000) {\n this.blacklistedTokens.clear();\n }\n }\n });\n }\n\n private async getSigningKey(kid: string): Promise<string> {\n return new Promise((resolve, reject) => {\n this.jwksClient.getSigningKey(kid, (err, key) => {\n if (err) {\n reject(err);\n } else {\n const signingKey = key?.getPublicKey();\n if (!signingKey) {\n reject(new Error('No signing key found'));\n } else {\n resolve(signingKey);\n }\n }\n });\n });\n }\n\n /**\n * Main authentication middleware\n */\n public authenticate = async (\n req: AuthRequest,\n res: Response,\n next: NextFunction\n ): Promise<any> => {\n const startTime = Date.now();\n\n try {\n // Bypass auth for health checks\n if (req.path === '/health' || req.path === '/metrics') {\n return next();\n }\n\n // Development bypass\n if (this.config.bypassAuth && process.env['NODE_ENV'] === 'development') {\n req.user = this.getMockUser();\n return next();\n }\n\n // Extract token or API key\n const token = this.extractToken(req);\n const apiKey = this.extractApiKey(req);\n\n if (!token && !apiKey) {\n metrics.increment('auth.missing_credentials');\n return res.status(401).json({\n error: 'Authentication required',\n code: 'MISSING_CREDENTIALS',\n });\n }\n\n // API Key authentication\n if (apiKey) {\n const user = await this.userModel.validateApiKey(apiKey);\n if (!user) {\n metrics.increment('auth.invalid_api_key');\n return res.status(401).json({\n error: 'Invalid API key',\n code: 'INVALID_API_KEY',\n });\n }\n\n // Convert to AuthUser format\n req.user = {\n id: user.id,\n sub: user.sub,\n email: user.email,\n name: user.name,\n picture: user.avatar,\n tier: user.tier,\n permissions: user.permissions,\n organizations: user.organizations.map((org) => org.id),\n metadata: { ...user.metadata, authMethod: 'api_key' },\n };\n\n metrics.increment('auth.api_key_success');\n await metrics.timing('auth.api_key_duration', Date.now() - startTime);\n return next();\n }\n\n // Check blacklist for JWT tokens\n if (token && this.blacklistedTokens.has(token)) {\n metrics.increment('auth.blacklisted_token');\n return res.status(401).json({\n error: 'Token has been revoked',\n code: 'TOKEN_REVOKED',\n });\n }\n\n // Ensure token exists for JWT processing\n if (!token) {\n // This should not happen as we checked earlier, but TypeScript needs this\n return res.status(401).json({\n error: 'No token provided',\n code: 'NO_TOKEN',\n });\n }\n\n // Decode and verify token\n const decoded = jwt.decode(token, { complete: true }) as any;\n if (!decoded) {\n metrics.increment('auth.invalid_token');\n return res.status(401).json({\n error: 'Invalid token format',\n code: 'INVALID_TOKEN',\n });\n }\n\n // Get signing key and verify\n const signingKey = await this.getSigningKey(decoded.header.kid);\n const verified = jwt.verify(token, signingKey, {\n algorithms: ['RS256'],\n audience: this.config.auth0Audience,\n issuer: `https://${this.config.auth0Domain}/`,\n }) as any;\n\n // Load user from database or cache\n const user = await this.loadUser(verified.sub, verified);\n if (!user) {\n metrics.increment('auth.user_not_found');\n return res.status(403).json({\n error: 'User not found',\n code: 'USER_NOT_FOUND',\n });\n }\n\n // Check user suspension\n if (user.metadata?.suspended) {\n metrics.increment('auth.user_suspended');\n return res.status(403).json({\n error: 'Account suspended',\n code: 'ACCOUNT_SUSPENDED',\n });\n }\n\n // Apply rate limiting\n const rateLimiter =\n this.rateLimiters.get(user.tier) || this.rateLimiters.get('free')!;\n try {\n const rateLimitRes = await rateLimiter.consume(user.id);\n req.rateLimitInfo = rateLimitRes;\n\n // Add rate limit headers\n res.setHeader('X-RateLimit-Limit', rateLimiter.points.toString());\n res.setHeader(\n 'X-RateLimit-Remaining',\n rateLimitRes.remainingPoints.toString()\n );\n res.setHeader(\n 'X-RateLimit-Reset',\n new Date(Date.now() + rateLimitRes.msBeforeNext).toISOString()\n );\n } catch (rateLimitError: any) {\n metrics.increment('auth.rate_limited');\n res.setHeader(\n 'Retry-After',\n Math.round(rateLimitError.msBeforeNext / 1000).toString()\n );\n return res.status(429).json({\n error: 'Too many requests',\n code: 'RATE_LIMITED',\n retryAfter: rateLimitError.msBeforeNext,\n });\n }\n\n // Attach user to request\n req.user = user;\n\n // Track metrics\n metrics.increment('auth.success', { tier: user.tier });\n metrics.timing('auth.duration', Date.now() - startTime);\n\n logger.info('Authentication successful', {\n userId: user.id,\n tier: user.tier,\n path: req.path,\n });\n\n next();\n } catch (error: any) {\n metrics.increment('auth.error');\n logger.error('Authentication error', error);\n\n if (error.name === 'TokenExpiredError') {\n return res.status(401).json({\n error: 'Token expired',\n code: 'TOKEN_EXPIRED',\n });\n }\n\n if (error.name === 'JsonWebTokenError') {\n return res.status(401).json({\n error: 'Invalid token',\n code: 'INVALID_TOKEN',\n });\n }\n\n res.status(500).json({\n error: 'Authentication failed',\n code: 'AUTH_ERROR',\n });\n }\n };\n\n /**\n * WebSocket authentication handler\n */\n public authenticateWebSocket = async (\n token: string\n ): Promise<AuthUser | null> => {\n try {\n const decoded = jwt.decode(token, { complete: true }) as any;\n if (!decoded || this.blacklistedTokens.has(token)) {\n return null;\n }\n\n const signingKey = await this.getSigningKey(decoded.header.kid);\n const verified = jwt.verify(token, signingKey, {\n algorithms: ['RS256'],\n audience: this.config.auth0Audience,\n issuer: `https://${this.config.auth0Domain}/`,\n }) as any;\n\n return await this.loadUser(verified.sub, verified);\n } catch (error: unknown) {\n logger.error(\n 'WebSocket authentication failed',\n error instanceof Error ? error : undefined\n );\n return null;\n }\n };\n\n /**\n * Permission checking middleware\n */\n public requirePermission = (permission: string) => {\n return (req: AuthRequest, res: Response, next: NextFunction) => {\n if (!req.user) {\n return res.status(401).json({\n error: 'Authentication required',\n code: 'NOT_AUTHENTICATED',\n });\n }\n\n if (!req.user.permissions.includes(permission)) {\n metrics.increment('auth.permission_denied', { permission });\n return res.status(403).json({\n error: 'Insufficient permissions',\n code: 'PERMISSION_DENIED',\n required: permission,\n });\n }\n\n return next();\n };\n };\n\n /**\n * Organization access middleware\n */\n public requireOrganization = (\n req: AuthRequest,\n res: Response,\n next: NextFunction\n ) => {\n const orgId = req.params.orgId || req.query.orgId;\n\n if (!req.user || !orgId) {\n return res.status(401).json({\n error: 'Authentication required',\n code: 'NOT_AUTHENTICATED',\n });\n }\n\n if (!req.user.organizations?.includes(orgId as string)) {\n return res.status(403).json({\n error: 'Organization access denied',\n code: 'ORG_ACCESS_DENIED',\n });\n }\n\n return next();\n };\n\n private extractApiKey(req: Request): string | null {\n // Check Authorization header for API key\n const authHeader = req.headers.authorization;\n if (authHeader?.startsWith('Bearer sk-')) {\n return authHeader.substring(7);\n }\n\n // Check X-API-Key header\n const apiKeyHeader = req.headers['x-api-key'] as string;\n if (apiKeyHeader?.startsWith('sk-')) {\n return apiKeyHeader;\n }\n\n // Query parameter support removed for security reasons\n // API keys should only be sent via headers to prevent:\n // - URL logging exposure\n // - Browser history leakage\n // - Referer header transmission\n\n return null;\n }\n\n private extractToken(req: Request): string | null {\n const authHeader = req.headers.authorization;\n if (\n authHeader?.startsWith('Bearer ') &&\n !authHeader.startsWith('Bearer sk-')\n ) {\n return authHeader.substring(7);\n }\n\n // Also check cookie for web clients\n return req.cookies?.access_token || null;\n }\n\n private async loadUser(\n sub: string,\n tokenPayload?: any\n ): Promise<AuthUser | null> {\n // Try cache first\n const cached = await this.redis.get(`user:${sub}`);\n if (cached) {\n const cachedUser = JSON.parse(cached);\n // Update last login time in background\n this.userModel\n .updateLastLogin(cachedUser.id)\n .catch((err) => logger.error('Failed to update last login', err));\n return cachedUser;\n }\n\n // Load from database\n let dbUser = await this.userModel.findUserBySub(sub);\n\n // If user doesn't exist, create from token payload\n if (!dbUser && tokenPayload) {\n dbUser = await this.userModel.createUser({\n sub,\n email: tokenPayload.email || `${sub}@auth.local`,\n name: tokenPayload.name,\n avatar: tokenPayload.picture,\n tier: this.determineTier(tokenPayload),\n permissions: this.determinePermissions(tokenPayload),\n organizations: this.extractOrganizations(tokenPayload),\n metadata: {\n auth0: tokenPayload,\n signupSource: 'auth0',\n createdVia: 'auth-middleware',\n },\n });\n logger.info('Auto-created user from auth token', {\n sub,\n email: dbUser.email,\n });\n }\n\n if (!dbUser) {\n return null;\n }\n\n // Update last login\n await this.userModel.updateLastLogin(dbUser.id);\n\n // Convert to AuthUser format\n const user: AuthUser = {\n id: dbUser.id,\n sub: dbUser.sub,\n email: dbUser.email,\n name: dbUser.name,\n picture: dbUser.avatar,\n tier: dbUser.tier,\n permissions: dbUser.permissions,\n organizations: dbUser.organizations.map((org) => org.id),\n metadata: dbUser.metadata,\n };\n\n // Cache for 5 minutes\n await this.redis.setex(`user:${sub}`, 300, JSON.stringify(user));\n\n return user;\n }\n\n private determineTier(tokenPayload: any): 'free' | 'pro' | 'enterprise' {\n // Check custom claims or metadata\n if (tokenPayload['https://stackmemory.ai/tier']) {\n return tokenPayload['https://stackmemory.ai/tier'];\n }\n\n // Check for subscription info\n if (tokenPayload.subscription?.plan) {\n const plan = tokenPayload.subscription.plan.toLowerCase();\n if (plan.includes('enterprise')) return 'enterprise';\n if (plan.includes('pro') || plan.includes('premium')) return 'pro';\n }\n\n // Default to free\n return 'free';\n }\n\n private determinePermissions(tokenPayload: any): string[] {\n const permissions: string[] = ['read', 'write'];\n\n // Check custom permissions claim\n if (tokenPayload['https://stackmemory.ai/permissions']) {\n return tokenPayload['https://stackmemory.ai/permissions'];\n }\n\n // Check standard permissions\n if (tokenPayload.permissions && Array.isArray(tokenPayload.permissions)) {\n return tokenPayload.permissions;\n }\n\n // Check roles\n if (tokenPayload.roles && Array.isArray(tokenPayload.roles)) {\n if (tokenPayload.roles.includes('admin')) {\n permissions.push('admin', 'delete');\n }\n if (tokenPayload.roles.includes('moderator')) {\n permissions.push('moderate');\n }\n }\n\n return permissions;\n }\n\n private extractOrganizations(\n tokenPayload: any\n ): Array<{ id: string; name: string; role: string }> {\n const orgs: Array<{ id: string; name: string; role: string }> = [];\n\n // Check custom organization claim\n if (tokenPayload['https://stackmemory.ai/organizations']) {\n return tokenPayload['https://stackmemory.ai/organizations'];\n }\n\n // Check Auth0 organizations\n if (tokenPayload.org_id) {\n orgs.push({\n id: tokenPayload.org_id,\n name: tokenPayload.org_name || tokenPayload.org_id,\n role: tokenPayload.org_role || 'member',\n });\n }\n\n return orgs;\n }\n\n private async initializeMockUser(): Promise<AuthUser> {\n const mockSub = 'dev-sub';\n\n // Check if user exists in database\n let dbUser = await this.userModel.findUserBySub(mockSub);\n\n if (!dbUser) {\n // Create mock user in database\n dbUser = await this.userModel.createUser({\n sub: mockSub,\n email: 'dev@stackmemory.local',\n name: 'Development User',\n tier: 'enterprise',\n permissions: ['read', 'write', 'admin', 'delete'],\n organizations: [\n {\n id: 'dev-org',\n name: 'Development Organization',\n role: 'admin',\n },\n ],\n metadata: {\n isDevelopmentUser: true,\n createdAt: new Date().toISOString(),\n },\n });\n logger.info('Created development mock user');\n }\n\n return {\n id: dbUser.id,\n sub: dbUser.sub,\n email: dbUser.email,\n name: dbUser.name,\n picture: dbUser.avatar,\n tier: dbUser.tier,\n permissions: dbUser.permissions,\n organizations: dbUser.organizations.map((org) => org.id),\n metadata: dbUser.metadata,\n };\n }\n\n private getMockUser(): AuthUser {\n // Return cached mock user if available\n if (this.mockUser) {\n return this.mockUser;\n }\n\n // Initialize mock user synchronously to prevent race conditions\n // This runs during constructor or first use\n if (!this.mockUserInitializing) {\n this.mockUserInitializing = true;\n\n // Initialize asynchronously but return a temporary user immediately\n this.initializeMockUser()\n .then((user) => {\n this.mockUser = user;\n this.mockUserInitializing = false;\n logger.info('Mock user initialized and cached');\n })\n .catch((err) => {\n logger.error('Failed to initialize mock user', err);\n this.mockUserInitializing = false;\n });\n }\n\n // Return temporary mock user while initialization is in progress\n return {\n id: 'temp-dev-user-id',\n sub: 'dev-sub',\n email: 'dev@stackmemory.local',\n name: 'Development User',\n tier: 'enterprise',\n permissions: ['read', 'write', 'admin', 'delete'],\n organizations: ['dev-org'],\n metadata: { temporary: true },\n };\n }\n\n /**\n * Revoke a token (add to blacklist)\n */\n public async revokeToken(token: string): Promise<void> {\n this.blacklistedTokens.add(token);\n await this.redis.publish('token:revoked', token);\n\n // Also store in Redis with TTL matching token expiry\n const decoded = jwt.decode(token) as any;\n if (decoded?.exp) {\n const ttl = decoded.exp - Math.floor(Date.now() / 1000);\n if (ttl > 0) {\n await this.redis.setex(`blacklist:${token}`, ttl, '1');\n }\n }\n }\n\n /**\n * Cleanup resources\n */\n public async close(): Promise<void> {\n await this.redis.quit();\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
6
|
"names": ["user"]
|
|
7
7
|
}
|
|
@@ -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// 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\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;AAEtC,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;
|
|
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": ";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;AAEtC,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;AAGA,MAAM,SAAS;AAAA,EACb,MAAM,SAAS,QAAQ,IAAI,MAAM,KAAK,MAAM;AAAA,EAC5C,aAAa,QAAQ,IAAI,UAAU,KAAK;AAAA,EACxC,aAAa,QAAQ,IAAI,cAAc,GAAG,MAAM,GAAG,KAAK;AAAA,IACtD;AAAA,EACF;AAAA,EACA,UAAU,QAAQ,IAAI,WAAW,KAAK;AAAA,EACtC,cAAc,QAAQ,IAAI,gBAAgB,KAAK;AAAA,EAC/C,WAAW,QAAQ,IAAI,YAAY,KAAK;AAAA,EACxC,aACE,QAAQ,IAAI,cAAc,KAC1B,KAAK,QAAQ,IAAI,GAAG,gBAAgB,YAAY;AAAA,EAClD,kBAAkB,QAAQ,IAAI,oBAAoB,MAAM;AAAA,EACxD,eAAe,SAAS,QAAQ,IAAI,iBAAiB,KAAK,KAAK;AAAA,EAC/D,iBAAiB,QAAQ,IAAI,kBAAkB,MAAM;AAAA,EACrD,iBAAiB,QAAQ,IAAI,kBAAkB,MAAM;AACvD;AAGA,MAAM,cAAc,oBAAI,IAAkD;AAE1E,MAAM,iBAAiB;AAAA,EACb;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA,cAAgC,oBAAI,IAAI;AAAA,EACxC;AAAA,EAER,cAAc;AACZ,SAAK,MAAM,QAAQ;AACnB,SAAK,aAAa,aAAa,KAAK,GAAG;AACvC,SAAK,mBAAmB;AACxB,SAAK,gBAAgB;AACrB,SAAK,YAAY;AACjB,SAAK,eAAe;AAGpB,SAAK,aAAa,IAAI,sBAAsB;AAAA,MAC1C,UAAU;AAAA;AAAA,MACV,iBAAiB,EAAE,OAAO,MAAM,QAAQ,IAAI;AAAA,IAC9C,CAAC;AAED,QAAI,OAAO,iBAAiB;AAC1B,WAAK,eAAe;AAAA,IACtB;AAAA,EACF;AAAA,EAEQ,qBAA2B;AAEjC,QACE,OAAO,gBAAgB,gBACvB,OAAO,YAAY,WAAW,eAAe,GAC7C;AACA,cAAQ,IAAI,2BAA2B;AAGvC,YAAM,SAAS;AACf,WAAK,KAAK,IAAI,SAAS,MAAM;AAAA,IAC/B,OAAO;AAEL,YAAM,QAAQ,QAAQ,OAAO,WAAW;AACxC,UAAI,CAAC,WAAW,KAAK,GAAG;AACtB,kBAAU,OAAO,EAAE,WAAW,KAAK,CAAC;AAAA,MACtC;AACA,WAAK,KAAK,IAAI,SAAS,OAAO,WAAW;AAAA,IAC3C;AAGA,SAAK,GAAG,KAAK;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,KAuBZ;AAAA,EACH;AAAA,EAEQ,kBAAwB;AAE9B,SAAK,IAAI;AAAA,MACP,KAAK;AAAA,QACH,QAAQ,OAAO;AAAA,QACf,aAAa;AAAA,MACf,CAAC;AAAA,IACH;AAGA,SAAK,IAAI,IAAI,QAAQ,KAAK,EAAE,OAAO,OAAO,CAAC,CAAC;AAG5C,SAAK,IAAI,IAAI,CAAC,KAAK,KAAK,SAAS;AAC/B,cAAQ,IAAI,IAAG,oBAAI,KAAK,GAAE,YAAY,CAAC,IAAI,IAAI,MAAM,IAAI,IAAI,IAAI,EAAE;AACnE,WAAK;AAAA,IACP,CAAC;AAGD,SAAK,IAAI,IAAI,QAAQ,KAAK,aAAa,KAAK,IAAI,CAAC;AAGjD,QAAI,OAAO,kBAAkB;AAC3B,WAAK,IAAI,IAAI,QAAQ,KAAK,UAAU,KAAK,IAAI,CAAC;AAAA,IAChD;AAAA,EACF;AAAA,EAEQ,aACN,KACA,KACA,MACK;AAEL,QAAI,IAAI,SAAS,WAAW;AAC1B,aAAO,KAAK;AAAA,IACd;AAEA,UAAM,aAAa,IAAI,QAAQ;AAE/B,QAAI,OAAO,aAAa,WAAW;AAEjC,UAAI,CAAC,YAAY,WAAW,SAAS,GAAG;AACtC,eAAO,IAAI,OAAO,GAAG,EAAE,KAAK,EAAE,OAAO,kBAAkB,CAAC;AAAA,MAC1D;AAEA,YAAM,SAAS,WAAW,UAAU,CAAC;AAIrC,UAAI,OAAO,SAAS,IAAI;AACtB,eAAO,IAAI,OAAO,GAAG,EAAE,KAAK,EAAE,OAAO,kBAAkB,CAAC;AAAA,MAC1D;AAEA,MAAC,IAAY,OAAO,EAAE,IAAI,YAAY,MAAM,OAAO;AACnD,WAAK;AAAA,IACP,OAAO;AAEL,WAAK;AAAA,IACP;AAAA,EACF;AAAA,EAEQ,UACN,KACA,KACA,MACK;AACL,UAAM,SAAU,IAAY,MAAM,MAAM,IAAI;AAC5C,UAAM,MAAM,KAAK,IAAI;AACrB,UAAM,WAAW,KAAK,KAAK;AAE3B,UAAM,YAAY,YAAY,IAAI,MAAM;AAExC,QAAI,CAAC,aAAa,UAAU,YAAY,KAAK;AAC3C,kBAAY,IAAI,QAAQ;AAAA,QACtB,OAAO;AAAA,QACP,WAAW,MAAM;AAAA,MACnB,CAAC;AACD,aAAO,KAAK;AAAA,IACd;AAEA,QAAI,UAAU,SAAS,OAAO,eAAe;AAC3C,YAAM,aAAa,KAAK,MAAM,UAAU,YAAY,OAAO,GAAI;AAC/D,UAAI,UAAU,eAAe,WAAW,SAAS,CAAC;AAClD,aAAO,IAAI,OAAO,GAAG,EAAE,KAAK;AAAA,QAC1B,OAAO;AAAA,QACP;AAAA,MACF,CAAC;AAAA,IACH;AAEA,cAAU;AACV,SAAK;AAAA,EACP;AAAA,EAEQ,cAAoB;AAE1B,SAAK,IAAI,IAAI,WAAW,CAAC,KAAK,QAAQ;AACpC,YAAM,SAAS;AAAA,QACb,QAAQ;AAAA,QACR,SAAS;AAAA,QACT,YAAW,oBAAI,KAAK,GAAE,YAAY;AAAA,QAClC,QAAQ,QAAQ,OAAO;AAAA,QACvB,aAAa,OAAO;AAAA,MACtB;AACA,UAAI,KAAK,MAAM;AAAA,IACjB,CAAC;AAGD,SAAK,IAAI,KAAK,qBAAqB,CAAC,KAAK,QAAQ;AAC/C,UAAI;AACF,cAAM;AAAA,UACJ,YAAY;AAAA,UACZ;AAAA,UACA,OAAO;AAAA,UACP,WAAW,CAAC;AAAA,QACd,IAAI,IAAI;AAER,cAAM,OAAO,KAAK,GAAG,QAAQ;AAAA;AAAA;AAAA,SAG5B;AAED,cAAM,SAAS,KAAK;AAAA,UAClB;AAAA,UACA;AAAA,UACA;AAAA,UACA,KAAK,UAAU,QAAQ;AAAA,QACzB;AAEA,YAAI,KAAK;AAAA,UACP,SAAS;AAAA,UACT,IAAI,OAAO;AAAA,QACb,CAAC;AAAA,MACH,SAAS,OAAY;AACnB,YAAI,OAAO,GAAG,EAAE,KAAK,EAAE,OAAO,MAAM,QAAQ,CAAC;AAAA,MAC/C;AAAA,IACF,CAAC;AAED,SAAK,IAAI,IAAI,qBAAqB,CAAC,KAAK,QAAQ;AAC9C,UAAI;AACF,cAAM,EAAE,YAAY,WAAW,QAAQ,IAAI,SAAS,EAAE,IAAI,IAAI;AAE9D,cAAM,OAAO,KAAK,GAAG,QAAQ;AAAA;AAAA;AAAA;AAAA;AAAA,SAK5B;AAED,cAAM,WAAW,KAAK,IAAI,WAAW,OAAO,MAAM;AAElD,YAAI,KAAK;AAAA,UACP,SAAS;AAAA,UACT,UAAU,SAAS,IAAI,CAAC,OAAY;AAAA,YAClC,GAAG;AAAA,YACH,UAAU,KAAK,MAAM,EAAE,YAAY,IAAI;AAAA,UACzC,EAAE;AAAA,QACJ,CAAC;AAAA,MACH,SAAS,OAAY;AACnB,YAAI,OAAO,GAAG,EAAE,KAAK,EAAE,OAAO,MAAM,QAAQ,CAAC;AAAA,MAC/C;AAAA,IACF,CAAC;AAGD,SAAK,IAAI,KAAK,sBAAsB,OAAO,KAAK,QAAQ;AACtD,UAAI;AACF,cAAM,EAAE,MAAM,OAAO,IAAI,IAAI;AAG7B,cAAM,SAAS,MAAM,KAAK,eAAe,MAAM,MAAM;AAErD,YAAI,KAAK;AAAA,UACP,SAAS;AAAA,UACT;AAAA,QACF,CAAC;AAAA,MACH,SAAS,OAAY;AACnB,YAAI,OAAO,GAAG,EAAE,KAAK,EAAE,OAAO,MAAM,QAAQ,CAAC;AAAA,MAC/C;AAAA,IACF,CAAC;AAGD,QAAI,OAAO,iBAAiB;AAC1B,WAAK,IAAI,IAAI,kBAAkB,CAAC,KAAK,QAAQ;AAC3C,YAAI;AACF,gBAAM,EAAE,YAAY,UAAU,IAAI,IAAI;AAEtC,gBAAM,QAAQ,KAAK,GAChB;AAAA,YACC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,UAQF,EACC,IAAI,SAAS;AAEhB,cAAI,KAAK;AAAA,YACP,SAAS;AAAA,YACT,WAAW;AAAA,UACb,CAAC;AAAA,QACH,SAAS,OAAY;AACnB,cAAI,OAAO,GAAG,EAAE,KAAK,EAAE,OAAO,MAAM,QAAQ,CAAC;AAAA,QAC/C;AAAA,MACF,CAAC;AAAA,IACH;AAAA,EACF;AAAA,EAEQ,iBAAuB;AAC7B,SAAK,MAAM,IAAI,gBAAgB;AAAA,MAC7B,QAAQ,KAAK;AAAA,MACb,MAAM;AAAA,IACR,CAAC;AAED,SAAK,IAAI,GAAG,cAAc,CAAC,IAAI,SAAS;AACtC,cAAQ,IAAI,kCAAkC;AAE9C,YAAM,eAAe,KAAK,OAAO,EAAE,SAAS,EAAE,EAAE,UAAU,CAAC;AAC3D,WAAK,YAAY,IAAI,cAAc,EAAE;AAErC,SAAG,GAAG,WAAW,OAAO,SAAS;AAC/B,YAAI;AACF,gBAAM,UAAU,KAAK,MAAM,KAAK,SAAS,CAAC;AAC1C,gBAAM,WAAW,MAAM,KAAK,uBAAuB,OAAO;AAC1D,aAAG,KAAK,KAAK,UAAU,QAAQ,CAAC;AAAA,QAClC,SAAS,OAAY;AACnB,aAAG;AAAA,YACD,KAAK,UAAU;AAAA,cACb,OAAO,MAAM;AAAA,YACf,CAAC;AAAA,UACH;AAAA,QACF;AAAA,MACF,CAAC;AAED,SAAG,GAAG,SAAS,MAAM;AACnB,aAAK,YAAY,OAAO,YAAY;AACpC,gBAAQ,IAAI,6BAA6B;AAAA,MAC3C,CAAC;AAAA,IACH,CAAC;AAAA,EACH;AAAA,EAEA,MAAc,uBAAuB,SAA4B;AAC/D,UAAM,EAAE,MAAM,MAAM,OAAO,IAAI;AAE/B,YAAQ,MAAM;AAAA,MACZ,KAAK;AACH,eAAO,MAAM,KAAK,eAAe,MAAM,MAAM;AAAA,MAE/C,KAAK;AACH,eAAO,EAAE,MAAM,OAAO;AAAA,MAExB;AACE,cAAM,IAAI,MAAM,yBAAyB,IAAI,EAAE;AAAA,IACnD;AAAA,EACF;AAAA,EAEA,MAAc,iBAAgC;AAC5C,SAAK,YAAY,IAAI;AAAA,MACnB;AAAA,QACE,MAAM;AAAA,QACN,SAAS;AAAA,MACX;AAAA,MACA;AAAA,QACE,cAAc;AAAA,UACZ,OAAO,CAAC;AAAA,UACR,WAAW,CAAC;AAAA,QACd;AAAA,MACF;AAAA,IACF;AAGA,UAAM,KAAK,WAAW,WAAW,KAAK,SAAS;AAG/C,SAAK,UAAU,kBAAkB,cAAqB,YAAY;AAChE,aAAO;AAAA,QACL,OAAO;AAAA,UACL;AAAA,YACE,MAAM;AAAA,YACN,aAAa;AAAA,YACb,aAAa;AAAA,cACX,MAAM;AAAA,cACN,YAAY;AAAA,gBACV,SAAS,EAAE,MAAM,SAAS;AAAA,gBAC1B,MAAM,EAAE,MAAM,SAAS;AAAA,cACzB;AAAA,YACF;AAAA,UACF;AAAA,UACA;AAAA,YACE,MAAM;AAAA,YACN,aAAa;AAAA,YACb,aAAa;AAAA,cACX,MAAM;AAAA,cACN,YAAY;AAAA,gBACV,OAAO,EAAE,MAAM,SAAS;AAAA,gBACxB,OAAO,EAAE,MAAM,SAAS;AAAA,cAC1B;AAAA,YACF;AAAA,UACF;AAAA,QACF;AAAA,MACF;AAAA,IACF,CAAC;AAED,SAAK,UAAU;AAAA,MACb;AAAA,MACA,OAAO,YAAiB;AACtB,cAAM,EAAE,MAAM,WAAW,KAAK,IAAI,QAAQ;AAC1C,eAAO,MAAM,KAAK,eAAe,MAAM,IAAI;AAAA,MAC7C;AAAA,IACF;AAAA,EACF;AAAA,EAEA,MAAc,eAAe,MAAc,QAA2B;AACpE,YAAQ,MAAM;AAAA,MACZ,KAAK,gBAAgB;AACnB,cAAM,OAAO,KAAK,GAAG,QAAQ;AAAA;AAAA;AAAA,SAG5B;AACD,cAAM,SAAS,KAAK;AAAA,UAClB,OAAO,aAAa;AAAA,UACpB,OAAO;AAAA,UACP,OAAO,QAAQ;AAAA,UACf,KAAK,UAAU,OAAO,YAAY,CAAC,CAAC;AAAA,QACtC;AACA,eAAO,EAAE,IAAI,OAAO,iBAAiB,SAAS,KAAK;AAAA,MACrD;AAAA,MAEA,KAAK,gBAAgB;AACnB,cAAM,OAAO,KAAK,GAAG,QAAQ;AAAA;AAAA;AAAA;AAAA;AAAA,SAK5B;AACD,cAAM,WAAW,KAAK;AAAA,UACpB,OAAO,aAAa;AAAA,UACpB,IAAI,OAAO,SAAS,EAAE;AAAA,UACtB,OAAO,SAAS;AAAA,QAClB;AACA,eAAO,EAAE,UAAU,SAAS,KAAK;AAAA,MACnC;AAAA,MAEA;AACE,cAAM,IAAI,MAAM,iBAAiB,IAAI,EAAE;AAAA,IAC3C;AAAA,EACF;AAAA,EAEO,QAAc;AACnB,SAAK,WAAW,OAAO,OAAO,MAAM,WAAW,MAAM;AACnD,cAAQ,IAAI;AAAA;AAAA;AAAA,eAGH,OAAO,WAAW;AAAA,QACzB,OAAO,IAAI;AAAA,aACN,OAAO,kBAAkB,YAAY,UAAU;AAAA,aAC/C,OAAO,kBAAkB,YAAY,UAAU;AAAA,iBAC3C,OAAO,mBAAmB,YAAY,UAAU;AAAA,aACpD,OAAO,QAAQ;AAAA;AAAA,2BAED,OAAO,IAAI;AAAA,OAC/B;AAAA,IACH,CAAC;AAAA,EACH;AACF;AAGA,MAAM,SAAS,IAAI,iBAAiB;AACpC,OAAO,MAAM;AAGb,QAAQ,GAAG,WAAW,MAAM;AAC1B,UAAQ,IAAI,6BAA6B;AACzC,UAAQ,KAAK,CAAC;AAChB,CAAC;AAED,QAAQ,GAAG,UAAU,MAAM;AACzB,UAAQ,IAAI,kBAAkB;AAC9B,UAAQ,KAAK,CAAC;AAChB,CAAC;",
|
|
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
|
}
|
|
@@ -541,7 +541,10 @@ class ArchaeologistSkill {
|
|
|
541
541
|
if (!timeline.has(date)) {
|
|
542
542
|
timeline.set(date, []);
|
|
543
543
|
}
|
|
544
|
-
timeline.get(date)
|
|
544
|
+
const dateItems = timeline.get(date);
|
|
545
|
+
if (dateItems) {
|
|
546
|
+
dateItems.push(result);
|
|
547
|
+
}
|
|
545
548
|
});
|
|
546
549
|
return Array.from(timeline.entries()).map(([date, items]) => ({
|
|
547
550
|
date,
|
|
@@ -625,7 +628,6 @@ class ClaudeSkillsManager {
|
|
|
625
628
|
checkpointSkill;
|
|
626
629
|
archaeologistSkill;
|
|
627
630
|
dashboardLauncher;
|
|
628
|
-
// DashboardLauncherSkill
|
|
629
631
|
repoIngestionSkill = null;
|
|
630
632
|
async executeSkill(skillName, args, options) {
|
|
631
633
|
switch (skillName) {
|