cognitive-core 0.2.1 → 0.2.2
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/atlas.d.ts +10 -0
- package/dist/atlas.d.ts.map +1 -1
- package/dist/atlas.js +65 -0
- package/dist/atlas.js.map +1 -1
- package/dist/learning/pipeline.d.ts +4 -31
- package/dist/learning/pipeline.d.ts.map +1 -1
- package/dist/learning/pipeline.js +12 -64
- package/dist/learning/pipeline.js.map +1 -1
- package/dist/memory/curated-loader.d.ts +21 -4
- package/dist/memory/curated-loader.d.ts.map +1 -1
- package/dist/memory/curated-loader.js +53 -16
- package/dist/memory/curated-loader.js.map +1 -1
- package/dist/memory/index.d.ts +2 -1
- package/dist/memory/index.d.ts.map +1 -1
- package/dist/memory/index.js +3 -1
- package/dist/memory/index.js.map +1 -1
- package/dist/memory/playbook.d.ts +6 -0
- package/dist/memory/playbook.d.ts.map +1 -1
- package/dist/memory/playbook.js +15 -0
- package/dist/memory/playbook.js.map +1 -1
- package/dist/memory/source-resolver.d.ts +120 -0
- package/dist/memory/source-resolver.d.ts.map +1 -0
- package/dist/memory/source-resolver.js +300 -0
- package/dist/memory/source-resolver.js.map +1 -0
- package/dist/types/config.d.ts +141 -0
- package/dist/types/config.d.ts.map +1 -1
- package/dist/types/config.js +40 -0
- package/dist/types/config.js.map +1 -1
- package/dist/types/index.d.ts +1 -1
- package/dist/types/index.d.ts.map +1 -1
- package/dist/types/index.js +1 -1
- package/dist/types/index.js.map +1 -1
- package/dist/workspace/types.d.ts +12 -54
- package/dist/workspace/types.d.ts.map +1 -1
- package/dist/workspace/types.js.map +1 -1
- package/package.json +2 -2
- package/playbooks/compound-engineering/adversarial-review.json +51 -0
- package/playbooks/compound-engineering/agent-native-architecture.json +59 -0
- package/playbooks/compound-engineering/agent-native-review.json +54 -0
- package/playbooks/compound-engineering/api-contract-review.json +52 -0
- package/playbooks/compound-engineering/brainstorm-requirements.json +55 -0
- package/playbooks/compound-engineering/bug-reproduction.json +62 -0
- package/playbooks/compound-engineering/confidence-calibration.json +49 -0
- package/playbooks/compound-engineering/correctness-review.json +49 -0
- package/playbooks/compound-engineering/data-migration-safety.json +59 -0
- package/playbooks/compound-engineering/deployment-verification.json +63 -0
- package/playbooks/compound-engineering/error-recovery-patterns.json +53 -0
- package/playbooks/compound-engineering/implementation-planning.json +64 -0
- package/playbooks/compound-engineering/issue-pattern-analysis.json +53 -0
- package/playbooks/compound-engineering/knowledge-compounding.json +63 -0
- package/playbooks/compound-engineering/learnings-research.json +54 -0
- package/playbooks/compound-engineering/maintainability-review.json +49 -0
- package/playbooks/compound-engineering/performance-review.json +54 -0
- package/playbooks/compound-engineering/plan-adversarial-review.json +56 -0
- package/playbooks/compound-engineering/plan-feasibility-review.json +56 -0
- package/playbooks/compound-engineering/project-standards-review.json +52 -0
- package/playbooks/compound-engineering/reliability-review.json +53 -0
- package/playbooks/compound-engineering/review-orchestration.json +64 -0
- package/playbooks/compound-engineering/security-review.json +54 -0
- package/playbooks/compound-engineering/systematic-execution.json +64 -0
- package/playbooks/compound-engineering/testing-review.json +50 -0
- package/src/atlas.ts +96 -0
- package/src/memory/curated-loader.ts +69 -16
- package/src/memory/index.ts +16 -0
- package/src/memory/playbook.ts +19 -0
- package/src/memory/source-resolver.ts +422 -0
- package/src/types/config.ts +46 -0
- package/src/types/index.ts +4 -0
- package/src/workspace/types.ts +22 -78
- package/tests/integration/curated-sources-e2e.test.ts +502 -0
- package/tests/memory/compound-engineering-seed.test.ts +338 -0
- package/tests/memory/curated-loader-extended.test.ts +225 -0
- package/tests/memory/playbook-quality-validation.test.ts +430 -0
- package/tests/memory/source-resolver.test.ts +700 -0
- package/.claude/settings.local.json +0 -11
- package/dist/learning/llm-extractor.d.ts +0 -88
- package/dist/learning/llm-extractor.d.ts.map +0 -1
- package/dist/learning/llm-extractor.js +0 -372
- package/dist/learning/llm-extractor.js.map +0 -1
- package/dist/learning/loop-coordinator.d.ts +0 -61
- package/dist/learning/loop-coordinator.d.ts.map +0 -1
- package/dist/learning/loop-coordinator.js +0 -96
- package/dist/learning/loop-coordinator.js.map +0 -1
- package/references/agent-workspace/CLAUDE.md +0 -74
- package/references/agent-workspace/README.md +0 -587
- package/references/agent-workspace/media/banner.png +0 -0
- package/references/agent-workspace/package-lock.json +0 -2061
- package/references/agent-workspace/package.json +0 -54
- package/references/agent-workspace/src/handle.ts +0 -122
- package/references/agent-workspace/src/index.ts +0 -32
- package/references/agent-workspace/src/manager.ts +0 -102
- package/references/agent-workspace/src/readers/json.ts +0 -71
- package/references/agent-workspace/src/readers/markdown.ts +0 -37
- package/references/agent-workspace/src/readers/raw.ts +0 -27
- package/references/agent-workspace/src/types.ts +0 -68
- package/references/agent-workspace/src/validation.ts +0 -93
- package/references/agent-workspace/src/writers/json.ts +0 -17
- package/references/agent-workspace/src/writers/markdown.ts +0 -27
- package/references/agent-workspace/src/writers/raw.ts +0 -22
- package/references/agent-workspace/tests/errors.test.ts +0 -652
- package/references/agent-workspace/tests/handle.test.ts +0 -144
- package/references/agent-workspace/tests/manager.test.ts +0 -124
- package/references/agent-workspace/tests/readers.test.ts +0 -205
- package/references/agent-workspace/tests/validation.test.ts +0 -196
- package/references/agent-workspace/tests/writers.test.ts +0 -108
- package/references/agent-workspace/tsconfig.json +0 -20
- package/references/agent-workspace/tsup.config.ts +0 -9
- package/references/minimem/.claude/settings.json +0 -7
- package/references/minimem/.sudocode/issues.jsonl +0 -18
- package/references/minimem/.sudocode/specs.jsonl +0 -1
- package/references/minimem/CLAUDE.md +0 -310
- package/references/minimem/README.md +0 -556
- package/references/minimem/claude-plugin/.claude-plugin/plugin.json +0 -10
- package/references/minimem/claude-plugin/.mcp.json +0 -7
- package/references/minimem/claude-plugin/README.md +0 -158
- package/references/minimem/claude-plugin/commands/recall.md +0 -47
- package/references/minimem/claude-plugin/commands/remember.md +0 -41
- package/references/minimem/claude-plugin/hooks/__tests__/hooks.test.ts +0 -272
- package/references/minimem/claude-plugin/hooks/hooks.json +0 -27
- package/references/minimem/claude-plugin/hooks/session-end.sh +0 -86
- package/references/minimem/claude-plugin/hooks/session-start.sh +0 -85
- package/references/minimem/claude-plugin/skills/memory/SKILL.md +0 -108
- package/references/minimem/package-lock.json +0 -5373
- package/references/minimem/package.json +0 -60
- package/references/minimem/scripts/postbuild.js +0 -35
- package/references/minimem/src/__tests__/edge-cases.test.ts +0 -371
- package/references/minimem/src/__tests__/errors.test.ts +0 -265
- package/references/minimem/src/__tests__/helpers.ts +0 -199
- package/references/minimem/src/__tests__/internal.test.ts +0 -407
- package/references/minimem/src/__tests__/knowledge.test.ts +0 -287
- package/references/minimem/src/__tests__/minimem.integration.test.ts +0 -1127
- package/references/minimem/src/__tests__/session.test.ts +0 -190
- package/references/minimem/src/cli/__tests__/commands.test.ts +0 -759
- package/references/minimem/src/cli/commands/__tests__/conflicts.test.ts +0 -141
- package/references/minimem/src/cli/commands/append.ts +0 -76
- package/references/minimem/src/cli/commands/config.ts +0 -262
- package/references/minimem/src/cli/commands/conflicts.ts +0 -413
- package/references/minimem/src/cli/commands/daemon.ts +0 -169
- package/references/minimem/src/cli/commands/index.ts +0 -12
- package/references/minimem/src/cli/commands/init.ts +0 -88
- package/references/minimem/src/cli/commands/mcp.ts +0 -177
- package/references/minimem/src/cli/commands/push-pull.ts +0 -213
- package/references/minimem/src/cli/commands/search.ts +0 -158
- package/references/minimem/src/cli/commands/status.ts +0 -84
- package/references/minimem/src/cli/commands/sync-init.ts +0 -290
- package/references/minimem/src/cli/commands/sync.ts +0 -70
- package/references/minimem/src/cli/commands/upsert.ts +0 -197
- package/references/minimem/src/cli/config.ts +0 -584
- package/references/minimem/src/cli/index.ts +0 -264
- package/references/minimem/src/cli/shared.ts +0 -161
- package/references/minimem/src/cli/sync/__tests__/central.test.ts +0 -152
- package/references/minimem/src/cli/sync/__tests__/conflicts.test.ts +0 -209
- package/references/minimem/src/cli/sync/__tests__/daemon.test.ts +0 -118
- package/references/minimem/src/cli/sync/__tests__/detection.test.ts +0 -207
- package/references/minimem/src/cli/sync/__tests__/integration.test.ts +0 -476
- package/references/minimem/src/cli/sync/__tests__/registry.test.ts +0 -363
- package/references/minimem/src/cli/sync/__tests__/state.test.ts +0 -255
- package/references/minimem/src/cli/sync/__tests__/validation.test.ts +0 -193
- package/references/minimem/src/cli/sync/__tests__/watcher.test.ts +0 -178
- package/references/minimem/src/cli/sync/central.ts +0 -292
- package/references/minimem/src/cli/sync/conflicts.ts +0 -204
- package/references/minimem/src/cli/sync/daemon.ts +0 -407
- package/references/minimem/src/cli/sync/detection.ts +0 -138
- package/references/minimem/src/cli/sync/index.ts +0 -107
- package/references/minimem/src/cli/sync/operations.ts +0 -373
- package/references/minimem/src/cli/sync/registry.ts +0 -279
- package/references/minimem/src/cli/sync/state.ts +0 -355
- package/references/minimem/src/cli/sync/validation.ts +0 -206
- package/references/minimem/src/cli/sync/watcher.ts +0 -234
- package/references/minimem/src/cli/version.ts +0 -34
- package/references/minimem/src/core/index.ts +0 -9
- package/references/minimem/src/core/indexer.ts +0 -628
- package/references/minimem/src/core/searcher.ts +0 -221
- package/references/minimem/src/db/schema.ts +0 -183
- package/references/minimem/src/db/sqlite-vec.ts +0 -24
- package/references/minimem/src/embeddings/__tests__/embeddings.test.ts +0 -431
- package/references/minimem/src/embeddings/batch-gemini.ts +0 -392
- package/references/minimem/src/embeddings/batch-openai.ts +0 -409
- package/references/minimem/src/embeddings/embeddings.ts +0 -434
- package/references/minimem/src/index.ts +0 -109
- package/references/minimem/src/internal.ts +0 -299
- package/references/minimem/src/minimem.ts +0 -1276
- package/references/minimem/src/search/__tests__/hybrid.test.ts +0 -247
- package/references/minimem/src/search/graph.ts +0 -234
- package/references/minimem/src/search/hybrid.ts +0 -151
- package/references/minimem/src/search/search.ts +0 -256
- package/references/minimem/src/server/__tests__/mcp.test.ts +0 -341
- package/references/minimem/src/server/__tests__/tools.test.ts +0 -364
- package/references/minimem/src/server/mcp.ts +0 -326
- package/references/minimem/src/server/tools.ts +0 -720
- package/references/minimem/src/session.ts +0 -460
- package/references/minimem/tsconfig.json +0 -19
- package/references/minimem/tsup.config.ts +0 -26
- package/references/minimem/vitest.config.ts +0 -24
- package/references/sessionlog/.husky/pre-commit +0 -1
- package/references/sessionlog/.lintstagedrc.json +0 -4
- package/references/sessionlog/.prettierignore +0 -4
- package/references/sessionlog/.prettierrc.json +0 -11
- package/references/sessionlog/LICENSE +0 -21
- package/references/sessionlog/README.md +0 -453
- package/references/sessionlog/eslint.config.js +0 -58
- package/references/sessionlog/package-lock.json +0 -3672
- package/references/sessionlog/package.json +0 -65
- package/references/sessionlog/src/__tests__/agent-hooks.test.ts +0 -570
- package/references/sessionlog/src/__tests__/agent-registry.test.ts +0 -127
- package/references/sessionlog/src/__tests__/claude-code-hooks.test.ts +0 -225
- package/references/sessionlog/src/__tests__/claude-generator.test.ts +0 -46
- package/references/sessionlog/src/__tests__/commit-msg.test.ts +0 -86
- package/references/sessionlog/src/__tests__/cursor-agent.test.ts +0 -224
- package/references/sessionlog/src/__tests__/e2e-live.test.ts +0 -890
- package/references/sessionlog/src/__tests__/event-log.test.ts +0 -183
- package/references/sessionlog/src/__tests__/flush-sentinel.test.ts +0 -105
- package/references/sessionlog/src/__tests__/gemini-agent.test.ts +0 -375
- package/references/sessionlog/src/__tests__/git-hooks.test.ts +0 -78
- package/references/sessionlog/src/__tests__/hook-managers.test.ts +0 -121
- package/references/sessionlog/src/__tests__/lifecycle-tasks.test.ts +0 -759
- package/references/sessionlog/src/__tests__/opencode-agent.test.ts +0 -338
- package/references/sessionlog/src/__tests__/redaction.test.ts +0 -136
- package/references/sessionlog/src/__tests__/session-repo.test.ts +0 -353
- package/references/sessionlog/src/__tests__/session-store.test.ts +0 -166
- package/references/sessionlog/src/__tests__/setup-ccweb.test.ts +0 -466
- package/references/sessionlog/src/__tests__/skill-live.test.ts +0 -461
- package/references/sessionlog/src/__tests__/summarize.test.ts +0 -348
- package/references/sessionlog/src/__tests__/task-plan-e2e.test.ts +0 -610
- package/references/sessionlog/src/__tests__/task-plan-live.test.ts +0 -632
- package/references/sessionlog/src/__tests__/transcript-timestamp.test.ts +0 -121
- package/references/sessionlog/src/__tests__/types.test.ts +0 -166
- package/references/sessionlog/src/__tests__/utils.test.ts +0 -333
- package/references/sessionlog/src/__tests__/validation.test.ts +0 -103
- package/references/sessionlog/src/__tests__/worktree.test.ts +0 -57
- package/references/sessionlog/src/agent/agents/claude-code.ts +0 -1089
- package/references/sessionlog/src/agent/agents/cursor.ts +0 -361
- package/references/sessionlog/src/agent/agents/gemini-cli.ts +0 -632
- package/references/sessionlog/src/agent/agents/opencode.ts +0 -540
- package/references/sessionlog/src/agent/registry.ts +0 -143
- package/references/sessionlog/src/agent/session-types.ts +0 -113
- package/references/sessionlog/src/agent/types.ts +0 -220
- package/references/sessionlog/src/cli.ts +0 -597
- package/references/sessionlog/src/commands/clean.ts +0 -133
- package/references/sessionlog/src/commands/disable.ts +0 -84
- package/references/sessionlog/src/commands/doctor.ts +0 -145
- package/references/sessionlog/src/commands/enable.ts +0 -202
- package/references/sessionlog/src/commands/explain.ts +0 -261
- package/references/sessionlog/src/commands/reset.ts +0 -105
- package/references/sessionlog/src/commands/resume.ts +0 -180
- package/references/sessionlog/src/commands/rewind.ts +0 -195
- package/references/sessionlog/src/commands/setup-ccweb.ts +0 -275
- package/references/sessionlog/src/commands/status.ts +0 -172
- package/references/sessionlog/src/config.ts +0 -165
- package/references/sessionlog/src/events/event-log.ts +0 -126
- package/references/sessionlog/src/git-operations.ts +0 -558
- package/references/sessionlog/src/hooks/git-hooks.ts +0 -165
- package/references/sessionlog/src/hooks/lifecycle.ts +0 -391
- package/references/sessionlog/src/index.ts +0 -650
- package/references/sessionlog/src/security/redaction.ts +0 -283
- package/references/sessionlog/src/session/state-machine.ts +0 -452
- package/references/sessionlog/src/store/checkpoint-store.ts +0 -509
- package/references/sessionlog/src/store/native-store.ts +0 -173
- package/references/sessionlog/src/store/provider-types.ts +0 -99
- package/references/sessionlog/src/store/session-store.ts +0 -266
- package/references/sessionlog/src/strategy/attribution.ts +0 -296
- package/references/sessionlog/src/strategy/common.ts +0 -207
- package/references/sessionlog/src/strategy/content-overlap.ts +0 -228
- package/references/sessionlog/src/strategy/manual-commit.ts +0 -988
- package/references/sessionlog/src/strategy/types.ts +0 -279
- package/references/sessionlog/src/summarize/claude-generator.ts +0 -115
- package/references/sessionlog/src/summarize/summarize.ts +0 -432
- package/references/sessionlog/src/types.ts +0 -508
- package/references/sessionlog/src/utils/chunk-files.ts +0 -49
- package/references/sessionlog/src/utils/commit-message.ts +0 -65
- package/references/sessionlog/src/utils/detect-agent.ts +0 -36
- package/references/sessionlog/src/utils/hook-managers.ts +0 -125
- package/references/sessionlog/src/utils/ide-tags.ts +0 -32
- package/references/sessionlog/src/utils/paths.ts +0 -79
- package/references/sessionlog/src/utils/preview-rewind.ts +0 -80
- package/references/sessionlog/src/utils/rewind-conflict.ts +0 -121
- package/references/sessionlog/src/utils/shadow-branch.ts +0 -109
- package/references/sessionlog/src/utils/string-utils.ts +0 -46
- package/references/sessionlog/src/utils/todo-extract.ts +0 -188
- package/references/sessionlog/src/utils/trailers.ts +0 -187
- package/references/sessionlog/src/utils/transcript-parse.ts +0 -177
- package/references/sessionlog/src/utils/transcript-timestamp.ts +0 -59
- package/references/sessionlog/src/utils/tree-ops.ts +0 -219
- package/references/sessionlog/src/utils/tty.ts +0 -72
- package/references/sessionlog/src/utils/validation.ts +0 -65
- package/references/sessionlog/src/utils/worktree.ts +0 -58
- package/references/sessionlog/src/wire-types.ts +0 -59
- package/references/sessionlog/templates/setup-env.sh +0 -153
- package/references/sessionlog/tsconfig.json +0 -18
- package/references/sessionlog/vitest.config.ts +0 -12
- package/references/skill-tree/.claude/settings.json +0 -6
- package/references/skill-tree/.sudocode/issues.jsonl +0 -19
- package/references/skill-tree/.sudocode/specs.jsonl +0 -3
- package/references/skill-tree/CLAUDE.md +0 -126
- package/references/skill-tree/README.md +0 -372
- package/references/skill-tree/docs/GAPS_v1.md +0 -221
- package/references/skill-tree/docs/INTEGRATION_PLAN.md +0 -467
- package/references/skill-tree/docs/TODOS.md +0 -91
- package/references/skill-tree/docs/anthropic_skill_guide.md +0 -1364
- package/references/skill-tree/docs/design/federated-skill-trees.md +0 -524
- package/references/skill-tree/docs/design/multi-agent-sync.md +0 -759
- package/references/skill-tree/docs/scraper/BRAINSTORM.md +0 -583
- package/references/skill-tree/docs/scraper/POC_PLAN.md +0 -420
- package/references/skill-tree/docs/scraper/README.md +0 -170
- package/references/skill-tree/examples/basic-usage.ts +0 -164
- package/references/skill-tree/package-lock.json +0 -1852
- package/references/skill-tree/package.json +0 -66
- package/references/skill-tree/scraper/README.md +0 -123
- package/references/skill-tree/scraper/docs/DESIGN.md +0 -683
- package/references/skill-tree/scraper/docs/PLAN.md +0 -336
- package/references/skill-tree/scraper/drizzle.config.ts +0 -10
- package/references/skill-tree/scraper/package-lock.json +0 -6329
- package/references/skill-tree/scraper/package.json +0 -68
- package/references/skill-tree/scraper/test/fixtures/invalid-skill/missing-description.md +0 -7
- package/references/skill-tree/scraper/test/fixtures/invalid-skill/missing-name.md +0 -7
- package/references/skill-tree/scraper/test/fixtures/minimal-skill/SKILL.md +0 -27
- package/references/skill-tree/scraper/test/fixtures/skill-json/SKILL.json +0 -21
- package/references/skill-tree/scraper/test/fixtures/skill-with-meta/SKILL.md +0 -54
- package/references/skill-tree/scraper/test/fixtures/skill-with-meta/_meta.json +0 -24
- package/references/skill-tree/scraper/test/fixtures/valid-skill/SKILL.md +0 -93
- package/references/skill-tree/scraper/test/fixtures/valid-skill/_meta.json +0 -22
- package/references/skill-tree/scraper/tsup.config.ts +0 -14
- package/references/skill-tree/scraper/vitest.config.ts +0 -17
- package/references/skill-tree/scripts/convert-to-vitest.ts +0 -166
- package/references/skill-tree/skills/skill-writer/SKILL.md +0 -339
- package/references/skill-tree/skills/skill-writer/references/examples.md +0 -326
- package/references/skill-tree/skills/skill-writer/references/patterns.md +0 -210
- package/references/skill-tree/skills/skill-writer/references/quality-checklist.md +0 -123
- package/references/skill-tree/test/run-all.ts +0 -106
- package/references/skill-tree/test/utils.ts +0 -128
- package/references/skill-tree/vitest.config.ts +0 -16
|
@@ -1,193 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Tests for registry validation
|
|
3
|
-
*/
|
|
4
|
-
|
|
5
|
-
import { describe, it, beforeEach, afterEach, expect } from "vitest";
|
|
6
|
-
import fs from "node:fs/promises";
|
|
7
|
-
import path from "node:path";
|
|
8
|
-
import os from "node:os";
|
|
9
|
-
|
|
10
|
-
import { validateRegistry, formatValidationResult } from "../validation.js";
|
|
11
|
-
import { initCentralRepo } from "../central.js";
|
|
12
|
-
import { writeRegistry, createEmptyRegistry, addMapping, type Registry } from "../registry.js";
|
|
13
|
-
|
|
14
|
-
describe("Registry Validation", () => {
|
|
15
|
-
let tempDir: string;
|
|
16
|
-
let centralRepo: string;
|
|
17
|
-
let localDir: string;
|
|
18
|
-
let xdgDir: string;
|
|
19
|
-
|
|
20
|
-
beforeEach(async () => {
|
|
21
|
-
tempDir = await fs.mkdtemp(path.join(os.tmpdir(), "minimem-validation-test-"));
|
|
22
|
-
centralRepo = path.join(tempDir, "central");
|
|
23
|
-
localDir = path.join(tempDir, "local");
|
|
24
|
-
xdgDir = path.join(tempDir, "xdg-config");
|
|
25
|
-
|
|
26
|
-
await fs.mkdir(localDir, { recursive: true });
|
|
27
|
-
await fs.mkdir(path.join(localDir, ".minimem"), { recursive: true });
|
|
28
|
-
await fs.mkdir(xdgDir, { recursive: true });
|
|
29
|
-
|
|
30
|
-
// Initialize central repo
|
|
31
|
-
await initCentralRepo(centralRepo);
|
|
32
|
-
|
|
33
|
-
// Set up XDG config
|
|
34
|
-
process.env.XDG_CONFIG_HOME = xdgDir;
|
|
35
|
-
await fs.mkdir(path.join(xdgDir, "minimem"), { recursive: true });
|
|
36
|
-
await fs.writeFile(
|
|
37
|
-
path.join(xdgDir, "minimem", "config.json"),
|
|
38
|
-
JSON.stringify({ centralRepo, machineId: "test-machine-1234" }, null, 2)
|
|
39
|
-
);
|
|
40
|
-
});
|
|
41
|
-
|
|
42
|
-
afterEach(async () => {
|
|
43
|
-
delete process.env.XDG_CONFIG_HOME;
|
|
44
|
-
try {
|
|
45
|
-
await fs.rm(tempDir, { recursive: true, force: true });
|
|
46
|
-
} catch {
|
|
47
|
-
// Ignore cleanup errors
|
|
48
|
-
}
|
|
49
|
-
});
|
|
50
|
-
|
|
51
|
-
describe("validateRegistry", () => {
|
|
52
|
-
it("should return valid for empty registry", async () => {
|
|
53
|
-
const result = await validateRegistry();
|
|
54
|
-
|
|
55
|
-
expect(result.valid).toBe(true);
|
|
56
|
-
expect(result.stats.totalMappings).toBe(0);
|
|
57
|
-
expect(result.issues.length).toBe(0);
|
|
58
|
-
});
|
|
59
|
-
|
|
60
|
-
it("should return valid for registry with active mappings", async () => {
|
|
61
|
-
const registry = createEmptyRegistry();
|
|
62
|
-
const updated = addMapping(registry, {
|
|
63
|
-
path: "test-project",
|
|
64
|
-
localPath: localDir,
|
|
65
|
-
machineId: "test-machine-1234",
|
|
66
|
-
lastSync: new Date().toISOString(),
|
|
67
|
-
});
|
|
68
|
-
await writeRegistry(centralRepo, updated);
|
|
69
|
-
|
|
70
|
-
const result = await validateRegistry();
|
|
71
|
-
|
|
72
|
-
expect(result.valid).toBe(true);
|
|
73
|
-
expect(result.stats.totalMappings).toBe(1);
|
|
74
|
-
expect(result.stats.activeMappings).toBe(1);
|
|
75
|
-
});
|
|
76
|
-
|
|
77
|
-
it("should detect path collisions", async () => {
|
|
78
|
-
const registry = createEmptyRegistry();
|
|
79
|
-
// Same path, different machines
|
|
80
|
-
registry.mappings.push({
|
|
81
|
-
path: "shared-project",
|
|
82
|
-
localPath: "/path/on/machine1",
|
|
83
|
-
machineId: "machine-1",
|
|
84
|
-
lastSync: new Date().toISOString(),
|
|
85
|
-
});
|
|
86
|
-
registry.mappings.push({
|
|
87
|
-
path: "shared-project",
|
|
88
|
-
localPath: "/path/on/machine2",
|
|
89
|
-
machineId: "machine-2",
|
|
90
|
-
lastSync: new Date().toISOString(),
|
|
91
|
-
});
|
|
92
|
-
await writeRegistry(centralRepo, registry);
|
|
93
|
-
|
|
94
|
-
const result = await validateRegistry();
|
|
95
|
-
|
|
96
|
-
expect(result.valid).toBe(false);
|
|
97
|
-
expect(result.stats.collisions).toBe(1);
|
|
98
|
-
expect(result.issues.some((i) => i.type === "collision")).toBeTruthy();
|
|
99
|
-
});
|
|
100
|
-
|
|
101
|
-
it("should detect stale mappings", async () => {
|
|
102
|
-
const registry = createEmptyRegistry();
|
|
103
|
-
const oldDate = new Date();
|
|
104
|
-
oldDate.setDate(oldDate.getDate() - 60); // 60 days ago
|
|
105
|
-
|
|
106
|
-
registry.mappings.push({
|
|
107
|
-
path: "old-project",
|
|
108
|
-
localPath: "/path/to/old",
|
|
109
|
-
machineId: "old-machine",
|
|
110
|
-
lastSync: oldDate.toISOString(),
|
|
111
|
-
});
|
|
112
|
-
await writeRegistry(centralRepo, registry);
|
|
113
|
-
|
|
114
|
-
const result = await validateRegistry();
|
|
115
|
-
|
|
116
|
-
expect(result.stats.staleMappings).toBe(1);
|
|
117
|
-
expect(result.issues.some((i) => i.type === "stale")).toBeTruthy();
|
|
118
|
-
});
|
|
119
|
-
|
|
120
|
-
it("should detect missing local directories", async () => {
|
|
121
|
-
// Read the machineId that was set in XDG config
|
|
122
|
-
const { loadXdgConfig } = await import("../../config.js");
|
|
123
|
-
const xdgConfig = await loadXdgConfig();
|
|
124
|
-
const currentMachineId = xdgConfig.machineId;
|
|
125
|
-
|
|
126
|
-
const registry = createEmptyRegistry();
|
|
127
|
-
registry.mappings.push({
|
|
128
|
-
path: "missing-project",
|
|
129
|
-
localPath: path.join(tempDir, "nonexistent"),
|
|
130
|
-
machineId: currentMachineId!, // Current machine
|
|
131
|
-
lastSync: new Date().toISOString(),
|
|
132
|
-
});
|
|
133
|
-
await writeRegistry(centralRepo, registry);
|
|
134
|
-
|
|
135
|
-
const result = await validateRegistry();
|
|
136
|
-
|
|
137
|
-
expect(result.stats.missingDirs).toBe(1);
|
|
138
|
-
expect(result.issues.some((i) => i.type === "missing")).toBeTruthy();
|
|
139
|
-
});
|
|
140
|
-
|
|
141
|
-
it("should not check local dirs for other machines", async () => {
|
|
142
|
-
const registry = createEmptyRegistry();
|
|
143
|
-
registry.mappings.push({
|
|
144
|
-
path: "other-project",
|
|
145
|
-
localPath: "/nonexistent/path",
|
|
146
|
-
machineId: "other-machine", // Different machine
|
|
147
|
-
lastSync: new Date().toISOString(),
|
|
148
|
-
});
|
|
149
|
-
await writeRegistry(centralRepo, registry);
|
|
150
|
-
|
|
151
|
-
const result = await validateRegistry();
|
|
152
|
-
|
|
153
|
-
// Should not report missing dir since it's on a different machine
|
|
154
|
-
expect(result.stats.missingDirs).toBe(0);
|
|
155
|
-
});
|
|
156
|
-
});
|
|
157
|
-
|
|
158
|
-
describe("formatValidationResult", () => {
|
|
159
|
-
it("should format valid result", async () => {
|
|
160
|
-
const result = await validateRegistry();
|
|
161
|
-
const formatted = formatValidationResult(result);
|
|
162
|
-
|
|
163
|
-
expect(formatted.includes("Registry Validation Results")).toBeTruthy();
|
|
164
|
-
expect(formatted.includes("No issues found")).toBeTruthy();
|
|
165
|
-
expect(formatted.includes("valid")).toBeTruthy();
|
|
166
|
-
});
|
|
167
|
-
|
|
168
|
-
it("should format result with issues", async () => {
|
|
169
|
-
// Create a collision
|
|
170
|
-
const registry = createEmptyRegistry();
|
|
171
|
-
registry.mappings.push({
|
|
172
|
-
path: "shared",
|
|
173
|
-
localPath: "/path1",
|
|
174
|
-
machineId: "machine-1",
|
|
175
|
-
lastSync: new Date().toISOString(),
|
|
176
|
-
});
|
|
177
|
-
registry.mappings.push({
|
|
178
|
-
path: "shared",
|
|
179
|
-
localPath: "/path2",
|
|
180
|
-
machineId: "machine-2",
|
|
181
|
-
lastSync: new Date().toISOString(),
|
|
182
|
-
});
|
|
183
|
-
await writeRegistry(centralRepo, registry);
|
|
184
|
-
|
|
185
|
-
const result = await validateRegistry();
|
|
186
|
-
const formatted = formatValidationResult(result);
|
|
187
|
-
|
|
188
|
-
expect(formatted.includes("Issues:")).toBeTruthy();
|
|
189
|
-
expect(formatted.includes("ERROR")).toBeTruthy();
|
|
190
|
-
expect(formatted.includes("collision") || formatted.includes("multiple machines")).toBeTruthy();
|
|
191
|
-
});
|
|
192
|
-
});
|
|
193
|
-
});
|
|
@@ -1,178 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Tests for file watcher
|
|
3
|
-
*
|
|
4
|
-
* Note: File event detection tests are skipped due to chokidar timing issues in test environments.
|
|
5
|
-
* The core watcher functionality is tested via creation, cleanup, and listener management.
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
import { describe, it, beforeEach, afterEach, expect } from "vitest";
|
|
9
|
-
import fs from "node:fs/promises";
|
|
10
|
-
import path from "node:path";
|
|
11
|
-
import os from "node:os";
|
|
12
|
-
|
|
13
|
-
import { createFileWatcher, createMultiDirWatcher, type FileChange } from "../watcher.js";
|
|
14
|
-
|
|
15
|
-
describe("File Watcher", () => {
|
|
16
|
-
let tempDir: string;
|
|
17
|
-
let memoryDir: string;
|
|
18
|
-
|
|
19
|
-
beforeEach(async () => {
|
|
20
|
-
tempDir = await fs.mkdtemp(path.join(os.tmpdir(), "minimem-watcher-test-"));
|
|
21
|
-
memoryDir = tempDir;
|
|
22
|
-
await fs.mkdir(path.join(memoryDir, "memory"), { recursive: true });
|
|
23
|
-
await fs.writeFile(path.join(memoryDir, "MEMORY.md"), "# Memory\n");
|
|
24
|
-
});
|
|
25
|
-
|
|
26
|
-
afterEach(async () => {
|
|
27
|
-
try {
|
|
28
|
-
await fs.rm(tempDir, { recursive: true, force: true });
|
|
29
|
-
} catch {
|
|
30
|
-
// Ignore cleanup errors
|
|
31
|
-
}
|
|
32
|
-
});
|
|
33
|
-
|
|
34
|
-
describe("createFileWatcher", () => {
|
|
35
|
-
it("should create watcher and become ready", async () => {
|
|
36
|
-
const watcher = createFileWatcher(memoryDir, { debounceMs: 100 });
|
|
37
|
-
|
|
38
|
-
// Wait for ready
|
|
39
|
-
await new Promise<void>((resolve) => {
|
|
40
|
-
const check = () => {
|
|
41
|
-
if (watcher.ready) {
|
|
42
|
-
resolve();
|
|
43
|
-
} else {
|
|
44
|
-
setTimeout(check, 50);
|
|
45
|
-
}
|
|
46
|
-
};
|
|
47
|
-
check();
|
|
48
|
-
});
|
|
49
|
-
|
|
50
|
-
expect(watcher.ready).toBe(true);
|
|
51
|
-
await watcher.close();
|
|
52
|
-
});
|
|
53
|
-
|
|
54
|
-
it("should clean up on close", async () => {
|
|
55
|
-
const watcher = createFileWatcher(memoryDir, { debounceMs: 100 });
|
|
56
|
-
|
|
57
|
-
await new Promise<void>((resolve) => {
|
|
58
|
-
const check = () => {
|
|
59
|
-
if (watcher.ready) resolve();
|
|
60
|
-
else setTimeout(check, 50);
|
|
61
|
-
};
|
|
62
|
-
check();
|
|
63
|
-
});
|
|
64
|
-
|
|
65
|
-
let changeCount = 0;
|
|
66
|
-
watcher.on("changes", () => {
|
|
67
|
-
changeCount++;
|
|
68
|
-
});
|
|
69
|
-
|
|
70
|
-
await watcher.close();
|
|
71
|
-
|
|
72
|
-
// Changes after close should not trigger events
|
|
73
|
-
await fs.writeFile(path.join(memoryDir, "MEMORY.md"), "# After close\n");
|
|
74
|
-
await new Promise((r) => setTimeout(r, 200));
|
|
75
|
-
|
|
76
|
-
expect(changeCount).toBe(0);
|
|
77
|
-
});
|
|
78
|
-
|
|
79
|
-
it("should allow adding and removing listeners", async () => {
|
|
80
|
-
const watcher = createFileWatcher(memoryDir, { debounceMs: 100 });
|
|
81
|
-
|
|
82
|
-
await new Promise<void>((resolve) => {
|
|
83
|
-
const check = () => {
|
|
84
|
-
if (watcher.ready) resolve();
|
|
85
|
-
else setTimeout(check, 50);
|
|
86
|
-
};
|
|
87
|
-
check();
|
|
88
|
-
});
|
|
89
|
-
|
|
90
|
-
let changeCount = 0;
|
|
91
|
-
const listener = () => {
|
|
92
|
-
changeCount++;
|
|
93
|
-
};
|
|
94
|
-
|
|
95
|
-
watcher.on("changes", listener);
|
|
96
|
-
watcher.off("changes", listener);
|
|
97
|
-
|
|
98
|
-
// No listener should be called
|
|
99
|
-
await watcher.close();
|
|
100
|
-
});
|
|
101
|
-
|
|
102
|
-
it("should accept custom options", async () => {
|
|
103
|
-
const watcher = createFileWatcher(memoryDir, {
|
|
104
|
-
debounceMs: 500,
|
|
105
|
-
include: ["*.md", "docs/**/*.md"],
|
|
106
|
-
exclude: ["temp/**"],
|
|
107
|
-
usePolling: false,
|
|
108
|
-
pollInterval: 2000,
|
|
109
|
-
});
|
|
110
|
-
|
|
111
|
-
await new Promise<void>((resolve) => {
|
|
112
|
-
const check = () => {
|
|
113
|
-
if (watcher.ready) resolve();
|
|
114
|
-
else setTimeout(check, 50);
|
|
115
|
-
};
|
|
116
|
-
check();
|
|
117
|
-
});
|
|
118
|
-
|
|
119
|
-
expect(watcher.ready).toBe(true);
|
|
120
|
-
await watcher.close();
|
|
121
|
-
});
|
|
122
|
-
});
|
|
123
|
-
|
|
124
|
-
describe("createMultiDirWatcher", () => {
|
|
125
|
-
it("should watch multiple directories", async () => {
|
|
126
|
-
const dir2 = await fs.mkdtemp(path.join(os.tmpdir(), "minimem-watcher-test2-"));
|
|
127
|
-
await fs.mkdir(path.join(dir2, "memory"), { recursive: true });
|
|
128
|
-
await fs.writeFile(path.join(dir2, "MEMORY.md"), "# Memory 2\n");
|
|
129
|
-
|
|
130
|
-
try {
|
|
131
|
-
const watcher = createMultiDirWatcher([memoryDir, dir2], { debounceMs: 100 });
|
|
132
|
-
|
|
133
|
-
await new Promise<void>((resolve) => {
|
|
134
|
-
const check = () => {
|
|
135
|
-
if (watcher.ready) resolve();
|
|
136
|
-
else setTimeout(check, 50);
|
|
137
|
-
};
|
|
138
|
-
check();
|
|
139
|
-
});
|
|
140
|
-
|
|
141
|
-
expect(watcher.ready).toBe(true);
|
|
142
|
-
await watcher.close();
|
|
143
|
-
} finally {
|
|
144
|
-
await fs.rm(dir2, { recursive: true, force: true });
|
|
145
|
-
}
|
|
146
|
-
});
|
|
147
|
-
|
|
148
|
-
it("should close all watchers", async () => {
|
|
149
|
-
const dir2 = await fs.mkdtemp(path.join(os.tmpdir(), "minimem-watcher-test3-"));
|
|
150
|
-
await fs.mkdir(path.join(dir2, "memory"), { recursive: true });
|
|
151
|
-
|
|
152
|
-
try {
|
|
153
|
-
const watcher = createMultiDirWatcher([memoryDir, dir2], { debounceMs: 100 });
|
|
154
|
-
|
|
155
|
-
await new Promise<void>((resolve) => {
|
|
156
|
-
const check = () => {
|
|
157
|
-
if (watcher.ready) resolve();
|
|
158
|
-
else setTimeout(check, 50);
|
|
159
|
-
};
|
|
160
|
-
check();
|
|
161
|
-
});
|
|
162
|
-
|
|
163
|
-
let changeCount = 0;
|
|
164
|
-
watcher.on("changes", () => changeCount++);
|
|
165
|
-
|
|
166
|
-
await watcher.close();
|
|
167
|
-
|
|
168
|
-
// Changes after close should not trigger events
|
|
169
|
-
await fs.writeFile(path.join(memoryDir, "MEMORY.md"), "# Changed\n");
|
|
170
|
-
await new Promise((r) => setTimeout(r, 200));
|
|
171
|
-
|
|
172
|
-
expect(changeCount).toBe(0);
|
|
173
|
-
} finally {
|
|
174
|
-
await fs.rm(dir2, { recursive: true, force: true });
|
|
175
|
-
}
|
|
176
|
-
});
|
|
177
|
-
});
|
|
178
|
-
});
|
|
@@ -1,292 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Central repository initialization and management
|
|
3
|
-
*/
|
|
4
|
-
|
|
5
|
-
import fs from "node:fs/promises";
|
|
6
|
-
import path from "node:path";
|
|
7
|
-
import { execSync } from "node:child_process";
|
|
8
|
-
|
|
9
|
-
import { expandPath, saveXdgConfig, loadXdgConfig } from "../config.js";
|
|
10
|
-
import {
|
|
11
|
-
createEmptyRegistry,
|
|
12
|
-
writeRegistry,
|
|
13
|
-
readRegistry,
|
|
14
|
-
getRegistryPath,
|
|
15
|
-
} from "./registry.js";
|
|
16
|
-
import { isInsideGitRepo } from "./detection.js";
|
|
17
|
-
|
|
18
|
-
const GITIGNORE_CONTENT = `# Minimem sync - ignore database files and temp directories
|
|
19
|
-
*.db
|
|
20
|
-
*.db-journal
|
|
21
|
-
*.db-wal
|
|
22
|
-
*.db-shm
|
|
23
|
-
staging/
|
|
24
|
-
conflicts/
|
|
25
|
-
shadows/
|
|
26
|
-
.DS_Store
|
|
27
|
-
`;
|
|
28
|
-
|
|
29
|
-
const README_CONTENT = `# Minimem Central Repository
|
|
30
|
-
|
|
31
|
-
This repository contains synchronized memory files from minimem.
|
|
32
|
-
|
|
33
|
-
## Structure
|
|
34
|
-
|
|
35
|
-
Each directory represents a mapped memory location:
|
|
36
|
-
|
|
37
|
-
\`\`\`
|
|
38
|
-
memories-repo/
|
|
39
|
-
├── global/ # ~/.minimem/
|
|
40
|
-
├── work/ # ~/work/memories/
|
|
41
|
-
└── projects/
|
|
42
|
-
└── myproject/ # ~/projects/myproject/
|
|
43
|
-
\`\`\`
|
|
44
|
-
|
|
45
|
-
## Registry
|
|
46
|
-
|
|
47
|
-
The \`.minimem-registry.json\` file tracks which local directories are mapped to which paths.
|
|
48
|
-
This prevents path collisions when multiple machines sync to the same repository.
|
|
49
|
-
|
|
50
|
-
## Usage
|
|
51
|
-
|
|
52
|
-
To sync a local memory directory to this repo:
|
|
53
|
-
|
|
54
|
-
\`\`\`bash
|
|
55
|
-
minimem config --xdg-global --set centralRepo=<path-to-this-repo>
|
|
56
|
-
minimem sync init --path <directory-name>/
|
|
57
|
-
\`\`\`
|
|
58
|
-
|
|
59
|
-
## Files
|
|
60
|
-
|
|
61
|
-
- \`.minimem-registry.json\` - Tracks sync mappings
|
|
62
|
-
- \`.gitignore\` - Ignores database and temp files
|
|
63
|
-
- \`*/MEMORY.md\` - Main memory files
|
|
64
|
-
- \`*/memory/*.md\` - Additional memory files
|
|
65
|
-
`;
|
|
66
|
-
|
|
67
|
-
export type InitCentralResult = {
|
|
68
|
-
success: boolean;
|
|
69
|
-
path: string;
|
|
70
|
-
created: boolean;
|
|
71
|
-
message: string;
|
|
72
|
-
};
|
|
73
|
-
|
|
74
|
-
/**
|
|
75
|
-
* Check if a path is writable
|
|
76
|
-
*/
|
|
77
|
-
async function isWritable(dirPath: string): Promise<boolean> {
|
|
78
|
-
try {
|
|
79
|
-
const testFile = path.join(dirPath, `.minimem-write-test-${Date.now()}`);
|
|
80
|
-
await fs.writeFile(testFile, "test");
|
|
81
|
-
await fs.unlink(testFile);
|
|
82
|
-
return true;
|
|
83
|
-
} catch {
|
|
84
|
-
return false;
|
|
85
|
-
}
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
/**
|
|
89
|
-
* Check if git is available
|
|
90
|
-
*/
|
|
91
|
-
function isGitAvailable(): boolean {
|
|
92
|
-
try {
|
|
93
|
-
execSync("git --version", { stdio: "pipe" });
|
|
94
|
-
return true;
|
|
95
|
-
} catch {
|
|
96
|
-
return false;
|
|
97
|
-
}
|
|
98
|
-
}
|
|
99
|
-
|
|
100
|
-
/**
|
|
101
|
-
* Initialize a git repository
|
|
102
|
-
*/
|
|
103
|
-
async function initGitRepo(dirPath: string): Promise<void> {
|
|
104
|
-
execSync("git init", { cwd: dirPath, stdio: "pipe" });
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
/**
|
|
108
|
-
* Initialize a new central repository or configure an existing one
|
|
109
|
-
*/
|
|
110
|
-
export async function initCentralRepo(
|
|
111
|
-
repoPath: string
|
|
112
|
-
): Promise<InitCentralResult> {
|
|
113
|
-
const expandedPath = expandPath(repoPath);
|
|
114
|
-
const resolvedPath = path.resolve(expandedPath);
|
|
115
|
-
|
|
116
|
-
// Check if git is available
|
|
117
|
-
if (!isGitAvailable()) {
|
|
118
|
-
return {
|
|
119
|
-
success: false,
|
|
120
|
-
path: resolvedPath,
|
|
121
|
-
created: false,
|
|
122
|
-
message: "Git is not installed or not available in PATH",
|
|
123
|
-
};
|
|
124
|
-
}
|
|
125
|
-
|
|
126
|
-
// Check if directory exists
|
|
127
|
-
let dirExists = false;
|
|
128
|
-
try {
|
|
129
|
-
const stat = await fs.stat(resolvedPath);
|
|
130
|
-
dirExists = stat.isDirectory();
|
|
131
|
-
} catch {
|
|
132
|
-
dirExists = false;
|
|
133
|
-
}
|
|
134
|
-
|
|
135
|
-
let created = false;
|
|
136
|
-
|
|
137
|
-
if (!dirExists) {
|
|
138
|
-
// Create new directory
|
|
139
|
-
try {
|
|
140
|
-
await fs.mkdir(resolvedPath, { recursive: true });
|
|
141
|
-
created = true;
|
|
142
|
-
} catch (error) {
|
|
143
|
-
return {
|
|
144
|
-
success: false,
|
|
145
|
-
path: resolvedPath,
|
|
146
|
-
created: false,
|
|
147
|
-
message: `Failed to create directory: ${error}`,
|
|
148
|
-
};
|
|
149
|
-
}
|
|
150
|
-
}
|
|
151
|
-
|
|
152
|
-
// Check if writable
|
|
153
|
-
if (!(await isWritable(resolvedPath))) {
|
|
154
|
-
return {
|
|
155
|
-
success: false,
|
|
156
|
-
path: resolvedPath,
|
|
157
|
-
created,
|
|
158
|
-
message: "Directory is not writable",
|
|
159
|
-
};
|
|
160
|
-
}
|
|
161
|
-
|
|
162
|
-
// Initialize git repo if not already
|
|
163
|
-
const isGitRepo = await isInsideGitRepo(resolvedPath);
|
|
164
|
-
if (!isGitRepo) {
|
|
165
|
-
try {
|
|
166
|
-
await initGitRepo(resolvedPath);
|
|
167
|
-
} catch (error) {
|
|
168
|
-
return {
|
|
169
|
-
success: false,
|
|
170
|
-
path: resolvedPath,
|
|
171
|
-
created,
|
|
172
|
-
message: `Failed to initialize git repository: ${error}`,
|
|
173
|
-
};
|
|
174
|
-
}
|
|
175
|
-
}
|
|
176
|
-
|
|
177
|
-
// Create .gitignore if not exists
|
|
178
|
-
const gitignorePath = path.join(resolvedPath, ".gitignore");
|
|
179
|
-
try {
|
|
180
|
-
await fs.access(gitignorePath);
|
|
181
|
-
} catch {
|
|
182
|
-
await fs.writeFile(gitignorePath, GITIGNORE_CONTENT, "utf-8");
|
|
183
|
-
}
|
|
184
|
-
|
|
185
|
-
// Create registry if not exists
|
|
186
|
-
const registryPath = getRegistryPath(resolvedPath);
|
|
187
|
-
try {
|
|
188
|
-
await fs.access(registryPath);
|
|
189
|
-
} catch {
|
|
190
|
-
await writeRegistry(resolvedPath, createEmptyRegistry());
|
|
191
|
-
}
|
|
192
|
-
|
|
193
|
-
// Create README if not exists
|
|
194
|
-
const readmePath = path.join(resolvedPath, "README.md");
|
|
195
|
-
try {
|
|
196
|
-
await fs.access(readmePath);
|
|
197
|
-
} catch {
|
|
198
|
-
await fs.writeFile(readmePath, README_CONTENT, "utf-8");
|
|
199
|
-
}
|
|
200
|
-
|
|
201
|
-
// Update global config with central repo path
|
|
202
|
-
const globalConfig = await loadXdgConfig();
|
|
203
|
-
globalConfig.centralRepo = repoPath; // Store original (possibly with ~)
|
|
204
|
-
await saveXdgConfig(globalConfig);
|
|
205
|
-
|
|
206
|
-
return {
|
|
207
|
-
success: true,
|
|
208
|
-
path: resolvedPath,
|
|
209
|
-
created,
|
|
210
|
-
message: created
|
|
211
|
-
? "Created new central repository"
|
|
212
|
-
: "Configured existing directory as central repository",
|
|
213
|
-
};
|
|
214
|
-
}
|
|
215
|
-
|
|
216
|
-
/**
|
|
217
|
-
* Validate an existing central repository
|
|
218
|
-
*/
|
|
219
|
-
export async function validateCentralRepo(repoPath: string): Promise<{
|
|
220
|
-
valid: boolean;
|
|
221
|
-
warnings: string[];
|
|
222
|
-
errors: string[];
|
|
223
|
-
}> {
|
|
224
|
-
const expandedPath = expandPath(repoPath);
|
|
225
|
-
const resolvedPath = path.resolve(expandedPath);
|
|
226
|
-
const warnings: string[] = [];
|
|
227
|
-
const errors: string[] = [];
|
|
228
|
-
|
|
229
|
-
// Check directory exists
|
|
230
|
-
try {
|
|
231
|
-
const stat = await fs.stat(resolvedPath);
|
|
232
|
-
if (!stat.isDirectory()) {
|
|
233
|
-
errors.push("Path is not a directory");
|
|
234
|
-
return { valid: false, warnings, errors };
|
|
235
|
-
}
|
|
236
|
-
} catch {
|
|
237
|
-
errors.push("Directory does not exist");
|
|
238
|
-
return { valid: false, warnings, errors };
|
|
239
|
-
}
|
|
240
|
-
|
|
241
|
-
// Check if git repo
|
|
242
|
-
const isGitRepo = await isInsideGitRepo(resolvedPath);
|
|
243
|
-
if (!isGitRepo) {
|
|
244
|
-
warnings.push("Not a git repository - sync features may be limited");
|
|
245
|
-
}
|
|
246
|
-
|
|
247
|
-
// Check for registry
|
|
248
|
-
const registryPath = getRegistryPath(resolvedPath);
|
|
249
|
-
try {
|
|
250
|
-
await fs.access(registryPath);
|
|
251
|
-
// Validate registry content
|
|
252
|
-
const registry = await readRegistry(resolvedPath);
|
|
253
|
-
if (!registry.mappings) {
|
|
254
|
-
warnings.push("Registry file is malformed");
|
|
255
|
-
}
|
|
256
|
-
} catch {
|
|
257
|
-
warnings.push("Registry file is missing - will be created on first sync");
|
|
258
|
-
}
|
|
259
|
-
|
|
260
|
-
// Check for .gitignore
|
|
261
|
-
const gitignorePath = path.join(resolvedPath, ".gitignore");
|
|
262
|
-
try {
|
|
263
|
-
const gitignore = await fs.readFile(gitignorePath, "utf-8");
|
|
264
|
-
if (!gitignore.includes("*.db")) {
|
|
265
|
-
warnings.push(".gitignore does not exclude database files");
|
|
266
|
-
}
|
|
267
|
-
} catch {
|
|
268
|
-
warnings.push(".gitignore is missing");
|
|
269
|
-
}
|
|
270
|
-
|
|
271
|
-
// Check if writable
|
|
272
|
-
if (!(await isWritable(resolvedPath))) {
|
|
273
|
-
errors.push("Directory is not writable");
|
|
274
|
-
}
|
|
275
|
-
|
|
276
|
-
return {
|
|
277
|
-
valid: errors.length === 0,
|
|
278
|
-
warnings,
|
|
279
|
-
errors,
|
|
280
|
-
};
|
|
281
|
-
}
|
|
282
|
-
|
|
283
|
-
/**
|
|
284
|
-
* Get the configured central repo path
|
|
285
|
-
*/
|
|
286
|
-
export async function getCentralRepoPath(): Promise<string | undefined> {
|
|
287
|
-
const globalConfig = await loadXdgConfig();
|
|
288
|
-
if (globalConfig.centralRepo) {
|
|
289
|
-
return path.resolve(expandPath(globalConfig.centralRepo));
|
|
290
|
-
}
|
|
291
|
-
return undefined;
|
|
292
|
-
}
|