@indigoai-us/hq-cloud 5.24.0 → 5.26.0

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.
Files changed (66) hide show
  1. package/dist/bin/sync-runner.d.ts +151 -17
  2. package/dist/bin/sync-runner.d.ts.map +1 -1
  3. package/dist/bin/sync-runner.js +280 -18
  4. package/dist/bin/sync-runner.js.map +1 -1
  5. package/dist/bin/sync-runner.test.js +429 -15
  6. package/dist/bin/sync-runner.test.js.map +1 -1
  7. package/dist/cli/share.d.ts +9 -0
  8. package/dist/cli/share.d.ts.map +1 -1
  9. package/dist/cli/share.js +54 -1
  10. package/dist/cli/share.js.map +1 -1
  11. package/dist/cli/share.test.js +6 -3
  12. package/dist/cli/share.test.js.map +1 -1
  13. package/dist/cli/sync.d.ts +21 -0
  14. package/dist/cli/sync.d.ts.map +1 -1
  15. package/dist/cli/sync.js.map +1 -1
  16. package/dist/index.d.ts +4 -0
  17. package/dist/index.d.ts.map +1 -1
  18. package/dist/index.js +6 -0
  19. package/dist/index.js.map +1 -1
  20. package/dist/personal-vault-exclusions.d.ts +128 -0
  21. package/dist/personal-vault-exclusions.d.ts.map +1 -0
  22. package/dist/personal-vault-exclusions.js +231 -0
  23. package/dist/personal-vault-exclusions.js.map +1 -0
  24. package/dist/personal-vault-exclusions.test.d.ts +22 -0
  25. package/dist/personal-vault-exclusions.test.d.ts.map +1 -0
  26. package/dist/personal-vault-exclusions.test.js +198 -0
  27. package/dist/personal-vault-exclusions.test.js.map +1 -0
  28. package/dist/sync/index.d.ts +11 -0
  29. package/dist/sync/index.d.ts.map +1 -0
  30. package/dist/sync/index.js +9 -0
  31. package/dist/sync/index.js.map +1 -0
  32. package/dist/sync/push-event.d.ts +110 -0
  33. package/dist/sync/push-event.d.ts.map +1 -0
  34. package/dist/sync/push-event.js +153 -0
  35. package/dist/sync/push-event.js.map +1 -0
  36. package/dist/sync/push-event.test.d.ts +15 -0
  37. package/dist/sync/push-event.test.d.ts.map +1 -0
  38. package/dist/sync/push-event.test.js +188 -0
  39. package/dist/sync/push-event.test.js.map +1 -0
  40. package/dist/sync/push-transport.d.ts +67 -0
  41. package/dist/sync/push-transport.d.ts.map +1 -0
  42. package/dist/sync/push-transport.js +66 -0
  43. package/dist/sync/push-transport.js.map +1 -0
  44. package/dist/watcher.d.ts +160 -0
  45. package/dist/watcher.d.ts.map +1 -1
  46. package/dist/watcher.js +298 -0
  47. package/dist/watcher.js.map +1 -1
  48. package/dist/watcher.test.d.ts +2 -0
  49. package/dist/watcher.test.d.ts.map +1 -0
  50. package/dist/watcher.test.js +334 -0
  51. package/dist/watcher.test.js.map +1 -0
  52. package/package.json +3 -2
  53. package/src/bin/sync-runner.test.ts +557 -15
  54. package/src/bin/sync-runner.ts +404 -27
  55. package/src/cli/share.test.ts +8 -3
  56. package/src/cli/share.ts +66 -1
  57. package/src/cli/sync.ts +22 -0
  58. package/src/index.ts +27 -0
  59. package/src/personal-vault-exclusions.test.ts +256 -0
  60. package/src/personal-vault-exclusions.ts +277 -0
  61. package/src/sync/index.ts +19 -0
  62. package/src/sync/push-event.test.ts +224 -0
  63. package/src/sync/push-event.ts +208 -0
  64. package/src/sync/push-transport.ts +84 -0
  65. package/src/watcher.test.ts +388 -0
  66. package/src/watcher.ts +386 -0
@@ -0,0 +1,128 @@
1
+ /**
2
+ * Personal-vault default exclusions — second-tier scope filter that complements
3
+ * `PERSONAL_VAULT_EXCLUDED_TOP_LEVEL` (`.git`, `companies`, `repos`,
4
+ * `workspace`) in `./personal-vault.ts`.
5
+ *
6
+ * Where the top-level constant filters the four big buckets at scope root, this
7
+ * file filters categories of nested files that ride along with an HQ install
8
+ * but have no business round-tripping to a personal vault — regardless of
9
+ * where they sit in the tree:
10
+ *
11
+ * 1. Secrets at root or nested. `.env`, `.env.local`, `.env.<anything>`,
12
+ * `.mcp.json`. Pre-fix these were being uploaded if the user hadn't
13
+ * added `.env` to their `.hqignore` (the default starter `.hqignore`
14
+ * only listed `companies/*\/settings/`).
15
+ *
16
+ * 2. Machine-local state. `.beads/` (SQLite issue-tracker + WAL/SHM),
17
+ * `.obsidian/`, `.vercel/`, `.cache_*`. Per-machine layouts/caches;
18
+ * multi-device sync either corrupts (SQLite WAL) or causes spurious
19
+ * churn (caches regenerate on demand).
20
+ *
21
+ * 3. Update-flow scratch. `output/`, `_legacy-*` directories. Created by
22
+ * `/update-hq` / `/promote-hq-core` workflows as checkout scratch;
23
+ * naming themselves "legacy" / sitting under "output" is a self-
24
+ * declared "do not preserve" signal.
25
+ *
26
+ * 4. Pre-5.24 conflict mirror dir. `.hq-conflicts/` is a directory of
27
+ * mirror copies from older sync runs (sibling to the now-fixed
28
+ * `.conflict-YYYY-MM-DDTHH-MM-SSZ-{hash}.{ext}` ephemeral files
29
+ * handled by `EPHEMERAL_PATH_PATTERN` in share.ts). Same logic
30
+ * applies: these are local-only safety backups.
31
+ *
32
+ * 5. OS / build cruft. `.DS_Store`, `node_modules/`, `dist/`, `.next/`,
33
+ * `build/`. Universally noise inside HQ (personal scope should not
34
+ * contain code projects, but defense-in-depth catches anyone who
35
+ * symlinks a project subtree in).
36
+ *
37
+ * Policy: refuse + warn (5.25 default). The walk filter drops these so they
38
+ * never upload; the delete-plan walker uses the same filter so already-
39
+ * journaled entries that match a new exclusion get orphaned in the journal
40
+ * (no DELETE issued, no churn). A one-shot purge script handles the cleanup
41
+ * of objects that landed on remote before this version.
42
+ *
43
+ * Application scope: personal vault only. Company vaults have separate
44
+ * first-push protection (settings/, data/, workers/, .git/ exclusion in
45
+ * `src-tauri/src/util/ignore.rs`) and may legitimately ship `output/` or
46
+ * `.env*` paths inside their data folders. Wired in `share.ts` only when
47
+ * `options.personalMode === true`.
48
+ *
49
+ * Wire-points (parallel to `EPHEMERAL_PATH_PATTERN`):
50
+ * - `collectFiles` / `walkDir` (push) — wrap `shouldSync` so excluded
51
+ * relative paths are rejected before upload.
52
+ * - `computeDeletePlan` (delete) — same `shouldSync` wrap, so journal
53
+ * entries matching an exclusion are skipped on the delete pass too.
54
+ */
55
+ /**
56
+ * Each entry is matched against the relative path from the personal-vault
57
+ * sync root (which IS hq_root in personalMode), using forward-slash
58
+ * separators. Patterns:
59
+ *
60
+ * - `prefix:foo/` — match if the path starts with `foo/` or equals `foo`
61
+ * (handles both the dir itself and any descendant)
62
+ * - `basename:.env` — match if any path segment equals `.env`
63
+ * - `basename:.env-prefix:.env.` — match if the basename starts with `.env.`
64
+ * - `segment:node_modules` — match if any path segment equals literally
65
+ *
66
+ * The literal-segment form catches nested cases (e.g., `repos/foo/node_modules/`
67
+ * is already excluded by top-level repos/ skip, but `personal/x/node_modules/`
68
+ * would slip through without segment matching).
69
+ */
70
+ export interface PersonalVaultExclusion {
71
+ /** Internal id for telemetry / explainability in events. */
72
+ id: string;
73
+ /** Human-readable category for the warning event. */
74
+ category: "secret" | "machine-local" | "scratch" | "conflict-mirror" | "os-cruft" | "build-output";
75
+ /**
76
+ * Predicate. Pure: relative path (forward-slash separated, no leading slash)
77
+ * + optional isDir hint. Returns true to EXCLUDE.
78
+ */
79
+ test: (relPath: string, isDir?: boolean) => boolean;
80
+ /** Doc-only — shown alongside `id` in the warning event sample. */
81
+ pattern: string;
82
+ }
83
+ /**
84
+ * Cheap segment-equality check. Avoids allocating a regex per call.
85
+ */
86
+ declare function hasSegment(relPath: string, segment: string): boolean;
87
+ declare function basename(relPath: string): string;
88
+ export declare const PERSONAL_VAULT_DEFAULT_EXCLUSIONS: readonly PersonalVaultExclusion[];
89
+ /**
90
+ * First-match result for a path. Returns the matching exclusion (so callers
91
+ * can surface category/id in events) or undefined for "not excluded".
92
+ */
93
+ export declare function matchPersonalVaultExclusion(relPath: string, isDir?: boolean): PersonalVaultExclusion | undefined;
94
+ /**
95
+ * Boolean version — true if any exclusion matches.
96
+ */
97
+ export declare function isPersonalVaultExcluded(relPath: string, isDir?: boolean): boolean;
98
+ /**
99
+ * Wrap an existing path filter (typically from `createIgnoreFilter`) with the
100
+ * personal-vault default exclusions. The wrapper:
101
+ *
102
+ * 1. Calls the underlying filter first; if it already rejects, defer
103
+ * (counter does not fire — we only want to count things the underlying
104
+ * filter would have allowed but the personal-vault defaults reject).
105
+ * 2. Computes a relative path from `syncRoot` (the personal-vault sync
106
+ * root, which is hq_root in personalMode).
107
+ * 3. Probes `matchPersonalVaultExclusion`; if it matches, returns false
108
+ * and tags the match via `onExcluded` so the runner can emit a single
109
+ * `personal-vault-out-of-policy` event with a count + sample at end of
110
+ * run.
111
+ *
112
+ * The wrapper keeps the `(absolutePath, isDir) => boolean` shape the existing
113
+ * collectFiles / walkDir / computeDeletePlan code already uses, so wiring is
114
+ * a single line change at the share() filter construction site.
115
+ */
116
+ export declare function wrapFilterWithPersonalVaultDefaults(underlying: (absPath: string, isDir?: boolean) => boolean, syncRoot: string, onExcluded: (relPath: string, match: PersonalVaultExclusion) => void): (absPath: string, isDir?: boolean) => boolean;
117
+ /**
118
+ * Test-only export. Mirrors the `_testing` namespace pattern used by
119
+ * `share.ts` for `EPHEMERAL_PATH_PATTERN` — direct access to the list +
120
+ * internal helpers for regression-critical pinning without round-tripping
121
+ * through share().
122
+ */
123
+ export declare const _testing: {
124
+ hasSegment: typeof hasSegment;
125
+ basename: typeof basename;
126
+ };
127
+ export {};
128
+ //# sourceMappingURL=personal-vault-exclusions.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"personal-vault-exclusions.d.ts","sourceRoot":"","sources":["../src/personal-vault-exclusions.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqDG;AAIH;;;;;;;;;;;;;;GAcG;AACH,MAAM,WAAW,sBAAsB;IACrC,4DAA4D;IAC5D,EAAE,EAAE,MAAM,CAAC;IACX,qDAAqD;IACrD,QAAQ,EACJ,QAAQ,GACR,eAAe,GACf,SAAS,GACT,iBAAiB,GACjB,UAAU,GACV,cAAc,CAAC;IACnB;;;OAGG;IACH,IAAI,EAAE,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,OAAO,KAAK,OAAO,CAAC;IACpD,mEAAmE;IACnE,OAAO,EAAE,MAAM,CAAC;CACjB;AAED;;GAEG;AACH,iBAAS,UAAU,CAAC,OAAO,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,GAAG,OAAO,CAI7D;AAED,iBAAS,QAAQ,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,CAGzC;AAED,eAAO,MAAM,iCAAiC,EAAE,SAAS,sBAAsB,EAsG9E,CAAC;AAEF;;;GAGG;AACH,wBAAgB,2BAA2B,CACzC,OAAO,EAAE,MAAM,EACf,KAAK,CAAC,EAAE,OAAO,GACd,sBAAsB,GAAG,SAAS,CAKpC;AAED;;GAEG;AACH,wBAAgB,uBAAuB,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,OAAO,GAAG,OAAO,CAEjF;AAED;;;;;;;;;;;;;;;;;GAiBG;AACH,wBAAgB,mCAAmC,CACjD,UAAU,EAAE,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,OAAO,KAAK,OAAO,EACzD,QAAQ,EAAE,MAAM,EAChB,UAAU,EAAE,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,EAAE,sBAAsB,KAAK,IAAI,GACnE,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,CAAC,EAAE,OAAO,KAAK,OAAO,CAY/C;AAED;;;;;GAKG;AACH,eAAO,MAAM,QAAQ;;;CAGpB,CAAC"}
@@ -0,0 +1,231 @@
1
+ /**
2
+ * Personal-vault default exclusions — second-tier scope filter that complements
3
+ * `PERSONAL_VAULT_EXCLUDED_TOP_LEVEL` (`.git`, `companies`, `repos`,
4
+ * `workspace`) in `./personal-vault.ts`.
5
+ *
6
+ * Where the top-level constant filters the four big buckets at scope root, this
7
+ * file filters categories of nested files that ride along with an HQ install
8
+ * but have no business round-tripping to a personal vault — regardless of
9
+ * where they sit in the tree:
10
+ *
11
+ * 1. Secrets at root or nested. `.env`, `.env.local`, `.env.<anything>`,
12
+ * `.mcp.json`. Pre-fix these were being uploaded if the user hadn't
13
+ * added `.env` to their `.hqignore` (the default starter `.hqignore`
14
+ * only listed `companies/*\/settings/`).
15
+ *
16
+ * 2. Machine-local state. `.beads/` (SQLite issue-tracker + WAL/SHM),
17
+ * `.obsidian/`, `.vercel/`, `.cache_*`. Per-machine layouts/caches;
18
+ * multi-device sync either corrupts (SQLite WAL) or causes spurious
19
+ * churn (caches regenerate on demand).
20
+ *
21
+ * 3. Update-flow scratch. `output/`, `_legacy-*` directories. Created by
22
+ * `/update-hq` / `/promote-hq-core` workflows as checkout scratch;
23
+ * naming themselves "legacy" / sitting under "output" is a self-
24
+ * declared "do not preserve" signal.
25
+ *
26
+ * 4. Pre-5.24 conflict mirror dir. `.hq-conflicts/` is a directory of
27
+ * mirror copies from older sync runs (sibling to the now-fixed
28
+ * `.conflict-YYYY-MM-DDTHH-MM-SSZ-{hash}.{ext}` ephemeral files
29
+ * handled by `EPHEMERAL_PATH_PATTERN` in share.ts). Same logic
30
+ * applies: these are local-only safety backups.
31
+ *
32
+ * 5. OS / build cruft. `.DS_Store`, `node_modules/`, `dist/`, `.next/`,
33
+ * `build/`. Universally noise inside HQ (personal scope should not
34
+ * contain code projects, but defense-in-depth catches anyone who
35
+ * symlinks a project subtree in).
36
+ *
37
+ * Policy: refuse + warn (5.25 default). The walk filter drops these so they
38
+ * never upload; the delete-plan walker uses the same filter so already-
39
+ * journaled entries that match a new exclusion get orphaned in the journal
40
+ * (no DELETE issued, no churn). A one-shot purge script handles the cleanup
41
+ * of objects that landed on remote before this version.
42
+ *
43
+ * Application scope: personal vault only. Company vaults have separate
44
+ * first-push protection (settings/, data/, workers/, .git/ exclusion in
45
+ * `src-tauri/src/util/ignore.rs`) and may legitimately ship `output/` or
46
+ * `.env*` paths inside their data folders. Wired in `share.ts` only when
47
+ * `options.personalMode === true`.
48
+ *
49
+ * Wire-points (parallel to `EPHEMERAL_PATH_PATTERN`):
50
+ * - `collectFiles` / `walkDir` (push) — wrap `shouldSync` so excluded
51
+ * relative paths are rejected before upload.
52
+ * - `computeDeletePlan` (delete) — same `shouldSync` wrap, so journal
53
+ * entries matching an exclusion are skipped on the delete pass too.
54
+ */
55
+ import * as path from "path";
56
+ /**
57
+ * Cheap segment-equality check. Avoids allocating a regex per call.
58
+ */
59
+ function hasSegment(relPath, segment) {
60
+ if (relPath === segment)
61
+ return true;
62
+ if (relPath.startsWith(`${segment}/`))
63
+ return true;
64
+ return relPath.includes(`/${segment}/`) || relPath.endsWith(`/${segment}`);
65
+ }
66
+ function basename(relPath) {
67
+ const i = relPath.lastIndexOf("/");
68
+ return i === -1 ? relPath : relPath.slice(i + 1);
69
+ }
70
+ export const PERSONAL_VAULT_DEFAULT_EXCLUSIONS = [
71
+ // ── secrets ───────────────────────────────────────────────────────────
72
+ {
73
+ id: "env-file",
74
+ category: "secret",
75
+ pattern: "**/.env, **/.env.*",
76
+ test: (p) => {
77
+ const b = basename(p);
78
+ return b === ".env" || b.startsWith(".env.");
79
+ },
80
+ },
81
+ {
82
+ id: "mcp-config",
83
+ category: "secret",
84
+ pattern: "**/.mcp.json",
85
+ test: (p) => basename(p) === ".mcp.json",
86
+ },
87
+ // ── machine-local tooling state ───────────────────────────────────────
88
+ {
89
+ id: "beads-db",
90
+ category: "machine-local",
91
+ pattern: "**/.beads/**",
92
+ test: (p) => hasSegment(p, ".beads"),
93
+ },
94
+ {
95
+ id: "obsidian-workspace",
96
+ category: "machine-local",
97
+ pattern: "**/.obsidian/**",
98
+ test: (p) => hasSegment(p, ".obsidian"),
99
+ },
100
+ {
101
+ id: "vercel-link",
102
+ category: "machine-local",
103
+ pattern: "**/.vercel/**",
104
+ test: (p) => hasSegment(p, ".vercel"),
105
+ },
106
+ {
107
+ id: "cache-dir",
108
+ category: "machine-local",
109
+ pattern: "**/.cache_*",
110
+ test: (p) => basename(p).startsWith(".cache_"),
111
+ },
112
+ // ── update-flow scratch ───────────────────────────────────────────────
113
+ {
114
+ id: "update-output",
115
+ category: "scratch",
116
+ pattern: "**/output/**",
117
+ test: (p) => hasSegment(p, "output"),
118
+ },
119
+ {
120
+ id: "legacy-dir",
121
+ category: "scratch",
122
+ pattern: "**/_legacy-*/**",
123
+ test: (p) => {
124
+ // Match any segment that starts with `_legacy-` or `_legacy_` or equals `_legacy`.
125
+ const segs = p.split("/");
126
+ return segs.some((s) => s === "_legacy" || s.startsWith("_legacy-") || s.startsWith("_legacy_"));
127
+ },
128
+ },
129
+ // ── pre-5.24 conflict mirror directory ────────────────────────────────
130
+ // Sibling to EPHEMERAL_PATH_PATTERN (handles the .conflict-<iso>-<hash>.ext
131
+ // file form). This handles the older directory form some HQ installs
132
+ // accumulated.
133
+ {
134
+ id: "hq-conflicts-dir",
135
+ category: "conflict-mirror",
136
+ pattern: "**/.hq-conflicts/**",
137
+ test: (p) => hasSegment(p, ".hq-conflicts"),
138
+ },
139
+ // ── OS / build cruft ──────────────────────────────────────────────────
140
+ {
141
+ id: "ds-store",
142
+ category: "os-cruft",
143
+ pattern: "**/.DS_Store",
144
+ test: (p) => basename(p) === ".DS_Store",
145
+ },
146
+ {
147
+ id: "node-modules",
148
+ category: "build-output",
149
+ pattern: "**/node_modules/**",
150
+ test: (p) => hasSegment(p, "node_modules"),
151
+ },
152
+ {
153
+ id: "dist-dir",
154
+ category: "build-output",
155
+ pattern: "**/dist/**",
156
+ test: (p) => hasSegment(p, "dist"),
157
+ },
158
+ {
159
+ id: "next-dir",
160
+ category: "build-output",
161
+ pattern: "**/.next/**",
162
+ test: (p) => hasSegment(p, ".next"),
163
+ },
164
+ {
165
+ id: "build-dir",
166
+ category: "build-output",
167
+ pattern: "**/build/**",
168
+ test: (p) => hasSegment(p, "build"),
169
+ },
170
+ ];
171
+ /**
172
+ * First-match result for a path. Returns the matching exclusion (so callers
173
+ * can surface category/id in events) or undefined for "not excluded".
174
+ */
175
+ export function matchPersonalVaultExclusion(relPath, isDir) {
176
+ for (const ex of PERSONAL_VAULT_DEFAULT_EXCLUSIONS) {
177
+ if (ex.test(relPath, isDir))
178
+ return ex;
179
+ }
180
+ return undefined;
181
+ }
182
+ /**
183
+ * Boolean version — true if any exclusion matches.
184
+ */
185
+ export function isPersonalVaultExcluded(relPath, isDir) {
186
+ return matchPersonalVaultExclusion(relPath, isDir) !== undefined;
187
+ }
188
+ /**
189
+ * Wrap an existing path filter (typically from `createIgnoreFilter`) with the
190
+ * personal-vault default exclusions. The wrapper:
191
+ *
192
+ * 1. Calls the underlying filter first; if it already rejects, defer
193
+ * (counter does not fire — we only want to count things the underlying
194
+ * filter would have allowed but the personal-vault defaults reject).
195
+ * 2. Computes a relative path from `syncRoot` (the personal-vault sync
196
+ * root, which is hq_root in personalMode).
197
+ * 3. Probes `matchPersonalVaultExclusion`; if it matches, returns false
198
+ * and tags the match via `onExcluded` so the runner can emit a single
199
+ * `personal-vault-out-of-policy` event with a count + sample at end of
200
+ * run.
201
+ *
202
+ * The wrapper keeps the `(absolutePath, isDir) => boolean` shape the existing
203
+ * collectFiles / walkDir / computeDeletePlan code already uses, so wiring is
204
+ * a single line change at the share() filter construction site.
205
+ */
206
+ export function wrapFilterWithPersonalVaultDefaults(underlying, syncRoot, onExcluded) {
207
+ return (absPath, isDir) => {
208
+ if (!underlying(absPath, isDir))
209
+ return false;
210
+ const rel = path.relative(syncRoot, absPath).split(path.sep).join("/");
211
+ if (rel === "" || rel.startsWith(".."))
212
+ return true; // outside scope, defer
213
+ const match = matchPersonalVaultExclusion(rel, isDir);
214
+ if (match) {
215
+ onExcluded(rel, match);
216
+ return false;
217
+ }
218
+ return true;
219
+ };
220
+ }
221
+ /**
222
+ * Test-only export. Mirrors the `_testing` namespace pattern used by
223
+ * `share.ts` for `EPHEMERAL_PATH_PATTERN` — direct access to the list +
224
+ * internal helpers for regression-critical pinning without round-tripping
225
+ * through share().
226
+ */
227
+ export const _testing = {
228
+ hasSegment,
229
+ basename,
230
+ };
231
+ //# sourceMappingURL=personal-vault-exclusions.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"personal-vault-exclusions.js","sourceRoot":"","sources":["../src/personal-vault-exclusions.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,22 @@
1
+ /**
2
+ * Tests for `personal-vault-exclusions.ts`. Pin the regex / segment contracts
3
+ * for every entry in PERSONAL_VAULT_DEFAULT_EXCLUSIONS so a future edit that
4
+ * relaxes one (e.g. dropping the `_legacy-` prefix match, or changing the
5
+ * `.cache_*` shape) surfaces here instead of in the field.
6
+ *
7
+ * Two test layers:
8
+ *
9
+ * 1. Per-rule predicate tests. Each exclusion is exercised with at least
10
+ * one match-positive and one match-negative path so the regex shape is
11
+ * pinned. The `byId` event field on the runner uses `ex.id` directly,
12
+ * so the id stability is part of the contract too — tests assert
13
+ * against the literal id string.
14
+ *
15
+ * 2. Filter-wrap integration. Confirms that `wrapFilterWithPersonalVaultDefaults`
16
+ * defers to the underlying filter, computes the relative path correctly,
17
+ * and only fires the `onExcluded` callback when the underlying filter
18
+ * WOULD have allowed the path (so the count tracks only "newly blocked
19
+ * by these defaults", not "blocked by .hqignore too").
20
+ */
21
+ export {};
22
+ //# sourceMappingURL=personal-vault-exclusions.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"personal-vault-exclusions.test.d.ts","sourceRoot":"","sources":["../src/personal-vault-exclusions.test.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;GAmBG"}
@@ -0,0 +1,198 @@
1
+ /**
2
+ * Tests for `personal-vault-exclusions.ts`. Pin the regex / segment contracts
3
+ * for every entry in PERSONAL_VAULT_DEFAULT_EXCLUSIONS so a future edit that
4
+ * relaxes one (e.g. dropping the `_legacy-` prefix match, or changing the
5
+ * `.cache_*` shape) surfaces here instead of in the field.
6
+ *
7
+ * Two test layers:
8
+ *
9
+ * 1. Per-rule predicate tests. Each exclusion is exercised with at least
10
+ * one match-positive and one match-negative path so the regex shape is
11
+ * pinned. The `byId` event field on the runner uses `ex.id` directly,
12
+ * so the id stability is part of the contract too — tests assert
13
+ * against the literal id string.
14
+ *
15
+ * 2. Filter-wrap integration. Confirms that `wrapFilterWithPersonalVaultDefaults`
16
+ * defers to the underlying filter, computes the relative path correctly,
17
+ * and only fires the `onExcluded` callback when the underlying filter
18
+ * WOULD have allowed the path (so the count tracks only "newly blocked
19
+ * by these defaults", not "blocked by .hqignore too").
20
+ */
21
+ import { describe, expect, it } from "vitest";
22
+ import * as path from "node:path";
23
+ import { PERSONAL_VAULT_DEFAULT_EXCLUSIONS, isPersonalVaultExcluded, matchPersonalVaultExclusion, wrapFilterWithPersonalVaultDefaults, _testing, } from "./personal-vault-exclusions.js";
24
+ describe("PERSONAL_VAULT_DEFAULT_EXCLUSIONS", () => {
25
+ it("exports a non-empty, frozen-shaped list", () => {
26
+ expect(PERSONAL_VAULT_DEFAULT_EXCLUSIONS.length).toBeGreaterThan(10);
27
+ // Every entry has the four required fields; this pins the public shape
28
+ // so a future addition without `category` or `id` fails fast.
29
+ for (const ex of PERSONAL_VAULT_DEFAULT_EXCLUSIONS) {
30
+ expect(typeof ex.id).toBe("string");
31
+ expect(ex.id.length).toBeGreaterThan(0);
32
+ expect(typeof ex.category).toBe("string");
33
+ expect(typeof ex.test).toBe("function");
34
+ expect(typeof ex.pattern).toBe("string");
35
+ }
36
+ });
37
+ it("has unique ids (event byId aggregation requires it)", () => {
38
+ const ids = new Set();
39
+ for (const ex of PERSONAL_VAULT_DEFAULT_EXCLUSIONS) {
40
+ expect(ids.has(ex.id)).toBe(false);
41
+ ids.add(ex.id);
42
+ }
43
+ });
44
+ });
45
+ // ── Per-rule predicate matrix ───────────────────────────────────────────────
46
+ //
47
+ // One block per rule. The positives prove "this path SHOULD be excluded"; the
48
+ // negatives prove "this path SHOULD NOT" (guards against over-broad regexes
49
+ // that would block legitimate user content).
50
+ describe("env-file exclusion (secrets)", () => {
51
+ it.each([".env", ".env.local", ".env.production", "core/.env", "personal/.env.local"])("matches %s", (p) => {
52
+ expect(isPersonalVaultExcluded(p)).toBe(true);
53
+ expect(matchPersonalVaultExclusion(p)?.id).toBe("env-file");
54
+ });
55
+ it.each(["envconfig.md", ".env-example.md", "core/policies/env-vars.md"])("does NOT match %s", (p) => {
56
+ // ".env-example.md" is intentionally a negative — the basename starts
57
+ // with ".env-", not ".env." (dot vs hyphen). We want to allow docs
58
+ // that reference env-vars without the regex grabbing them.
59
+ const m = matchPersonalVaultExclusion(p);
60
+ expect(m?.id).not.toBe("env-file");
61
+ });
62
+ });
63
+ describe("mcp-config exclusion (secrets)", () => {
64
+ it.each([".mcp.json", "personal/.mcp.json", "core/.claude/.mcp.json"])("matches %s", (p) => {
65
+ expect(matchPersonalVaultExclusion(p)?.id).toBe("mcp-config");
66
+ });
67
+ it.each(["mcp.json", "core/policies/mcp.md", ".mcp.example.json"])("does NOT match %s", (p) => {
68
+ expect(matchPersonalVaultExclusion(p)?.id).not.toBe("mcp-config");
69
+ });
70
+ });
71
+ describe("beads exclusion (machine-local SQLite)", () => {
72
+ it.each([
73
+ ".beads/beads.db",
74
+ ".beads/beads.db-wal",
75
+ ".beads/beads.db-shm",
76
+ ".beads/beads.left.jsonl",
77
+ "personal/.beads/issue.json",
78
+ ])("matches %s", (p) => {
79
+ expect(matchPersonalVaultExclusion(p)?.id).toBe("beads-db");
80
+ });
81
+ it.each(["beads.md", "core/knowledge/beads-guide.md"])("does NOT match %s", (p) => {
82
+ expect(matchPersonalVaultExclusion(p)?.id).not.toBe("beads-db");
83
+ });
84
+ });
85
+ describe("obsidian / vercel / cache exclusions", () => {
86
+ it("matches .obsidian/", () => {
87
+ expect(matchPersonalVaultExclusion(".obsidian/workspace.json")?.id).toBe("obsidian-workspace");
88
+ });
89
+ it("matches .vercel/", () => {
90
+ expect(matchPersonalVaultExclusion(".vercel/project.json")?.id).toBe("vercel-link");
91
+ });
92
+ it("matches .cache_ggshield", () => {
93
+ expect(matchPersonalVaultExclusion(".cache_ggshield")?.id).toBe("cache-dir");
94
+ });
95
+ it("matches .cache_anything", () => {
96
+ expect(matchPersonalVaultExclusion(".cache_foo")?.id).toBe("cache-dir");
97
+ });
98
+ it("does NOT match .cache (no underscore)", () => {
99
+ // `.cache_*` is the segment shape — bare `.cache` should be left for
100
+ // other rules / .hqignore. Pinned so a future widen-of-pattern surfaces here.
101
+ expect(matchPersonalVaultExclusion(".cache")).toBeUndefined();
102
+ });
103
+ });
104
+ describe("update-output / legacy / hq-conflicts exclusions (scratch)", () => {
105
+ it.each([
106
+ "output/hatch-pet/scaffold.md",
107
+ "personal/data/output/hatch-pet/file.md",
108
+ "core/output/x.md",
109
+ ])("matches %s as update-output", (p) => {
110
+ expect(matchPersonalVaultExclusion(p)?.id).toBe("update-output");
111
+ });
112
+ it.each([
113
+ "_legacy-pre14.2/x.md",
114
+ "personal/_legacy-2024/old.md",
115
+ "_legacy_v1/y.md",
116
+ "_legacy/z.md",
117
+ ])("matches %s as legacy-dir", (p) => {
118
+ expect(matchPersonalVaultExclusion(p)?.id).toBe("legacy-dir");
119
+ });
120
+ it.each([".hq-conflicts/file.md", "personal/.hq-conflicts/x.md"])("matches %s as hq-conflicts-dir", (p) => {
121
+ expect(matchPersonalVaultExclusion(p)?.id).toBe("hq-conflicts-dir");
122
+ });
123
+ it("does NOT match 'output.md' (basename collision guard)", () => {
124
+ expect(matchPersonalVaultExclusion("core/output.md")).toBeUndefined();
125
+ });
126
+ });
127
+ describe("OS / build cruft exclusions", () => {
128
+ it("matches .DS_Store anywhere", () => {
129
+ expect(matchPersonalVaultExclusion(".DS_Store")?.id).toBe("ds-store");
130
+ expect(matchPersonalVaultExclusion("personal/.DS_Store")?.id).toBe("ds-store");
131
+ });
132
+ it.each([
133
+ ["node_modules/x.js", "node-modules"],
134
+ ["personal/x/node_modules/y.js", "node-modules"],
135
+ ["dist/bundle.js", "dist-dir"],
136
+ [".next/build/file.js", "next-dir"],
137
+ ["build/output.js", "build-dir"],
138
+ ])("matches %s as %s", (p, expectedId) => {
139
+ expect(matchPersonalVaultExclusion(p)?.id).toBe(expectedId);
140
+ });
141
+ });
142
+ // ── _testing internals ───────────────────────────────────────────────────────
143
+ describe("_testing internals", () => {
144
+ it("hasSegment matches exact / prefix / middle / suffix forms", () => {
145
+ expect(_testing.hasSegment("foo", "foo")).toBe(true);
146
+ expect(_testing.hasSegment("foo/bar", "foo")).toBe(true);
147
+ expect(_testing.hasSegment("a/foo/b", "foo")).toBe(true);
148
+ expect(_testing.hasSegment("a/foo", "foo")).toBe(true);
149
+ expect(_testing.hasSegment("foobar", "foo")).toBe(false);
150
+ expect(_testing.hasSegment("a/foobar", "foo")).toBe(false);
151
+ });
152
+ it("basename returns the trailing segment", () => {
153
+ expect(_testing.basename("a/b/c.md")).toBe("c.md");
154
+ expect(_testing.basename("c.md")).toBe("c.md");
155
+ expect(_testing.basename("")).toBe("");
156
+ });
157
+ });
158
+ // ── Filter-wrap integration ──────────────────────────────────────────────────
159
+ describe("wrapFilterWithPersonalVaultDefaults", () => {
160
+ // Use a syncRoot path independent of the test runner's cwd so the
161
+ // path.relative() calls inside the wrapper produce predictable results
162
+ // regardless of where the test runs from.
163
+ const syncRoot = "/tmp/hq-root";
164
+ it("defers to the underlying filter when it rejects", () => {
165
+ const calls = [];
166
+ const wrapped = wrapFilterWithPersonalVaultDefaults(() => false, // underlying rejects everything
167
+ syncRoot, (rel, match) => calls.push({ rel, id: match.id }));
168
+ // Even though `.env` matches a default exclusion, the underlying
169
+ // filter already rejected it — onExcluded must NOT fire (we only
170
+ // want to count things that would have made it past the user's
171
+ // .hqignore but are blocked by our defaults).
172
+ expect(wrapped(path.join(syncRoot, ".env"))).toBe(false);
173
+ expect(calls).toEqual([]);
174
+ });
175
+ it("rejects and tags onExcluded when underlying allows + default blocks", () => {
176
+ const calls = [];
177
+ const wrapped = wrapFilterWithPersonalVaultDefaults(() => true, // underlying allows everything
178
+ syncRoot, (rel, match) => calls.push({ rel, id: match.id }));
179
+ expect(wrapped(path.join(syncRoot, ".env"))).toBe(false);
180
+ expect(wrapped(path.join(syncRoot, "personal", ".env.local"))).toBe(false);
181
+ expect(wrapped(path.join(syncRoot, "personal", "agents.md"))).toBe(true);
182
+ expect(calls).toEqual([
183
+ { rel: ".env", id: "env-file" },
184
+ { rel: "personal/.env.local", id: "env-file" },
185
+ // "personal/agents.md" was allowed — no onExcluded callback fired.
186
+ ]);
187
+ });
188
+ it("does not fire onExcluded for paths outside syncRoot (defensive)", () => {
189
+ const calls = [];
190
+ const wrapped = wrapFilterWithPersonalVaultDefaults(() => true, syncRoot, (rel, match) => calls.push({ rel, id: match.id }));
191
+ // Outside-syncRoot paths come back with a `..` prefix from path.relative;
192
+ // the wrapper defers (returns true) without consulting the exclusion
193
+ // list since "outside scope" is the upstream containment check's job.
194
+ expect(wrapped("/tmp/other/.env")).toBe(true);
195
+ expect(calls).toEqual([]);
196
+ });
197
+ });
198
+ //# sourceMappingURL=personal-vault-exclusions.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"personal-vault-exclusions.test.js","sourceRoot":"","sources":["../src/personal-vault-exclusions.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,11 @@
1
+ /**
2
+ * @indigoai-us/hq-cloud — event-driven sync (`src/sync`) barrel.
3
+ *
4
+ * Surfaces the PushEvent wire contract + the PushTransport shipping seam
5
+ * ported from hq-pro PR #112 (project event-driven-sync-menubar US-007).
6
+ */
7
+ export { CONTENT_HASH_PATTERN, ISO8601_DATETIME_PATTERN, PushEventSchema, PushEventDecodeError, encodePushEvent, decodePushEvent, } from "./push-event.js";
8
+ export type { PushEvent, PushEventDecodeIssue } from "./push-event.js";
9
+ export { NoopPushTransport } from "./push-transport.js";
10
+ export type { PushTransport } from "./push-transport.js";
11
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/sync/index.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,OAAO,EACL,oBAAoB,EACpB,wBAAwB,EACxB,eAAe,EACf,oBAAoB,EACpB,eAAe,EACf,eAAe,GAChB,MAAM,iBAAiB,CAAC;AACzB,YAAY,EAAE,SAAS,EAAE,oBAAoB,EAAE,MAAM,iBAAiB,CAAC;AAEvE,OAAO,EAAE,iBAAiB,EAAE,MAAM,qBAAqB,CAAC;AACxD,YAAY,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC"}
@@ -0,0 +1,9 @@
1
+ /**
2
+ * @indigoai-us/hq-cloud — event-driven sync (`src/sync`) barrel.
3
+ *
4
+ * Surfaces the PushEvent wire contract + the PushTransport shipping seam
5
+ * ported from hq-pro PR #112 (project event-driven-sync-menubar US-007).
6
+ */
7
+ export { CONTENT_HASH_PATTERN, ISO8601_DATETIME_PATTERN, PushEventSchema, PushEventDecodeError, encodePushEvent, decodePushEvent, } from "./push-event.js";
8
+ export { NoopPushTransport } from "./push-transport.js";
9
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/sync/index.ts"],"names":[],"mappings":"AAAA;;;;;GAKG;AAEH,OAAO,EACL,oBAAoB,EACpB,wBAAwB,EACxB,eAAe,EACf,oBAAoB,EACpB,eAAe,EACf,eAAe,GAChB,MAAM,iBAAiB,CAAC;AAGzB,OAAO,EAAE,iBAAiB,EAAE,MAAM,qBAAqB,CAAC"}