@stackmemoryai/stackmemory 0.5.58 → 0.5.59
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/cli/commands/search.js +20 -3
- package/dist/cli/commands/search.js.map +2 -2
- package/dist/core/database/sqlite-adapter.js +13 -3
- package/dist/core/database/sqlite-adapter.js.map +2 -2
- package/dist/core/errors/error-utils.js +208 -0
- package/dist/core/errors/error-utils.js.map +7 -0
- package/dist/core/errors/index.js +13 -4
- package/dist/core/errors/index.js.map +2 -2
- package/dist/core/merge/unified-merge-resolver.js +303 -0
- package/dist/core/merge/unified-merge-resolver.js.map +7 -0
- package/dist/core/monitoring/logger.js +61 -9
- package/dist/core/monitoring/logger.js.map +2 -2
- package/dist/core/security/index.js +35 -0
- package/dist/core/security/index.js.map +7 -0
- package/dist/core/security/input-sanitizer.js +321 -0
- package/dist/core/security/input-sanitizer.js.map +7 -0
- package/dist/integrations/linear/client.js +5 -1
- package/dist/integrations/linear/client.js.map +2 -2
- package/dist/integrations/mcp/remote-server.js +27 -36
- package/dist/integrations/mcp/remote-server.js.map +2 -2
- package/dist/integrations/mcp/server.js +44 -29
- package/dist/integrations/mcp/server.js.map +3 -3
- package/dist/scripts/benchmark-performance.js +48 -0
- package/dist/scripts/benchmark-performance.js.map +7 -0
- package/dist/scripts/check-redis.js +42 -0
- package/dist/scripts/check-redis.js.map +7 -0
- package/dist/scripts/initialize.js +116 -0
- package/dist/scripts/initialize.js.map +7 -0
- package/dist/scripts/list-linear-tasks.js +124 -0
- package/dist/scripts/list-linear-tasks.js.map +7 -0
- package/dist/scripts/measure-handoff-impact.js +340 -0
- package/dist/scripts/measure-handoff-impact.js.map +7 -0
- package/dist/scripts/query-chromadb.js +160 -0
- package/dist/scripts/query-chromadb.js.map +7 -0
- package/dist/scripts/show-linear-summary.js +119 -0
- package/dist/scripts/show-linear-summary.js.map +7 -0
- package/dist/scripts/simple-swarm-demo.js +90 -0
- package/dist/scripts/simple-swarm-demo.js.map +7 -0
- package/dist/scripts/status.js +155 -0
- package/dist/scripts/status.js.map +7 -0
- package/dist/scripts/test-chromadb-sync.js +192 -0
- package/dist/scripts/test-chromadb-sync.js.map +7 -0
- package/dist/scripts/test-ralph-iteration-fix.js +86 -0
- package/dist/scripts/test-ralph-iteration-fix.js.map +7 -0
- package/dist/scripts/test-ralph-iterations.js +121 -0
- package/dist/scripts/test-ralph-iterations.js.map +7 -0
- package/dist/scripts/test-redis-storage.js +389 -0
- package/dist/scripts/test-redis-storage.js.map +7 -0
- package/dist/scripts/test-simple-ralph-state-sync.js +115 -0
- package/dist/scripts/test-simple-ralph-state-sync.js.map +7 -0
- package/dist/scripts/test-swarm-fixes.js +125 -0
- package/dist/scripts/test-swarm-fixes.js.map +7 -0
- package/dist/scripts/test-swarm-tui.js +23 -0
- package/dist/scripts/test-swarm-tui.js.map +7 -0
- package/dist/scripts/test-tui-shortcuts.js +52 -0
- package/dist/scripts/test-tui-shortcuts.js.map +7 -0
- package/dist/scripts/validate-tui-shortcuts.js +60 -0
- package/dist/scripts/validate-tui-shortcuts.js.map +7 -0
- package/dist/src/agents/core/agent-task-manager.js +527 -0
- package/dist/src/agents/core/agent-task-manager.js.map +7 -0
- package/dist/src/agents/verifiers/base-verifier.js +133 -0
- package/dist/src/agents/verifiers/base-verifier.js.map +7 -0
- package/dist/src/agents/verifiers/formatter-verifier.js +130 -0
- package/dist/src/agents/verifiers/formatter-verifier.js.map +7 -0
- package/dist/src/agents/verifiers/llm-judge.js +252 -0
- package/dist/src/agents/verifiers/llm-judge.js.map +7 -0
- package/dist/src/cli/auto-detect.js +321 -0
- package/dist/src/cli/auto-detect.js.map +7 -0
- package/dist/src/cli/claude-sm-danger.js +21 -0
- package/dist/src/cli/claude-sm-danger.js.map +7 -0
- package/dist/src/cli/claude-sm.js +1156 -0
- package/dist/src/cli/claude-sm.js.map +7 -0
- package/dist/src/cli/codex-sm-danger.js +21 -0
- package/dist/src/cli/codex-sm-danger.js.map +7 -0
- package/dist/src/cli/codex-sm.js +349 -0
- package/dist/src/cli/codex-sm.js.map +7 -0
- package/dist/src/cli/commands/api.js +232 -0
- package/dist/src/cli/commands/api.js.map +7 -0
- package/dist/src/cli/commands/auto-background.js +180 -0
- package/dist/src/cli/commands/auto-background.js.map +7 -0
- package/dist/src/cli/commands/cleanup-processes.js +68 -0
- package/dist/src/cli/commands/cleanup-processes.js.map +7 -0
- package/dist/src/cli/commands/clear.js +202 -0
- package/dist/src/cli/commands/clear.js.map +7 -0
- package/dist/src/cli/commands/config.js +445 -0
- package/dist/src/cli/commands/config.js.map +7 -0
- package/dist/src/cli/commands/context-rehydrate.js +751 -0
- package/dist/src/cli/commands/context-rehydrate.js.map +7 -0
- package/dist/src/cli/commands/context.js +343 -0
- package/dist/src/cli/commands/context.js.map +7 -0
- package/dist/src/cli/commands/daemon.js +392 -0
- package/dist/src/cli/commands/daemon.js.map +7 -0
- package/dist/src/cli/commands/dashboard.js +210 -0
- package/dist/src/cli/commands/dashboard.js.map +7 -0
- package/dist/src/cli/commands/db.js +147 -0
- package/dist/src/cli/commands/db.js.map +7 -0
- package/dist/src/cli/commands/decision.js +266 -0
- package/dist/src/cli/commands/decision.js.map +7 -0
- package/dist/src/cli/commands/discovery.js +279 -0
- package/dist/src/cli/commands/discovery.js.map +7 -0
- package/dist/src/cli/commands/handoff.js +624 -0
- package/dist/src/cli/commands/handoff.js.map +7 -0
- package/dist/src/cli/commands/hooks.js +298 -0
- package/dist/src/cli/commands/hooks.js.map +7 -0
- package/dist/src/cli/commands/linear.js +529 -0
- package/dist/src/cli/commands/linear.js.map +7 -0
- package/dist/src/cli/commands/log.js +169 -0
- package/dist/src/cli/commands/log.js.map +7 -0
- package/dist/src/cli/commands/login.js +172 -0
- package/dist/src/cli/commands/login.js.map +7 -0
- package/dist/src/cli/commands/migrate.js +240 -0
- package/dist/src/cli/commands/migrate.js.map +7 -0
- package/dist/src/cli/commands/model.js +533 -0
- package/dist/src/cli/commands/model.js.map +7 -0
- package/dist/src/cli/commands/onboard.js +536 -0
- package/dist/src/cli/commands/onboard.js.map +7 -0
- package/dist/src/cli/commands/projects.js +199 -0
- package/dist/src/cli/commands/projects.js.map +7 -0
- package/dist/src/cli/commands/ralph.js +909 -0
- package/dist/src/cli/commands/ralph.js.map +7 -0
- package/dist/src/cli/commands/retrieval.js +248 -0
- package/dist/src/cli/commands/retrieval.js.map +7 -0
- package/dist/src/cli/commands/search.js +173 -0
- package/dist/src/cli/commands/search.js.map +7 -0
- package/dist/src/cli/commands/service.js +749 -0
- package/dist/src/cli/commands/service.js.map +7 -0
- package/dist/src/cli/commands/session.js +200 -0
- package/dist/src/cli/commands/session.js.map +7 -0
- package/dist/src/cli/commands/settings.js +306 -0
- package/dist/src/cli/commands/settings.js.map +7 -0
- package/dist/src/cli/commands/setup.js +701 -0
- package/dist/src/cli/commands/setup.js.map +7 -0
- package/dist/src/cli/commands/shell.js +249 -0
- package/dist/src/cli/commands/shell.js.map +7 -0
- package/dist/src/cli/commands/signup.js +50 -0
- package/dist/src/cli/commands/signup.js.map +7 -0
- package/dist/src/cli/commands/skills.js +470 -0
- package/dist/src/cli/commands/skills.js.map +7 -0
- package/dist/src/cli/commands/sms-notify.js +795 -0
- package/dist/src/cli/commands/sms-notify.js.map +7 -0
- package/dist/src/cli/commands/storage-tier.js +183 -0
- package/dist/src/cli/commands/storage-tier.js.map +7 -0
- package/dist/src/cli/commands/sweep.js +249 -0
- package/dist/src/cli/commands/sweep.js.map +7 -0
- package/dist/src/cli/commands/tasks.js +213 -0
- package/dist/src/cli/commands/tasks.js.map +7 -0
- package/dist/src/cli/commands/worktree.js +319 -0
- package/dist/src/cli/commands/worktree.js.map +7 -0
- package/dist/src/cli/index.js +594 -0
- package/dist/src/cli/index.js.map +7 -0
- package/dist/src/cli/opencode-sm.js +448 -0
- package/dist/src/cli/opencode-sm.js.map +7 -0
- package/dist/src/cli/utils/viewer.js +96 -0
- package/dist/src/cli/utils/viewer.js.map +7 -0
- package/dist/src/core/config/config-manager.js +398 -0
- package/dist/src/core/config/config-manager.js.map +7 -0
- package/dist/src/core/config/feature-flags.js +76 -0
- package/dist/src/core/config/feature-flags.js.map +7 -0
- package/dist/src/core/config/storage-config.js +115 -0
- package/dist/src/core/config/storage-config.js.map +7 -0
- package/dist/src/core/config/types.js +144 -0
- package/dist/src/core/config/types.js.map +7 -0
- package/dist/src/core/context/auto-context.js +80 -0
- package/dist/src/core/context/auto-context.js.map +7 -0
- package/dist/src/core/context/dual-stack-manager.js +870 -0
- package/dist/src/core/context/dual-stack-manager.js.map +7 -0
- package/dist/src/core/context/enhanced-rehydration.js +994 -0
- package/dist/src/core/context/enhanced-rehydration.js.map +7 -0
- package/dist/src/core/context/frame-database.js +479 -0
- package/dist/src/core/context/frame-database.js.map +7 -0
- package/dist/src/core/context/frame-digest.js +250 -0
- package/dist/src/core/context/frame-digest.js.map +7 -0
- package/dist/src/core/context/frame-handoff-manager.js +778 -0
- package/dist/src/core/context/frame-handoff-manager.js.map +7 -0
- package/dist/src/core/context/frame-lifecycle-hooks.js +119 -0
- package/dist/src/core/context/frame-lifecycle-hooks.js.map +7 -0
- package/dist/src/core/context/frame-recovery.js +302 -0
- package/dist/src/core/context/frame-recovery.js.map +7 -0
- package/dist/src/core/context/frame-stack.js +314 -0
- package/dist/src/core/context/frame-stack.js.map +7 -0
- package/dist/src/core/context/frame-types.js +5 -0
- package/dist/src/core/context/frame-types.js.map +7 -0
- package/dist/src/core/context/index.js +25 -0
- package/dist/src/core/context/index.js.map +7 -0
- package/dist/src/core/context/permission-manager.js +185 -0
- package/dist/src/core/context/permission-manager.js.map +7 -0
- package/dist/src/core/context/recursive-context-manager.js +592 -0
- package/dist/src/core/context/recursive-context-manager.js.map +7 -0
- package/dist/src/core/context/refactored-frame-manager.js +754 -0
- package/dist/src/core/context/refactored-frame-manager.js.map +7 -0
- package/dist/src/core/context/shared-context-layer.js +621 -0
- package/dist/src/core/context/shared-context-layer.js.map +7 -0
- package/dist/src/core/context/stack-merge-resolver.js +749 -0
- package/dist/src/core/context/stack-merge-resolver.js.map +7 -0
- package/dist/src/core/context/validation.js +130 -0
- package/dist/src/core/context/validation.js.map +7 -0
- package/dist/src/core/database/batch-operations.js +384 -0
- package/dist/src/core/database/batch-operations.js.map +7 -0
- package/dist/src/core/database/connection-pool.js +330 -0
- package/dist/src/core/database/connection-pool.js.map +7 -0
- package/dist/src/core/database/database-adapter.js +60 -0
- package/dist/src/core/database/database-adapter.js.map +7 -0
- package/dist/src/core/database/migration-manager.js +614 -0
- package/dist/src/core/database/migration-manager.js.map +7 -0
- package/dist/src/core/database/query-cache.js +298 -0
- package/dist/src/core/database/query-cache.js.map +7 -0
- package/dist/src/core/database/query-router.js +430 -0
- package/dist/src/core/database/query-router.js.map +7 -0
- package/dist/src/core/database/sqlite-adapter.js +738 -0
- package/dist/src/core/database/sqlite-adapter.js.map +7 -0
- package/dist/src/core/digest/enhanced-hybrid-digest.js +277 -0
- package/dist/src/core/digest/enhanced-hybrid-digest.js.map +7 -0
- package/dist/src/core/digest/frame-digest-integration.js +176 -0
- package/dist/src/core/digest/frame-digest-integration.js.map +7 -0
- package/dist/src/core/digest/hybrid-digest-generator.js +553 -0
- package/dist/src/core/digest/hybrid-digest-generator.js.map +7 -0
- package/dist/src/core/digest/index.js +9 -0
- package/dist/src/core/digest/index.js.map +7 -0
- package/dist/src/core/digest/types.js +25 -0
- package/dist/src/core/digest/types.js.map +7 -0
- package/dist/src/core/errors/error-utils.js +208 -0
- package/dist/src/core/errors/error-utils.js.map +7 -0
- package/dist/src/core/errors/index.js +521 -0
- package/dist/src/core/errors/index.js.map +7 -0
- package/dist/src/core/errors/recovery.js +269 -0
- package/dist/src/core/errors/recovery.js.map +7 -0
- package/dist/src/core/execution/parallel-executor.js +258 -0
- package/dist/src/core/execution/parallel-executor.js.map +7 -0
- package/dist/src/core/frame/workflow-templates.js +319 -0
- package/dist/src/core/frame/workflow-templates.js.map +7 -0
- package/dist/src/core/merge/conflict-detector.js +431 -0
- package/dist/src/core/merge/conflict-detector.js.map +7 -0
- package/dist/src/core/merge/index.js +9 -0
- package/dist/src/core/merge/index.js.map +7 -0
- package/dist/src/core/merge/resolution-engine.js +558 -0
- package/dist/src/core/merge/resolution-engine.js.map +7 -0
- package/dist/src/core/merge/stack-diff.js +532 -0
- package/dist/src/core/merge/stack-diff.js.map +7 -0
- package/dist/src/core/merge/types.js +5 -0
- package/dist/src/core/merge/types.js.map +7 -0
- package/dist/src/core/merge/unified-merge-resolver.js +303 -0
- package/dist/src/core/merge/unified-merge-resolver.js.map +7 -0
- package/dist/src/core/models/fallback-monitor.js +232 -0
- package/dist/src/core/models/fallback-monitor.js.map +7 -0
- package/dist/src/core/models/model-router.js +340 -0
- package/dist/src/core/models/model-router.js.map +7 -0
- package/dist/src/core/monitoring/error-handler.js +49 -0
- package/dist/src/core/monitoring/error-handler.js.map +7 -0
- package/dist/src/core/monitoring/logger.js +202 -0
- package/dist/src/core/monitoring/logger.js.map +7 -0
- package/dist/src/core/monitoring/metrics.js +172 -0
- package/dist/src/core/monitoring/metrics.js.map +7 -0
- package/dist/src/core/monitoring/progress-tracker.js +189 -0
- package/dist/src/core/monitoring/progress-tracker.js.map +7 -0
- package/dist/src/core/monitoring/session-monitor.js +300 -0
- package/dist/src/core/monitoring/session-monitor.js.map +7 -0
- package/dist/src/core/performance/context-cache.js +273 -0
- package/dist/src/core/performance/context-cache.js.map +7 -0
- package/dist/src/core/performance/index.js +11 -0
- package/dist/src/core/performance/index.js.map +7 -0
- package/dist/src/core/performance/lazy-context-loader.js +327 -0
- package/dist/src/core/performance/lazy-context-loader.js.map +7 -0
- package/dist/src/core/performance/monitor.js +221 -0
- package/dist/src/core/performance/monitor.js.map +7 -0
- package/dist/src/core/performance/optimized-frame-context.js +345 -0
- package/dist/src/core/performance/optimized-frame-context.js.map +7 -0
- package/dist/src/core/performance/performance-benchmark.js +277 -0
- package/dist/src/core/performance/performance-benchmark.js.map +7 -0
- package/dist/src/core/performance/performance-profiler.js +370 -0
- package/dist/src/core/performance/performance-profiler.js.map +7 -0
- package/dist/src/core/performance/streaming-jsonl-parser.js +195 -0
- package/dist/src/core/performance/streaming-jsonl-parser.js.map +7 -0
- package/dist/src/core/persistence/postgres-adapter.js +349 -0
- package/dist/src/core/persistence/postgres-adapter.js.map +7 -0
- package/dist/src/core/projects/project-isolation.js +201 -0
- package/dist/src/core/projects/project-isolation.js.map +7 -0
- package/dist/src/core/projects/project-manager.js +697 -0
- package/dist/src/core/projects/project-manager.js.map +7 -0
- package/dist/src/core/query/query-parser.js +370 -0
- package/dist/src/core/query/query-parser.js.map +7 -0
- package/dist/src/core/query/query-templates.js +321 -0
- package/dist/src/core/query/query-templates.js.map +7 -0
- package/dist/src/core/retrieval/context-retriever.js +479 -0
- package/dist/src/core/retrieval/context-retriever.js.map +7 -0
- package/dist/src/core/retrieval/index.js +8 -0
- package/dist/src/core/retrieval/index.js.map +7 -0
- package/dist/src/core/retrieval/llm-context-retrieval.js +613 -0
- package/dist/src/core/retrieval/llm-context-retrieval.js.map +7 -0
- package/dist/src/core/retrieval/llm-provider.js +151 -0
- package/dist/src/core/retrieval/llm-provider.js.map +7 -0
- package/dist/src/core/retrieval/retrieval-audit.js +236 -0
- package/dist/src/core/retrieval/retrieval-audit.js.map +7 -0
- package/dist/src/core/retrieval/summary-generator.js +589 -0
- package/dist/src/core/retrieval/summary-generator.js.map +7 -0
- package/dist/src/core/retrieval/types.js +21 -0
- package/dist/src/core/retrieval/types.js.map +7 -0
- package/dist/src/core/security/index.js +35 -0
- package/dist/src/core/security/index.js.map +7 -0
- package/dist/src/core/security/input-sanitizer.js +321 -0
- package/dist/src/core/security/input-sanitizer.js.map +7 -0
- package/dist/src/core/session/clear-survival.js +465 -0
- package/dist/src/core/session/clear-survival.js.map +7 -0
- package/dist/src/core/session/enhanced-handoff.js +792 -0
- package/dist/src/core/session/enhanced-handoff.js.map +7 -0
- package/dist/src/core/session/handoff-generator.js +343 -0
- package/dist/src/core/session/handoff-generator.js.map +7 -0
- package/dist/src/core/session/index.js +15 -0
- package/dist/src/core/session/index.js.map +7 -0
- package/dist/src/core/session/session-manager.js +347 -0
- package/dist/src/core/session/session-manager.js.map +7 -0
- package/dist/src/core/skills/index.js +7 -0
- package/dist/src/core/skills/index.js.map +7 -0
- package/dist/src/core/skills/skill-storage.js +764 -0
- package/dist/src/core/skills/skill-storage.js.map +7 -0
- package/dist/src/core/skills/types.js +193 -0
- package/dist/src/core/skills/types.js.map +7 -0
- package/dist/src/core/storage/chromadb-adapter.js +354 -0
- package/dist/src/core/storage/chromadb-adapter.js.map +7 -0
- package/dist/src/core/storage/infinite-storage.js +510 -0
- package/dist/src/core/storage/infinite-storage.js.map +7 -0
- package/dist/src/core/storage/remote-storage.js +489 -0
- package/dist/src/core/storage/remote-storage.js.map +7 -0
- package/dist/src/core/storage/two-tier-storage.js +766 -0
- package/dist/src/core/storage/two-tier-storage.js.map +7 -0
- package/dist/src/core/trace/cli-trace-wrapper.js +132 -0
- package/dist/src/core/trace/cli-trace-wrapper.js.map +7 -0
- package/dist/src/core/trace/db-trace-wrapper.js +247 -0
- package/dist/src/core/trace/db-trace-wrapper.js.map +7 -0
- package/dist/src/core/trace/debug-trace.js +417 -0
- package/dist/src/core/trace/debug-trace.js.map +7 -0
- package/dist/src/core/trace/index.js +109 -0
- package/dist/src/core/trace/index.js.map +7 -0
- package/dist/src/core/trace/linear-api-wrapper.js +178 -0
- package/dist/src/core/trace/linear-api-wrapper.js.map +7 -0
- package/dist/src/core/trace/trace-detector.js +528 -0
- package/dist/src/core/trace/trace-detector.js.map +7 -0
- package/dist/src/core/trace/trace-store.js +345 -0
- package/dist/src/core/trace/trace-store.js.map +7 -0
- package/dist/src/core/trace/types.js +77 -0
- package/dist/src/core/trace/types.js.map +7 -0
- package/dist/src/core/types.js +5 -0
- package/dist/src/core/types.js.map +7 -0
- package/dist/src/core/utils/async-mutex.js +114 -0
- package/dist/src/core/utils/async-mutex.js.map +7 -0
- package/dist/src/core/utils/compression.js +83 -0
- package/dist/src/core/utils/compression.js.map +7 -0
- package/dist/src/core/utils/update-checker.js +218 -0
- package/dist/src/core/utils/update-checker.js.map +7 -0
- package/dist/src/core/worktree/worktree-manager.js +465 -0
- package/dist/src/core/worktree/worktree-manager.js.map +7 -0
- package/dist/src/daemon/daemon-config.js +149 -0
- package/dist/src/daemon/daemon-config.js.map +7 -0
- package/dist/src/daemon/services/context-service.js +122 -0
- package/dist/src/daemon/services/context-service.js.map +7 -0
- package/dist/src/daemon/services/linear-service.js +136 -0
- package/dist/src/daemon/services/linear-service.js.map +7 -0
- package/dist/src/daemon/session-daemon.js +312 -0
- package/dist/src/daemon/session-daemon.js.map +7 -0
- package/dist/src/daemon/unified-daemon.js +276 -0
- package/dist/src/daemon/unified-daemon.js.map +7 -0
- package/dist/src/features/analytics/api/analytics-api.js +287 -0
- package/dist/src/features/analytics/api/analytics-api.js.map +7 -0
- package/dist/src/features/analytics/core/analytics-service.js +282 -0
- package/dist/src/features/analytics/core/analytics-service.js.map +7 -0
- package/dist/src/features/analytics/index.js +18 -0
- package/dist/src/features/analytics/index.js.map +7 -0
- package/dist/src/features/analytics/queries/metrics-queries.js +277 -0
- package/dist/src/features/analytics/queries/metrics-queries.js.map +7 -0
- package/dist/src/features/analytics/types/metrics.js +5 -0
- package/dist/src/features/analytics/types/metrics.js.map +7 -0
- package/dist/src/features/browser/browser-mcp.js +492 -0
- package/dist/src/features/browser/browser-mcp.js.map +7 -0
- package/dist/src/features/sweep/index.js +20 -0
- package/dist/src/features/sweep/index.js.map +7 -0
- package/dist/src/features/sweep/prediction-client.js +155 -0
- package/dist/src/features/sweep/prediction-client.js.map +7 -0
- package/dist/src/features/sweep/prompt-builder.js +85 -0
- package/dist/src/features/sweep/prompt-builder.js.map +7 -0
- package/dist/src/features/sweep/pty-wrapper.js +171 -0
- package/dist/src/features/sweep/pty-wrapper.js.map +7 -0
- package/dist/src/features/sweep/state-watcher.js +87 -0
- package/dist/src/features/sweep/state-watcher.js.map +7 -0
- package/dist/src/features/sweep/status-bar.js +88 -0
- package/dist/src/features/sweep/status-bar.js.map +7 -0
- package/dist/src/features/sweep/sweep-server-manager.js +226 -0
- package/dist/src/features/sweep/sweep-server-manager.js.map +7 -0
- package/dist/src/features/sweep/tab-interceptor.js +38 -0
- package/dist/src/features/sweep/tab-interceptor.js.map +7 -0
- package/dist/src/features/sweep/types.js +18 -0
- package/dist/src/features/sweep/types.js.map +7 -0
- package/dist/src/features/tasks/linear-task-manager.js +487 -0
- package/dist/src/features/tasks/linear-task-manager.js.map +7 -0
- package/dist/src/features/tasks/task-aware-context.js +410 -0
- package/dist/src/features/tasks/task-aware-context.js.map +7 -0
- package/dist/src/features/tui/simple-monitor.js +116 -0
- package/dist/src/features/tui/simple-monitor.js.map +7 -0
- package/dist/src/features/tui/swarm-monitor.js +648 -0
- package/dist/src/features/tui/swarm-monitor.js.map +7 -0
- package/dist/src/features/web/client/stores/task-store.js +26 -0
- package/dist/src/features/web/client/stores/task-store.js.map +7 -0
- package/dist/src/features/web/server/index.js +194 -0
- package/dist/src/features/web/server/index.js.map +7 -0
- package/dist/src/hooks/auto-background.js +151 -0
- package/dist/src/hooks/auto-background.js.map +7 -0
- package/dist/src/hooks/claude-code-whatsapp-hook.js +197 -0
- package/dist/src/hooks/claude-code-whatsapp-hook.js.map +7 -0
- package/dist/src/hooks/config.js +150 -0
- package/dist/src/hooks/config.js.map +7 -0
- package/dist/src/hooks/daemon.js +364 -0
- package/dist/src/hooks/daemon.js.map +7 -0
- package/dist/src/hooks/events.js +58 -0
- package/dist/src/hooks/events.js.map +7 -0
- package/dist/src/hooks/index.js +12 -0
- package/dist/src/hooks/index.js.map +7 -0
- package/dist/src/hooks/linear-task-picker.js +186 -0
- package/dist/src/hooks/linear-task-picker.js.map +7 -0
- package/dist/src/hooks/schemas.js +197 -0
- package/dist/src/hooks/schemas.js.map +7 -0
- package/dist/src/hooks/secure-fs.js +49 -0
- package/dist/src/hooks/secure-fs.js.map +7 -0
- package/dist/src/hooks/security-logger.js +155 -0
- package/dist/src/hooks/security-logger.js.map +7 -0
- package/dist/src/hooks/session-summary.js +222 -0
- package/dist/src/hooks/session-summary.js.map +7 -0
- package/dist/src/hooks/sms-action-runner.js +371 -0
- package/dist/src/hooks/sms-action-runner.js.map +7 -0
- package/dist/src/hooks/sms-notify.js +506 -0
- package/dist/src/hooks/sms-notify.js.map +7 -0
- package/dist/src/hooks/sms-watcher.js +93 -0
- package/dist/src/hooks/sms-watcher.js.map +7 -0
- package/dist/src/hooks/sms-webhook.js +555 -0
- package/dist/src/hooks/sms-webhook.js.map +7 -0
- package/dist/src/hooks/whatsapp-commands.js +479 -0
- package/dist/src/hooks/whatsapp-commands.js.map +7 -0
- package/dist/src/hooks/whatsapp-scheduler.js +317 -0
- package/dist/src/hooks/whatsapp-scheduler.js.map +7 -0
- package/dist/src/hooks/whatsapp-sync.js +409 -0
- package/dist/src/hooks/whatsapp-sync.js.map +7 -0
- package/dist/src/index.js +25 -0
- package/dist/src/index.js.map +7 -0
- package/dist/src/integrations/anthropic/client.js +263 -0
- package/dist/src/integrations/anthropic/client.js.map +7 -0
- package/dist/src/integrations/claude-code/agent-bridge.js +768 -0
- package/dist/src/integrations/claude-code/agent-bridge.js.map +7 -0
- package/dist/src/integrations/claude-code/enhanced-pre-clear-hooks.js +459 -0
- package/dist/src/integrations/claude-code/enhanced-pre-clear-hooks.js.map +7 -0
- package/dist/src/integrations/claude-code/lifecycle-hooks.js +254 -0
- package/dist/src/integrations/claude-code/lifecycle-hooks.js.map +7 -0
- package/dist/src/integrations/claude-code/post-task-hooks.js +545 -0
- package/dist/src/integrations/claude-code/post-task-hooks.js.map +7 -0
- package/dist/src/integrations/claude-code/subagent-client-stub.js +20 -0
- package/dist/src/integrations/claude-code/subagent-client-stub.js.map +7 -0
- package/dist/src/integrations/claude-code/subagent-client.js +511 -0
- package/dist/src/integrations/claude-code/subagent-client.js.map +7 -0
- package/dist/src/integrations/claude-code/task-coordinator.js +360 -0
- package/dist/src/integrations/claude-code/task-coordinator.js.map +7 -0
- package/dist/src/integrations/linear/auth.js +337 -0
- package/dist/src/integrations/linear/auth.js.map +7 -0
- package/dist/src/integrations/linear/auto-sync.js +258 -0
- package/dist/src/integrations/linear/auto-sync.js.map +7 -0
- package/dist/src/integrations/linear/client.js +634 -0
- package/dist/src/integrations/linear/client.js.map +7 -0
- package/dist/src/integrations/linear/config.js +130 -0
- package/dist/src/integrations/linear/config.js.map +7 -0
- package/dist/src/integrations/linear/migration.js +361 -0
- package/dist/src/integrations/linear/migration.js.map +7 -0
- package/dist/src/integrations/linear/oauth-server.js +454 -0
- package/dist/src/integrations/linear/oauth-server.js.map +7 -0
- package/dist/src/integrations/linear/rest-client.js +213 -0
- package/dist/src/integrations/linear/rest-client.js.map +7 -0
- package/dist/src/integrations/linear/sync-manager.js +236 -0
- package/dist/src/integrations/linear/sync-manager.js.map +7 -0
- package/dist/src/integrations/linear/sync-service.js +231 -0
- package/dist/src/integrations/linear/sync-service.js.map +7 -0
- package/dist/src/integrations/linear/sync.js +782 -0
- package/dist/src/integrations/linear/sync.js.map +7 -0
- package/dist/src/integrations/linear/types.js +5 -0
- package/dist/src/integrations/linear/types.js.map +7 -0
- package/dist/src/integrations/linear/unified-sync.js +589 -0
- package/dist/src/integrations/linear/unified-sync.js.map +7 -0
- package/dist/src/integrations/linear/webhook-handler.js +219 -0
- package/dist/src/integrations/linear/webhook-handler.js.map +7 -0
- package/dist/src/integrations/linear/webhook-server.js +218 -0
- package/dist/src/integrations/linear/webhook-server.js.map +7 -0
- package/dist/src/integrations/linear/webhook.js +291 -0
- package/dist/src/integrations/linear/webhook.js.map +7 -0
- package/dist/src/integrations/mcp/handlers/code-execution-handlers.js +266 -0
- package/dist/src/integrations/mcp/handlers/code-execution-handlers.js.map +7 -0
- package/dist/src/integrations/mcp/handlers/context-handlers.js +257 -0
- package/dist/src/integrations/mcp/handlers/context-handlers.js.map +7 -0
- package/dist/src/integrations/mcp/handlers/discovery-handlers.js +497 -0
- package/dist/src/integrations/mcp/handlers/discovery-handlers.js.map +7 -0
- package/dist/src/integrations/mcp/handlers/index.js +166 -0
- package/dist/src/integrations/mcp/handlers/index.js.map +7 -0
- package/dist/src/integrations/mcp/handlers/linear-handlers.js +247 -0
- package/dist/src/integrations/mcp/handlers/linear-handlers.js.map +7 -0
- package/dist/src/integrations/mcp/handlers/skill-handlers.js +529 -0
- package/dist/src/integrations/mcp/handlers/skill-handlers.js.map +7 -0
- package/dist/src/integrations/mcp/handlers/task-handlers.js +239 -0
- package/dist/src/integrations/mcp/handlers/task-handlers.js.map +7 -0
- package/dist/src/integrations/mcp/handlers/trace-handlers.js +308 -0
- package/dist/src/integrations/mcp/handlers/trace-handlers.js.map +7 -0
- package/dist/src/integrations/mcp/index.js +23 -0
- package/dist/src/integrations/mcp/index.js.map +7 -0
- package/dist/src/integrations/mcp/middleware/tool-scoring.js +356 -0
- package/dist/src/integrations/mcp/middleware/tool-scoring.js.map +7 -0
- package/dist/src/integrations/mcp/refactored-server.js +374 -0
- package/dist/src/integrations/mcp/refactored-server.js.map +7 -0
- package/dist/src/integrations/mcp/remote-server.js +682 -0
- package/dist/src/integrations/mcp/remote-server.js.map +7 -0
- package/dist/src/integrations/mcp/schemas.js +147 -0
- package/dist/src/integrations/mcp/schemas.js.map +7 -0
- package/dist/src/integrations/mcp/server.js +1975 -0
- package/dist/src/integrations/mcp/server.js.map +7 -0
- package/dist/src/integrations/mcp/tool-definitions-code.js +125 -0
- package/dist/src/integrations/mcp/tool-definitions-code.js.map +7 -0
- package/dist/src/integrations/mcp/tool-definitions.js +702 -0
- package/dist/src/integrations/mcp/tool-definitions.js.map +7 -0
- package/dist/src/integrations/ralph/bridge/ralph-stackmemory-bridge.js +860 -0
- package/dist/src/integrations/ralph/bridge/ralph-stackmemory-bridge.js.map +7 -0
- package/dist/src/integrations/ralph/context/context-budget-manager.js +301 -0
- package/dist/src/integrations/ralph/context/context-budget-manager.js.map +7 -0
- package/dist/src/integrations/ralph/context/stackmemory-context-loader.js +360 -0
- package/dist/src/integrations/ralph/context/stackmemory-context-loader.js.map +7 -0
- package/dist/src/integrations/ralph/coordination/enhanced-coordination.js +410 -0
- package/dist/src/integrations/ralph/coordination/enhanced-coordination.js.map +7 -0
- package/dist/src/integrations/ralph/index.js +18 -0
- package/dist/src/integrations/ralph/index.js.map +7 -0
- package/dist/src/integrations/ralph/learning/pattern-learner.js +401 -0
- package/dist/src/integrations/ralph/learning/pattern-learner.js.map +7 -0
- package/dist/src/integrations/ralph/lifecycle/iteration-lifecycle.js +448 -0
- package/dist/src/integrations/ralph/lifecycle/iteration-lifecycle.js.map +7 -0
- package/dist/src/integrations/ralph/monitoring/swarm-dashboard.js +294 -0
- package/dist/src/integrations/ralph/monitoring/swarm-dashboard.js.map +7 -0
- package/dist/src/integrations/ralph/monitoring/swarm-registry.js +108 -0
- package/dist/src/integrations/ralph/monitoring/swarm-registry.js.map +7 -0
- package/dist/src/integrations/ralph/orchestration/multi-loop-orchestrator.js +463 -0
- package/dist/src/integrations/ralph/orchestration/multi-loop-orchestrator.js.map +7 -0
- package/dist/src/integrations/ralph/patterns/compounding-engineering-pattern.js +400 -0
- package/dist/src/integrations/ralph/patterns/compounding-engineering-pattern.js.map +7 -0
- package/dist/src/integrations/ralph/patterns/extended-coherence-sessions.js +473 -0
- package/dist/src/integrations/ralph/patterns/extended-coherence-sessions.js.map +7 -0
- package/dist/src/integrations/ralph/patterns/oracle-worker-pattern.js +388 -0
- package/dist/src/integrations/ralph/patterns/oracle-worker-pattern.js.map +7 -0
- package/dist/src/integrations/ralph/performance/performance-optimizer.js +358 -0
- package/dist/src/integrations/ralph/performance/performance-optimizer.js.map +7 -0
- package/dist/src/integrations/ralph/recovery/crash-recovery.js +462 -0
- package/dist/src/integrations/ralph/recovery/crash-recovery.js.map +7 -0
- package/dist/src/integrations/ralph/state/state-reconciler.js +404 -0
- package/dist/src/integrations/ralph/state/state-reconciler.js.map +7 -0
- package/dist/src/integrations/ralph/swarm/git-workflow-manager.js +428 -0
- package/dist/src/integrations/ralph/swarm/git-workflow-manager.js.map +7 -0
- package/dist/src/integrations/ralph/swarm/swarm-coordinator.js +996 -0
- package/dist/src/integrations/ralph/swarm/swarm-coordinator.js.map +7 -0
- package/dist/src/integrations/ralph/types.js +5 -0
- package/dist/src/integrations/ralph/types.js.map +7 -0
- package/dist/src/integrations/ralph/visualization/ralph-debugger.js +585 -0
- package/dist/src/integrations/ralph/visualization/ralph-debugger.js.map +7 -0
- package/dist/src/mcp/stackmemory-mcp-server.js +554 -0
- package/dist/src/mcp/stackmemory-mcp-server.js.map +7 -0
- package/dist/src/middleware/exponential-rate-limiter.js +289 -0
- package/dist/src/middleware/exponential-rate-limiter.js.map +7 -0
- package/dist/src/models/user.model.js +358 -0
- package/dist/src/models/user.model.js.map +7 -0
- package/dist/src/servers/production/auth-middleware.js +528 -0
- package/dist/src/servers/production/auth-middleware.js.map +7 -0
- package/dist/src/services/config-service.js +65 -0
- package/dist/src/services/config-service.js.map +7 -0
- package/dist/src/services/context-service.js +194 -0
- package/dist/src/services/context-service.js.map +7 -0
- package/dist/src/skills/api-discovery.js +354 -0
- package/dist/src/skills/api-discovery.js.map +7 -0
- package/dist/src/skills/api-skill.js +475 -0
- package/dist/src/skills/api-skill.js.map +7 -0
- package/dist/src/skills/claude-skills.js +1061 -0
- package/dist/src/skills/claude-skills.js.map +7 -0
- package/dist/src/skills/dashboard-launcher.js +216 -0
- package/dist/src/skills/dashboard-launcher.js.map +7 -0
- package/dist/src/skills/recursive-agent-orchestrator.js +575 -0
- package/dist/src/skills/recursive-agent-orchestrator.js.map +7 -0
- package/dist/src/skills/repo-ingestion-skill.js +609 -0
- package/dist/src/skills/repo-ingestion-skill.js.map +7 -0
- package/dist/src/skills/unified-rlm-orchestrator.js +404 -0
- package/dist/src/skills/unified-rlm-orchestrator.js.map +7 -0
- package/dist/src/types/task.js +5 -0
- package/dist/src/types/task.js.map +7 -0
- package/dist/src/utils/env.js +50 -0
- package/dist/src/utils/env.js.map +7 -0
- package/dist/src/utils/formatting.js +62 -0
- package/dist/src/utils/formatting.js.map +7 -0
- package/dist/src/utils/process-cleanup.js +136 -0
- package/dist/src/utils/process-cleanup.js.map +7 -0
- package/package.json +3 -3
- package/scripts/initialize.ts +16 -7
- package/scripts/install.sh +14 -62
- package/scripts/status.ts +111 -46
|
@@ -0,0 +1,169 @@
|
|
|
1
|
+
import { fileURLToPath as __fileURLToPath } from 'url';
|
|
2
|
+
import { dirname as __pathDirname } from 'path';
|
|
3
|
+
const __filename = __fileURLToPath(import.meta.url);
|
|
4
|
+
const __dirname = __pathDirname(__filename);
|
|
5
|
+
import { Command } from "commander";
|
|
6
|
+
import Database from "better-sqlite3";
|
|
7
|
+
import { join } from "path";
|
|
8
|
+
import { existsSync, readFileSync } from "fs";
|
|
9
|
+
function createLogCommand() {
|
|
10
|
+
const log = new Command("log").alias("history").description("View recent activity log").option("-n, --lines <n>", "Number of entries to show", "20").option("-t, --type <type>", "Filter by type (task, frame, event, sync)").option("-f, --follow", "Follow log in real-time").action(async (options) => {
|
|
11
|
+
const projectRoot = process.cwd();
|
|
12
|
+
const dbPath = join(projectRoot, ".stackmemory", "context.db");
|
|
13
|
+
const tasksPath = join(projectRoot, ".stackmemory", "tasks.jsonl");
|
|
14
|
+
if (!existsSync(dbPath)) {
|
|
15
|
+
console.log(
|
|
16
|
+
'\u274C StackMemory not initialized. Run "stackmemory init" first.'
|
|
17
|
+
);
|
|
18
|
+
return;
|
|
19
|
+
}
|
|
20
|
+
const limit = parseInt(options.lines);
|
|
21
|
+
const activities = [];
|
|
22
|
+
const db = new Database(dbPath);
|
|
23
|
+
if (!options.type || options.type === "frame") {
|
|
24
|
+
try {
|
|
25
|
+
const frames = db.prepare(
|
|
26
|
+
`
|
|
27
|
+
SELECT id, type, name, state, created_at, updated_at
|
|
28
|
+
FROM frames
|
|
29
|
+
ORDER BY updated_at DESC
|
|
30
|
+
LIMIT ?
|
|
31
|
+
`
|
|
32
|
+
).all(limit);
|
|
33
|
+
frames.forEach((f) => {
|
|
34
|
+
activities.push({
|
|
35
|
+
timestamp: f.updated_at || f.created_at,
|
|
36
|
+
type: "frame",
|
|
37
|
+
action: f.state === "closed" ? "closed" : "opened",
|
|
38
|
+
details: `[${f.type}] ${f.name || f.id.slice(0, 10)}`
|
|
39
|
+
});
|
|
40
|
+
});
|
|
41
|
+
} catch {
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
if (!options.type || options.type === "event") {
|
|
45
|
+
try {
|
|
46
|
+
const events = db.prepare(
|
|
47
|
+
`
|
|
48
|
+
SELECT id, type, data, timestamp
|
|
49
|
+
FROM events
|
|
50
|
+
ORDER BY timestamp DESC
|
|
51
|
+
LIMIT ?
|
|
52
|
+
`
|
|
53
|
+
).all(limit);
|
|
54
|
+
events.forEach((e) => {
|
|
55
|
+
let data = {};
|
|
56
|
+
try {
|
|
57
|
+
data = JSON.parse(e.data);
|
|
58
|
+
} catch {
|
|
59
|
+
}
|
|
60
|
+
activities.push({
|
|
61
|
+
timestamp: e.timestamp,
|
|
62
|
+
type: "event",
|
|
63
|
+
action: e.type,
|
|
64
|
+
details: data.message || data.content || data.decision || ""
|
|
65
|
+
});
|
|
66
|
+
});
|
|
67
|
+
} catch {
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
if (!options.type || options.type === "task") {
|
|
71
|
+
try {
|
|
72
|
+
const tasks = db.prepare(
|
|
73
|
+
`
|
|
74
|
+
SELECT id, title, status, type, timestamp
|
|
75
|
+
FROM task_cache
|
|
76
|
+
ORDER BY timestamp DESC
|
|
77
|
+
LIMIT ?
|
|
78
|
+
`
|
|
79
|
+
).all(limit);
|
|
80
|
+
tasks.forEach((t) => {
|
|
81
|
+
activities.push({
|
|
82
|
+
timestamp: t.timestamp,
|
|
83
|
+
type: "task",
|
|
84
|
+
action: t.type?.replace("task_", "") || t.status,
|
|
85
|
+
details: t.title
|
|
86
|
+
});
|
|
87
|
+
});
|
|
88
|
+
} catch {
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
db.close();
|
|
92
|
+
if (!options.type || options.type === "sync") {
|
|
93
|
+
const mappingsPath = join(
|
|
94
|
+
projectRoot,
|
|
95
|
+
".stackmemory",
|
|
96
|
+
"linear-mappings.json"
|
|
97
|
+
);
|
|
98
|
+
if (existsSync(mappingsPath)) {
|
|
99
|
+
try {
|
|
100
|
+
const mappings = JSON.parse(readFileSync(mappingsPath, "utf-8"));
|
|
101
|
+
mappings.slice(-limit).forEach((m) => {
|
|
102
|
+
activities.push({
|
|
103
|
+
timestamp: Math.floor(m.lastSyncTimestamp / 1e3),
|
|
104
|
+
type: "sync",
|
|
105
|
+
action: "synced",
|
|
106
|
+
details: `${m.linearIdentifier}`
|
|
107
|
+
});
|
|
108
|
+
});
|
|
109
|
+
} catch {
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
activities.sort((a, b) => b.timestamp - a.timestamp);
|
|
114
|
+
console.log(`
|
|
115
|
+
\u{1F4DC} Activity Log
|
|
116
|
+
`);
|
|
117
|
+
const typeIcon = {
|
|
118
|
+
frame: "\u{1F4C1}",
|
|
119
|
+
event: "\u26A1",
|
|
120
|
+
task: "\u{1F4CB}",
|
|
121
|
+
sync: "\u{1F504}"
|
|
122
|
+
};
|
|
123
|
+
activities.slice(0, limit).forEach((activity) => {
|
|
124
|
+
const date = new Date(activity.timestamp * 1e3);
|
|
125
|
+
const timeStr = date.toLocaleTimeString("en-US", {
|
|
126
|
+
hour: "2-digit",
|
|
127
|
+
minute: "2-digit"
|
|
128
|
+
});
|
|
129
|
+
const dateStr = date.toLocaleDateString("en-US", {
|
|
130
|
+
month: "short",
|
|
131
|
+
day: "numeric"
|
|
132
|
+
});
|
|
133
|
+
const icon = typeIcon[activity.type] || "\u{1F4DD}";
|
|
134
|
+
console.log(
|
|
135
|
+
`${icon} ${dateStr} ${timeStr} ${activity.action.padEnd(12)} ${activity.details.slice(0, 50)}`
|
|
136
|
+
);
|
|
137
|
+
});
|
|
138
|
+
console.log("");
|
|
139
|
+
if (options.follow) {
|
|
140
|
+
console.log("\u{1F440} Watching for changes... (Ctrl+C to stop)\n");
|
|
141
|
+
try {
|
|
142
|
+
const chokidar = await import("chokidar");
|
|
143
|
+
const watcher = chokidar.watch(join(projectRoot, ".stackmemory"), {
|
|
144
|
+
persistent: true,
|
|
145
|
+
ignoreInitial: true
|
|
146
|
+
});
|
|
147
|
+
watcher.on("change", (path) => {
|
|
148
|
+
const now = /* @__PURE__ */ new Date();
|
|
149
|
+
const timeStr = now.toLocaleTimeString("en-US", {
|
|
150
|
+
hour: "2-digit",
|
|
151
|
+
minute: "2-digit"
|
|
152
|
+
});
|
|
153
|
+
console.log(
|
|
154
|
+
`\u{1F504} ${timeStr} File changed: ${path.split("/").pop()}`
|
|
155
|
+
);
|
|
156
|
+
});
|
|
157
|
+
await new Promise(() => {
|
|
158
|
+
});
|
|
159
|
+
} catch {
|
|
160
|
+
console.log("Install chokidar for follow mode: npm i chokidar");
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
});
|
|
164
|
+
return log;
|
|
165
|
+
}
|
|
166
|
+
export {
|
|
167
|
+
createLogCommand
|
|
168
|
+
};
|
|
169
|
+
//# sourceMappingURL=log.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../../../src/cli/commands/log.ts"],
|
|
4
|
+
"sourcesContent": ["/**\n * Log Command for StackMemory CLI\n * View recent activity log\n */\n\nimport { Command } from 'commander';\nimport Database from 'better-sqlite3';\nimport { join } from 'path';\nimport { existsSync, readFileSync } from 'fs';\n\nexport function createLogCommand(): Command {\n const log = new Command('log')\n .alias('history')\n .description('View recent activity log')\n .option('-n, --lines <n>', 'Number of entries to show', '20')\n .option('-t, --type <type>', 'Filter by type (task, frame, event, sync)')\n .option('-f, --follow', 'Follow log in real-time')\n .action(async (options) => {\n const projectRoot = process.cwd();\n const dbPath = join(projectRoot, '.stackmemory', 'context.db');\n const tasksPath = join(projectRoot, '.stackmemory', 'tasks.jsonl');\n\n if (!existsSync(dbPath)) {\n console.log(\n '\u274C StackMemory not initialized. Run \"stackmemory init\" first.'\n );\n return;\n }\n\n const limit = parseInt(options.lines);\n const activities: Array<{\n timestamp: number;\n type: string;\n action: string;\n details: string;\n }> = [];\n\n const db = new Database(dbPath);\n\n // Get frame activity\n if (!options.type || options.type === 'frame') {\n try {\n const frames = db\n .prepare(\n `\n SELECT id, type, name, state, created_at, updated_at\n FROM frames \n ORDER BY updated_at DESC\n LIMIT ?\n `\n )\n .all(limit) as any[];\n\n frames.forEach((f) => {\n activities.push({\n timestamp: f.updated_at || f.created_at,\n type: 'frame',\n action: f.state === 'closed' ? 'closed' : 'opened',\n details: `[${f.type}] ${f.name || f.id.slice(0, 10)}`,\n });\n });\n } catch {}\n }\n\n // Get event activity\n if (!options.type || options.type === 'event') {\n try {\n const events = db\n .prepare(\n `\n SELECT id, type, data, timestamp\n FROM events \n ORDER BY timestamp DESC\n LIMIT ?\n `\n )\n .all(limit) as any[];\n\n events.forEach((e) => {\n let data: any = {};\n try {\n data = JSON.parse(e.data);\n } catch {}\n\n activities.push({\n timestamp: e.timestamp,\n type: 'event',\n action: e.type,\n details: data.message || data.content || data.decision || '',\n });\n });\n } catch {}\n }\n\n // Get task activity\n if (!options.type || options.type === 'task') {\n try {\n const tasks = db\n .prepare(\n `\n SELECT id, title, status, type, timestamp\n FROM task_cache \n ORDER BY timestamp DESC\n LIMIT ?\n `\n )\n .all(limit) as any[];\n\n tasks.forEach((t) => {\n activities.push({\n timestamp: t.timestamp,\n type: 'task',\n action: t.type?.replace('task_', '') || t.status,\n details: t.title,\n });\n });\n } catch {}\n }\n\n db.close();\n\n // Get sync activity from Linear mappings\n if (!options.type || options.type === 'sync') {\n const mappingsPath = join(\n projectRoot,\n '.stackmemory',\n 'linear-mappings.json'\n );\n if (existsSync(mappingsPath)) {\n try {\n const mappings = JSON.parse(readFileSync(mappingsPath, 'utf-8'));\n mappings.slice(-limit).forEach((m: any) => {\n activities.push({\n timestamp: Math.floor(m.lastSyncTimestamp / 1000),\n type: 'sync',\n action: 'synced',\n details: `${m.linearIdentifier}`,\n });\n });\n } catch {}\n }\n }\n\n // Sort by timestamp descending\n activities.sort((a, b) => b.timestamp - a.timestamp);\n\n console.log(`\\n\uD83D\uDCDC Activity Log\\n`);\n\n const typeIcon: Record<string, string> = {\n frame: '\uD83D\uDCC1',\n event: '\u26A1',\n task: '\uD83D\uDCCB',\n sync: '\uD83D\uDD04',\n };\n\n activities.slice(0, limit).forEach((activity) => {\n const date = new Date(activity.timestamp * 1000);\n const timeStr = date.toLocaleTimeString('en-US', {\n hour: '2-digit',\n minute: '2-digit',\n });\n const dateStr = date.toLocaleDateString('en-US', {\n month: 'short',\n day: 'numeric',\n });\n const icon = typeIcon[activity.type] || '\uD83D\uDCDD';\n\n console.log(\n `${icon} ${dateStr} ${timeStr} ${activity.action.padEnd(12)} ${activity.details.slice(0, 50)}`\n );\n });\n\n console.log('');\n\n // Follow mode\n if (options.follow) {\n console.log('\uD83D\uDC40 Watching for changes... (Ctrl+C to stop)\\n');\n\n // Watch for file changes\n try {\n const chokidar = await import('chokidar');\n const watcher = chokidar.watch(join(projectRoot, '.stackmemory'), {\n persistent: true,\n ignoreInitial: true,\n });\n\n watcher.on('change', (path: string) => {\n const now = new Date();\n const timeStr = now.toLocaleTimeString('en-US', {\n hour: '2-digit',\n minute: '2-digit',\n });\n console.log(\n `\uD83D\uDD04 ${timeStr} File changed: ${path.split('/').pop()}`\n );\n });\n\n // Keep process alive\n await new Promise(() => {});\n } catch {\n console.log('Install chokidar for follow mode: npm i chokidar');\n }\n }\n });\n\n return log;\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": []
|
|
7
|
+
}
|
|
@@ -0,0 +1,172 @@
|
|
|
1
|
+
import { fileURLToPath as __fileURLToPath } from 'url';
|
|
2
|
+
import { dirname as __pathDirname } from 'path';
|
|
3
|
+
const __filename = __fileURLToPath(import.meta.url);
|
|
4
|
+
const __dirname = __pathDirname(__filename);
|
|
5
|
+
import inquirer from "inquirer";
|
|
6
|
+
import chalk from "chalk";
|
|
7
|
+
import { homedir } from "os";
|
|
8
|
+
import { join } from "path";
|
|
9
|
+
import { existsSync, mkdirSync, writeFileSync, readFileSync } from "fs";
|
|
10
|
+
import open from "open";
|
|
11
|
+
import { IntegrationError, ErrorCode } from "../../core/errors/index.js";
|
|
12
|
+
function registerLoginCommand(program) {
|
|
13
|
+
program.command("login").description("Login to hosted StackMemory service").option("--api-url <url>", "Custom API URL", "https://api.stackmemory.ai").option("--email <email>", "Email address for login").option("--password <password>", "Password (not recommended in CLI)").action(async (options) => {
|
|
14
|
+
const cfgDir = join(homedir(), ".stackmemory");
|
|
15
|
+
if (!existsSync(cfgDir)) mkdirSync(cfgDir, { recursive: true });
|
|
16
|
+
console.log(chalk.cyan("\u{1F510} StackMemory Hosted Service Login\n"));
|
|
17
|
+
const credentials = await inquirer.prompt([
|
|
18
|
+
{
|
|
19
|
+
type: "input",
|
|
20
|
+
name: "email",
|
|
21
|
+
message: "Email:",
|
|
22
|
+
default: options.email,
|
|
23
|
+
validate: (input) => {
|
|
24
|
+
const emailRegex = /^[^\s@]+@[^\s@]+\.[^\s@]+$/;
|
|
25
|
+
return emailRegex.test(input) ? true : "Please enter a valid email";
|
|
26
|
+
}
|
|
27
|
+
},
|
|
28
|
+
{
|
|
29
|
+
type: "password",
|
|
30
|
+
name: "password",
|
|
31
|
+
message: "Password:",
|
|
32
|
+
default: options.password,
|
|
33
|
+
mask: "*",
|
|
34
|
+
validate: (input) => input.length >= 6 ? true : "Password must be at least 6 characters"
|
|
35
|
+
}
|
|
36
|
+
]);
|
|
37
|
+
console.log(chalk.gray("\nAuthenticating with StackMemory API..."));
|
|
38
|
+
try {
|
|
39
|
+
const apiUrl = options.apiUrl || process.env.STACKMEMORY_API_URL || "https://api.stackmemory.ai";
|
|
40
|
+
const response = await fetch(`${apiUrl}/auth/login`, {
|
|
41
|
+
method: "POST",
|
|
42
|
+
headers: {
|
|
43
|
+
"Content-Type": "application/json",
|
|
44
|
+
"User-Agent": "StackMemory-CLI/0.3.19"
|
|
45
|
+
},
|
|
46
|
+
body: JSON.stringify({
|
|
47
|
+
email: credentials.email,
|
|
48
|
+
password: credentials.password
|
|
49
|
+
})
|
|
50
|
+
});
|
|
51
|
+
const data = await response.json();
|
|
52
|
+
if (!response.ok || !data.success) {
|
|
53
|
+
if (response.status === 404) {
|
|
54
|
+
console.log(
|
|
55
|
+
chalk.yellow("\n\u26A0\uFE0F Hosted API not available. Would you like to:")
|
|
56
|
+
);
|
|
57
|
+
const { choice } = await inquirer.prompt([
|
|
58
|
+
{
|
|
59
|
+
type: "list",
|
|
60
|
+
name: "choice",
|
|
61
|
+
message: "Select an option:",
|
|
62
|
+
choices: [
|
|
63
|
+
{ name: "Open signup page in browser", value: "signup" },
|
|
64
|
+
{ name: "Configure database URL manually", value: "manual" },
|
|
65
|
+
{ name: "Use local database", value: "local" },
|
|
66
|
+
{ name: "Cancel", value: "cancel" }
|
|
67
|
+
]
|
|
68
|
+
}
|
|
69
|
+
]);
|
|
70
|
+
if (choice === "signup") {
|
|
71
|
+
await open("https://stackmemory.ai/signup");
|
|
72
|
+
console.log(chalk.cyan("Opening signup page in browser..."));
|
|
73
|
+
return;
|
|
74
|
+
} else if (choice === "manual") {
|
|
75
|
+
const { databaseUrl } = await inquirer.prompt([
|
|
76
|
+
{
|
|
77
|
+
type: "password",
|
|
78
|
+
name: "databaseUrl",
|
|
79
|
+
message: "Enter your DATABASE_URL (postgres://...):",
|
|
80
|
+
validate: (input) => input.startsWith("postgres://") || input.startsWith("postgresql://") ? true : "Must start with postgres:// or postgresql://"
|
|
81
|
+
}
|
|
82
|
+
]);
|
|
83
|
+
const cfgPath2 = join(cfgDir, "config.json");
|
|
84
|
+
let cfg2 = {};
|
|
85
|
+
try {
|
|
86
|
+
if (existsSync(cfgPath2))
|
|
87
|
+
cfg2 = JSON.parse(readFileSync(cfgPath2, "utf-8"));
|
|
88
|
+
} catch {
|
|
89
|
+
}
|
|
90
|
+
cfg2.database = { mode: "hosted", url: databaseUrl };
|
|
91
|
+
cfg2.auth = { email: credentials.email };
|
|
92
|
+
writeFileSync(cfgPath2, JSON.stringify(cfg2, null, 2));
|
|
93
|
+
console.log(chalk.green("\u2713 Database configured successfully"));
|
|
94
|
+
return;
|
|
95
|
+
} else if (choice === "local") {
|
|
96
|
+
const cfgPath2 = join(cfgDir, "config.json");
|
|
97
|
+
let cfg2 = {};
|
|
98
|
+
try {
|
|
99
|
+
if (existsSync(cfgPath2))
|
|
100
|
+
cfg2 = JSON.parse(readFileSync(cfgPath2, "utf-8"));
|
|
101
|
+
} catch {
|
|
102
|
+
}
|
|
103
|
+
cfg2.database = { mode: "local" };
|
|
104
|
+
writeFileSync(cfgPath2, JSON.stringify(cfg2, null, 2));
|
|
105
|
+
console.log(chalk.green("\u2713 Switched to local database mode"));
|
|
106
|
+
return;
|
|
107
|
+
} else {
|
|
108
|
+
console.log(chalk.gray("Login cancelled"));
|
|
109
|
+
return;
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
throw new IntegrationError(
|
|
113
|
+
data.error || "Authentication failed",
|
|
114
|
+
ErrorCode.LINEAR_AUTH_FAILED,
|
|
115
|
+
{ email: credentials.email, apiUrl }
|
|
116
|
+
);
|
|
117
|
+
}
|
|
118
|
+
const cfgPath = join(cfgDir, "config.json");
|
|
119
|
+
let cfg = {};
|
|
120
|
+
try {
|
|
121
|
+
if (existsSync(cfgPath))
|
|
122
|
+
cfg = JSON.parse(readFileSync(cfgPath, "utf-8"));
|
|
123
|
+
} catch {
|
|
124
|
+
}
|
|
125
|
+
cfg.auth = {
|
|
126
|
+
apiKey: data.apiKey,
|
|
127
|
+
apiUrl,
|
|
128
|
+
email: credentials.email
|
|
129
|
+
};
|
|
130
|
+
if (data.databaseUrl) {
|
|
131
|
+
cfg.database = {
|
|
132
|
+
mode: "hosted",
|
|
133
|
+
url: data.databaseUrl
|
|
134
|
+
};
|
|
135
|
+
}
|
|
136
|
+
writeFileSync(cfgPath, JSON.stringify(cfg, null, 2));
|
|
137
|
+
const envFile = join(cfgDir, "stackmemory.env");
|
|
138
|
+
const envContent = `# StackMemory Authentication
|
|
139
|
+
STACKMEMORY_API_KEY=${data.apiKey}
|
|
140
|
+
STACKMEMORY_API_URL=${apiUrl}
|
|
141
|
+
${data.databaseUrl ? `DATABASE_URL=${data.databaseUrl}` : ""}
|
|
142
|
+
`;
|
|
143
|
+
writeFileSync(envFile, envContent);
|
|
144
|
+
console.log(chalk.green("\n\u2705 Successfully logged in to StackMemory"));
|
|
145
|
+
console.log(
|
|
146
|
+
chalk.green(`\u2713 Configuration saved to ~/.stackmemory/config.json`)
|
|
147
|
+
);
|
|
148
|
+
console.log(chalk.gray("\nYou can now use:"));
|
|
149
|
+
console.log(
|
|
150
|
+
chalk.cyan(" stackmemory sync ") + chalk.gray("- Sync your context to the cloud")
|
|
151
|
+
);
|
|
152
|
+
console.log(
|
|
153
|
+
chalk.cyan(" stackmemory db status") + chalk.gray("- Check database connection")
|
|
154
|
+
);
|
|
155
|
+
console.log(
|
|
156
|
+
chalk.cyan(" stackmemory context ") + chalk.gray("- Manage your contexts")
|
|
157
|
+
);
|
|
158
|
+
} catch (error) {
|
|
159
|
+
console.error(chalk.red("\n\u274C Login failed:"), error.message);
|
|
160
|
+
console.log(
|
|
161
|
+
chalk.yellow(
|
|
162
|
+
"\nTip: Visit https://stackmemory.ai/signup to create an account"
|
|
163
|
+
)
|
|
164
|
+
);
|
|
165
|
+
process.exit(1);
|
|
166
|
+
}
|
|
167
|
+
});
|
|
168
|
+
}
|
|
169
|
+
export {
|
|
170
|
+
registerLoginCommand
|
|
171
|
+
};
|
|
172
|
+
//# sourceMappingURL=login.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../../../src/cli/commands/login.ts"],
|
|
4
|
+
"sourcesContent": ["import { Command } from 'commander';\nimport inquirer from 'inquirer';\nimport chalk from 'chalk';\nimport { homedir } from 'os';\nimport { join } from 'path';\nimport { existsSync, mkdirSync, writeFileSync, readFileSync } from 'fs';\nimport open from 'open';\nimport { IntegrationError, ErrorCode } from '../../core/errors/index.js';\n\ninterface ConfigShape {\n version?: string;\n setupCompleted?: string;\n features?: any;\n paths?: any;\n database?: { mode?: 'local' | 'hosted'; url?: string };\n auth?: {\n apiKey?: string;\n apiUrl?: string;\n email?: string;\n };\n}\n\ninterface AuthResponse {\n success: boolean;\n apiKey?: string;\n databaseUrl?: string;\n email?: string;\n error?: string;\n}\n\nexport function registerLoginCommand(program: Command): void {\n program\n .command('login')\n .description('Login to hosted StackMemory service')\n .option('--api-url <url>', 'Custom API URL', 'https://api.stackmemory.ai')\n .option('--email <email>', 'Email address for login')\n .option('--password <password>', 'Password (not recommended in CLI)')\n .action(async (options) => {\n const cfgDir = join(homedir(), '.stackmemory');\n if (!existsSync(cfgDir)) mkdirSync(cfgDir, { recursive: true });\n\n console.log(chalk.cyan('\uD83D\uDD10 StackMemory Hosted Service Login\\n'));\n\n // Prompt for credentials\n const credentials = await inquirer.prompt([\n {\n type: 'input',\n name: 'email',\n message: 'Email:',\n default: options.email,\n validate: (input: string) => {\n const emailRegex = /^[^\\s@]+@[^\\s@]+\\.[^\\s@]+$/;\n return emailRegex.test(input) ? true : 'Please enter a valid email';\n },\n },\n {\n type: 'password',\n name: 'password',\n message: 'Password:',\n default: options.password,\n mask: '*',\n validate: (input: string) =>\n input.length >= 6 ? true : 'Password must be at least 6 characters',\n },\n ]);\n\n console.log(chalk.gray('\\nAuthenticating with StackMemory API...'));\n\n try {\n // Authenticate with the hosted API\n const apiUrl =\n options.apiUrl ||\n process.env.STACKMEMORY_API_URL ||\n 'https://api.stackmemory.ai';\n const response = await fetch(`${apiUrl}/auth/login`, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'User-Agent': 'StackMemory-CLI/0.3.19',\n },\n body: JSON.stringify({\n email: credentials.email,\n password: credentials.password,\n }),\n });\n\n const data: AuthResponse = await response.json();\n\n if (!response.ok || !data.success) {\n if (response.status === 404) {\n // Fallback to Railway server if hosted API not available\n console.log(\n chalk.yellow('\\n\u26A0\uFE0F Hosted API not available. Would you like to:')\n );\n const { choice } = await inquirer.prompt([\n {\n type: 'list',\n name: 'choice',\n message: 'Select an option:',\n choices: [\n { name: 'Open signup page in browser', value: 'signup' },\n { name: 'Configure database URL manually', value: 'manual' },\n { name: 'Use local database', value: 'local' },\n { name: 'Cancel', value: 'cancel' },\n ],\n },\n ]);\n\n if (choice === 'signup') {\n await open('https://stackmemory.ai/signup');\n console.log(chalk.cyan('Opening signup page in browser...'));\n return;\n } else if (choice === 'manual') {\n const { databaseUrl } = await inquirer.prompt([\n {\n type: 'password',\n name: 'databaseUrl',\n message: 'Enter your DATABASE_URL (postgres://...):',\n validate: (input: string) =>\n input.startsWith('postgres://') ||\n input.startsWith('postgresql://')\n ? true\n : 'Must start with postgres:// or postgresql://',\n },\n ]);\n\n // Save manual configuration\n const cfgPath = join(cfgDir, 'config.json');\n let cfg: ConfigShape = {};\n try {\n if (existsSync(cfgPath))\n cfg = JSON.parse(readFileSync(cfgPath, 'utf-8'));\n } catch {}\n\n cfg.database = { mode: 'hosted', url: databaseUrl };\n cfg.auth = { email: credentials.email };\n\n writeFileSync(cfgPath, JSON.stringify(cfg, null, 2));\n console.log(chalk.green('\u2713 Database configured successfully'));\n return;\n } else if (choice === 'local') {\n const cfgPath = join(cfgDir, 'config.json');\n let cfg: ConfigShape = {};\n try {\n if (existsSync(cfgPath))\n cfg = JSON.parse(readFileSync(cfgPath, 'utf-8'));\n } catch {}\n\n cfg.database = { mode: 'local' };\n writeFileSync(cfgPath, JSON.stringify(cfg, null, 2));\n console.log(chalk.green('\u2713 Switched to local database mode'));\n return;\n } else {\n console.log(chalk.gray('Login cancelled'));\n return;\n }\n }\n\n throw new IntegrationError(\n data.error || 'Authentication failed',\n ErrorCode.LINEAR_AUTH_FAILED,\n { email: credentials.email, apiUrl }\n );\n }\n\n // Save configuration\n const cfgPath = join(cfgDir, 'config.json');\n let cfg: ConfigShape = {};\n try {\n if (existsSync(cfgPath))\n cfg = JSON.parse(readFileSync(cfgPath, 'utf-8'));\n } catch {}\n\n cfg.auth = {\n apiKey: data.apiKey,\n apiUrl: apiUrl,\n email: credentials.email,\n };\n\n if (data.databaseUrl) {\n cfg.database = {\n mode: 'hosted',\n url: data.databaseUrl,\n };\n }\n\n writeFileSync(cfgPath, JSON.stringify(cfg, null, 2));\n\n // Save environment variables\n const envFile = join(cfgDir, 'stackmemory.env');\n const envContent = `# StackMemory Authentication\nSTACKMEMORY_API_KEY=${data.apiKey}\nSTACKMEMORY_API_URL=${apiUrl}\n${data.databaseUrl ? `DATABASE_URL=${data.databaseUrl}` : ''}\n`;\n writeFileSync(envFile, envContent);\n\n console.log(chalk.green('\\n\u2705 Successfully logged in to StackMemory'));\n console.log(\n chalk.green(`\u2713 Configuration saved to ~/.stackmemory/config.json`)\n );\n console.log(chalk.gray('\\nYou can now use:'));\n console.log(\n chalk.cyan(' stackmemory sync ') +\n chalk.gray('- Sync your context to the cloud')\n );\n console.log(\n chalk.cyan(' stackmemory db status') +\n chalk.gray('- Check database connection')\n );\n console.log(\n chalk.cyan(' stackmemory context ') +\n chalk.gray('- Manage your contexts')\n );\n } catch (error: any) {\n console.error(chalk.red('\\n\u274C Login failed:'), error.message);\n console.log(\n chalk.yellow(\n '\\nTip: Visit https://stackmemory.ai/signup to create an account'\n )\n );\n process.exit(1);\n }\n });\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": ["cfgPath", "cfg"]
|
|
7
|
+
}
|
|
@@ -0,0 +1,240 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
import { fileURLToPath as __fileURLToPath } from 'url';
|
|
3
|
+
import { dirname as __pathDirname } from 'path';
|
|
4
|
+
const __filename = __fileURLToPath(import.meta.url);
|
|
5
|
+
const __dirname = __pathDirname(__filename);
|
|
6
|
+
import { Command } from "commander";
|
|
7
|
+
import { Pool } from "pg";
|
|
8
|
+
import Database from "better-sqlite3";
|
|
9
|
+
import {
|
|
10
|
+
DatabaseError,
|
|
11
|
+
ValidationError,
|
|
12
|
+
ErrorCode
|
|
13
|
+
} from "../../core/errors/index.js";
|
|
14
|
+
const MIGRATIONS = [
|
|
15
|
+
{
|
|
16
|
+
version: 1,
|
|
17
|
+
description: "base schema",
|
|
18
|
+
statements: [
|
|
19
|
+
// contexts
|
|
20
|
+
`CREATE TABLE IF NOT EXISTS contexts (id ${isPg() ? "BIGSERIAL" : "INTEGER PRIMARY KEY AUTOINCREMENT"} PRIMARY KEY, project_id TEXT NOT NULL, content TEXT NOT NULL, type TEXT DEFAULT 'general', ${isPg() ? "metadata JSONB DEFAULT '{}'::jsonb" : "metadata TEXT DEFAULT '{}'"}, created_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"}, updated_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"})`,
|
|
21
|
+
// api_keys
|
|
22
|
+
`CREATE TABLE IF NOT EXISTS api_keys (id ${isPg() ? "BIGSERIAL" : "INTEGER PRIMARY KEY AUTOINCREMENT"} PRIMARY KEY, key_hash TEXT UNIQUE NOT NULL, user_id TEXT NOT NULL, name TEXT, created_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"}, ${isPg() ? "last_used TIMESTAMPTZ" : "last_used DATETIME"}, revoked ${isPg() ? "BOOLEAN" : "BOOLEAN"} DEFAULT ${isPg() ? "false" : "0"})`,
|
|
23
|
+
// users with role
|
|
24
|
+
`CREATE TABLE IF NOT EXISTS users (id TEXT PRIMARY KEY, email TEXT, name TEXT, tier TEXT DEFAULT 'free', role TEXT DEFAULT 'user', created_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"}, updated_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"})`,
|
|
25
|
+
// projects
|
|
26
|
+
`CREATE TABLE IF NOT EXISTS projects (id TEXT PRIMARY KEY, name TEXT, is_public ${isPg() ? "BOOLEAN" : "BOOLEAN"} DEFAULT ${isPg() ? "false" : "0"}, created_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"}, updated_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"})`,
|
|
27
|
+
// project members
|
|
28
|
+
`CREATE TABLE IF NOT EXISTS project_members (project_id TEXT NOT NULL, user_id TEXT NOT NULL, role TEXT NOT NULL ${isPg() ? "" : "CHECK (role IN ('admin','owner','editor','viewer'))"}, created_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"}, PRIMARY KEY (project_id, user_id))`,
|
|
29
|
+
// indexes
|
|
30
|
+
`CREATE INDEX IF NOT EXISTS idx_contexts_project ON contexts(project_id)`,
|
|
31
|
+
`CREATE INDEX IF NOT EXISTS idx_api_keys_hash ON api_keys(key_hash)`,
|
|
32
|
+
`CREATE INDEX IF NOT EXISTS idx_users_email ON users(email)`,
|
|
33
|
+
`CREATE INDEX IF NOT EXISTS idx_project_members_user ON project_members(user_id)`
|
|
34
|
+
]
|
|
35
|
+
},
|
|
36
|
+
{
|
|
37
|
+
version: 2,
|
|
38
|
+
description: "admin sessions",
|
|
39
|
+
statements: [
|
|
40
|
+
`CREATE TABLE IF NOT EXISTS admin_sessions (id TEXT PRIMARY KEY, user_id TEXT NOT NULL, created_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} DEFAULT ${isPg() ? "NOW()" : "CURRENT_TIMESTAMP"}, expires_at ${isPg() ? "TIMESTAMPTZ" : "DATETIME"} NOT NULL, user_agent TEXT, ip TEXT)`,
|
|
41
|
+
`CREATE INDEX IF NOT EXISTS idx_admin_sessions_user ON admin_sessions(user_id)`
|
|
42
|
+
]
|
|
43
|
+
},
|
|
44
|
+
{
|
|
45
|
+
version: 3,
|
|
46
|
+
description: "role enums & checks",
|
|
47
|
+
statements: [
|
|
48
|
+
// PG enum upgrades; for SQLite CHECK already present
|
|
49
|
+
`CREATE TYPE user_role AS ENUM ('admin','user')`,
|
|
50
|
+
`CREATE TYPE member_role AS ENUM ('admin','owner','editor','viewer')`,
|
|
51
|
+
`ALTER TABLE users ALTER COLUMN role TYPE user_role USING role::user_role`,
|
|
52
|
+
`ALTER TABLE project_members ALTER COLUMN role TYPE member_role USING role::member_role`,
|
|
53
|
+
`ALTER TABLE project_members ADD CONSTRAINT project_members_role_check CHECK (role IN ('admin','owner','editor','viewer'))`,
|
|
54
|
+
`ALTER TABLE users ADD CONSTRAINT users_role_check CHECK (role IN ('admin','user'))`
|
|
55
|
+
]
|
|
56
|
+
}
|
|
57
|
+
];
|
|
58
|
+
function isPg() {
|
|
59
|
+
const url = process.env.DATABASE_URL || "";
|
|
60
|
+
return url.startsWith("postgres://") || url.startsWith("postgresql://");
|
|
61
|
+
}
|
|
62
|
+
async function connect() {
|
|
63
|
+
if (isPg()) {
|
|
64
|
+
const pool = new Pool({ connectionString: process.env.DATABASE_URL });
|
|
65
|
+
await pool.query(
|
|
66
|
+
`CREATE TABLE IF NOT EXISTS railway_schema_version (version INTEGER PRIMARY KEY, applied_at TIMESTAMPTZ DEFAULT NOW(), description TEXT)`
|
|
67
|
+
);
|
|
68
|
+
return { kind: "pg", pg: pool };
|
|
69
|
+
} else {
|
|
70
|
+
const path = process.env.DATABASE_URL || ".stackmemory/railway.db";
|
|
71
|
+
const db = new Database(path);
|
|
72
|
+
db.exec(
|
|
73
|
+
`CREATE TABLE IF NOT EXISTS railway_schema_version (version INTEGER PRIMARY KEY, applied_at DATETIME DEFAULT CURRENT_TIMESTAMP, description TEXT)`
|
|
74
|
+
);
|
|
75
|
+
return { kind: "sqlite", sqlite: db };
|
|
76
|
+
}
|
|
77
|
+
}
|
|
78
|
+
async function getCurrentVersion(m) {
|
|
79
|
+
if (m.kind === "pg") {
|
|
80
|
+
const r = await m.pg.query(
|
|
81
|
+
"SELECT COALESCE(MAX(version), 0) AS v FROM railway_schema_version"
|
|
82
|
+
);
|
|
83
|
+
return Number(r.rows[0]?.v || 0);
|
|
84
|
+
}
|
|
85
|
+
const row = m.sqlite.prepare(
|
|
86
|
+
"SELECT COALESCE(MAX(version), 0) AS v FROM railway_schema_version"
|
|
87
|
+
).get();
|
|
88
|
+
return Number(row?.v || 0);
|
|
89
|
+
}
|
|
90
|
+
async function listApplied(m) {
|
|
91
|
+
if (m.kind === "pg") {
|
|
92
|
+
const r = await m.pg.query(
|
|
93
|
+
"SELECT version, description, applied_at FROM railway_schema_version ORDER BY version ASC"
|
|
94
|
+
);
|
|
95
|
+
return r.rows.map((row) => ({
|
|
96
|
+
version: Number(row.version),
|
|
97
|
+
description: row.description
|
|
98
|
+
}));
|
|
99
|
+
}
|
|
100
|
+
const rows = m.sqlite.prepare(
|
|
101
|
+
"SELECT version, description, applied_at FROM railway_schema_version ORDER BY version ASC"
|
|
102
|
+
).all();
|
|
103
|
+
return rows.map((row) => ({
|
|
104
|
+
version: Number(row.version),
|
|
105
|
+
description: row.description
|
|
106
|
+
}));
|
|
107
|
+
}
|
|
108
|
+
async function applyTo(m, target) {
|
|
109
|
+
const current = await getCurrentVersion(m);
|
|
110
|
+
const pending = MIGRATIONS.filter(
|
|
111
|
+
(mig) => mig.version > current && mig.version <= target
|
|
112
|
+
);
|
|
113
|
+
for (const mig of pending) {
|
|
114
|
+
if (m.kind === "pg") {
|
|
115
|
+
for (const s of mig.statements) {
|
|
116
|
+
try {
|
|
117
|
+
await m.pg.query(s);
|
|
118
|
+
} catch {
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
await m.pg.query(
|
|
122
|
+
"INSERT INTO railway_schema_version (version, description) VALUES ($1, $2) ON CONFLICT (version) DO NOTHING",
|
|
123
|
+
[mig.version, mig.description]
|
|
124
|
+
);
|
|
125
|
+
} else {
|
|
126
|
+
m.sqlite.exec("BEGIN");
|
|
127
|
+
try {
|
|
128
|
+
for (const s of mig.statements) {
|
|
129
|
+
try {
|
|
130
|
+
m.sqlite.exec(s);
|
|
131
|
+
} catch {
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
m.sqlite.prepare(
|
|
135
|
+
"INSERT OR IGNORE INTO railway_schema_version (version, description) VALUES (?, ?)"
|
|
136
|
+
).run(mig.version, mig.description);
|
|
137
|
+
m.sqlite.exec("COMMIT");
|
|
138
|
+
} catch {
|
|
139
|
+
m.sqlite.exec("ROLLBACK");
|
|
140
|
+
throw new DatabaseError(
|
|
141
|
+
`Migration ${mig.version} failed`,
|
|
142
|
+
ErrorCode.DB_MIGRATION_FAILED,
|
|
143
|
+
{ version: mig.version, description: mig.description }
|
|
144
|
+
);
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
console.log(`Applied migration v${mig.version}: ${mig.description}`);
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
async function rollbackTo(m, target) {
|
|
151
|
+
const current = await getCurrentVersion(m);
|
|
152
|
+
if (target >= current) {
|
|
153
|
+
console.log("Nothing to rollback");
|
|
154
|
+
return;
|
|
155
|
+
}
|
|
156
|
+
if (m.kind === "pg") {
|
|
157
|
+
await m.pg.query("DELETE FROM railway_schema_version WHERE version > $1", [
|
|
158
|
+
target
|
|
159
|
+
]);
|
|
160
|
+
} else {
|
|
161
|
+
m.sqlite.prepare(
|
|
162
|
+
"DELETE FROM railway_schema_version WHERE version > ?"
|
|
163
|
+
).run(target);
|
|
164
|
+
}
|
|
165
|
+
console.log(
|
|
166
|
+
`Rolled back schema version pointer from ${current} to ${target}`
|
|
167
|
+
);
|
|
168
|
+
}
|
|
169
|
+
async function main() {
|
|
170
|
+
const program = new Command();
|
|
171
|
+
program.name("railway-migrate").description("Manage Railway server schema migrations").option("-d, --database <url>", "DATABASE_URL override");
|
|
172
|
+
program.command("list").description("List applied migrations").action(async () => {
|
|
173
|
+
if (program.opts().database)
|
|
174
|
+
process.env.DATABASE_URL = program.opts().database;
|
|
175
|
+
const m = await connect();
|
|
176
|
+
const applied = await listApplied(m);
|
|
177
|
+
const current = await getCurrentVersion(m);
|
|
178
|
+
console.log("Current version:", current);
|
|
179
|
+
if (applied.length === 0) console.log("(no migrations applied)");
|
|
180
|
+
applied.forEach((a) => console.log(`v${a.version} - ${a.description}`));
|
|
181
|
+
process.exit(0);
|
|
182
|
+
});
|
|
183
|
+
program.command("status").description("Show current version and pending migrations").action(async () => {
|
|
184
|
+
if (program.opts().database)
|
|
185
|
+
process.env.DATABASE_URL = program.opts().database;
|
|
186
|
+
const m = await connect();
|
|
187
|
+
const current = await getCurrentVersion(m);
|
|
188
|
+
const latest = Math.max(...MIGRATIONS.map((m2) => m2.version));
|
|
189
|
+
const pending = MIGRATIONS.filter((mig) => mig.version > current);
|
|
190
|
+
console.log("Current version:", current);
|
|
191
|
+
console.log("Latest available:", latest);
|
|
192
|
+
if (pending.length === 0) console.log("No pending migrations.");
|
|
193
|
+
else {
|
|
194
|
+
console.log("Pending:");
|
|
195
|
+
pending.forEach((p) => console.log(`- v${p.version} ${p.description}`));
|
|
196
|
+
}
|
|
197
|
+
process.exit(0);
|
|
198
|
+
});
|
|
199
|
+
program.command("apply").description("Apply migrations up to a target").option(
|
|
200
|
+
"--to <version|latest>",
|
|
201
|
+
'Target version (number or "latest")',
|
|
202
|
+
"latest"
|
|
203
|
+
).action(async (cmd) => {
|
|
204
|
+
if (program.opts().database)
|
|
205
|
+
process.env.DATABASE_URL = program.opts().database;
|
|
206
|
+
const m = await connect();
|
|
207
|
+
const latest = Math.max(...MIGRATIONS.map((m2) => m2.version));
|
|
208
|
+
const target = cmd.to === "latest" ? latest : parseInt(cmd.to, 10);
|
|
209
|
+
if (!Number.isFinite(target))
|
|
210
|
+
throw new ValidationError(
|
|
211
|
+
"Invalid target version",
|
|
212
|
+
ErrorCode.INVALID_INPUT,
|
|
213
|
+
{ target: cmd.to }
|
|
214
|
+
);
|
|
215
|
+
await applyTo(m, target);
|
|
216
|
+
console.log("Done.");
|
|
217
|
+
process.exit(0);
|
|
218
|
+
});
|
|
219
|
+
program.command("rollback").description("Rollback schema version pointer (non-destructive)").option("--to <version>", "Target version number", "0").action(async (cmd) => {
|
|
220
|
+
if (program.opts().database)
|
|
221
|
+
process.env.DATABASE_URL = program.opts().database;
|
|
222
|
+
const m = await connect();
|
|
223
|
+
const target = parseInt(cmd.to, 10);
|
|
224
|
+
if (!Number.isFinite(target))
|
|
225
|
+
throw new ValidationError(
|
|
226
|
+
"Invalid target version",
|
|
227
|
+
ErrorCode.INVALID_INPUT,
|
|
228
|
+
{ target: cmd.to }
|
|
229
|
+
);
|
|
230
|
+
await rollbackTo(m, target);
|
|
231
|
+
console.log("Done.");
|
|
232
|
+
process.exit(0);
|
|
233
|
+
});
|
|
234
|
+
await program.parseAsync(process.argv);
|
|
235
|
+
}
|
|
236
|
+
main().catch((e) => {
|
|
237
|
+
console.error(e);
|
|
238
|
+
process.exit(1);
|
|
239
|
+
});
|
|
240
|
+
//# sourceMappingURL=migrate.js.map
|