@nocoo/otter 1.0.3 → 1.2.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 (65) hide show
  1. package/dist/cli.d.ts.map +1 -1
  2. package/dist/cli.js +162 -85
  3. package/dist/cli.js.map +1 -1
  4. package/dist/collectors/applications.d.ts +7 -2
  5. package/dist/collectors/applications.d.ts.map +1 -1
  6. package/dist/collectors/applications.js +55 -21
  7. package/dist/collectors/applications.js.map +1 -1
  8. package/dist/collectors/base.d.ts.map +1 -1
  9. package/dist/collectors/base.js +1 -0
  10. package/dist/collectors/base.js.map +1 -1
  11. package/dist/collectors/cloud-cli.d.ts +9 -0
  12. package/dist/collectors/cloud-cli.d.ts.map +1 -0
  13. package/dist/collectors/cloud-cli.js +61 -0
  14. package/dist/collectors/cloud-cli.js.map +1 -0
  15. package/dist/collectors/dev-toolchain.d.ts +21 -0
  16. package/dist/collectors/dev-toolchain.d.ts.map +1 -0
  17. package/dist/collectors/dev-toolchain.js +252 -0
  18. package/dist/collectors/dev-toolchain.js.map +1 -0
  19. package/dist/collectors/docker.d.ts +10 -0
  20. package/dist/collectors/docker.d.ts.map +1 -0
  21. package/dist/collectors/docker.js +57 -0
  22. package/dist/collectors/docker.js.map +1 -0
  23. package/dist/collectors/fonts.d.ts +9 -0
  24. package/dist/collectors/fonts.d.ts.map +1 -0
  25. package/dist/collectors/fonts.js +36 -0
  26. package/dist/collectors/fonts.js.map +1 -0
  27. package/dist/collectors/homebrew.d.ts +2 -1
  28. package/dist/collectors/homebrew.d.ts.map +1 -1
  29. package/dist/collectors/homebrew.js +56 -12
  30. package/dist/collectors/homebrew.js.map +1 -1
  31. package/dist/collectors/index.d.ts +7 -0
  32. package/dist/collectors/index.d.ts.map +1 -1
  33. package/dist/collectors/index.js +21 -0
  34. package/dist/collectors/index.js.map +1 -1
  35. package/dist/collectors/launch-agents.d.ts +10 -0
  36. package/dist/collectors/launch-agents.d.ts.map +1 -0
  37. package/dist/collectors/launch-agents.js +53 -0
  38. package/dist/collectors/launch-agents.js.map +1 -0
  39. package/dist/collectors/macos-defaults.d.ts +10 -0
  40. package/dist/collectors/macos-defaults.d.ts.map +1 -0
  41. package/dist/collectors/macos-defaults.js +53 -0
  42. package/dist/collectors/macos-defaults.js.map +1 -0
  43. package/dist/collectors/vscode.d.ts +15 -0
  44. package/dist/collectors/vscode.d.ts.map +1 -0
  45. package/dist/collectors/vscode.js +118 -0
  46. package/dist/collectors/vscode.js.map +1 -0
  47. package/dist/commands/scan.d.ts +2 -0
  48. package/dist/commands/scan.d.ts.map +1 -1
  49. package/dist/commands/scan.js +2 -0
  50. package/dist/commands/scan.js.map +1 -1
  51. package/dist/commands/snapshot.d.ts +2 -9
  52. package/dist/commands/snapshot.d.ts.map +1 -1
  53. package/dist/commands/snapshot.js +74 -64
  54. package/dist/commands/snapshot.js.map +1 -1
  55. package/dist/snapshot/builder.d.ts.map +1 -1
  56. package/dist/snapshot/builder.js +1 -0
  57. package/dist/snapshot/builder.js.map +1 -1
  58. package/dist/ui.d.ts +63 -0
  59. package/dist/ui.d.ts.map +1 -0
  60. package/dist/ui.js +167 -0
  61. package/dist/ui.js.map +1 -0
  62. package/dist/utils/icons.d.ts.map +1 -1
  63. package/dist/utils/icons.js +59 -7
  64. package/dist/utils/icons.js.map +1 -1
  65. package/package.json +3 -2
@@ -0,0 +1,36 @@
1
+ import { readdir } from "node:fs/promises";
2
+ import { extname, join } from "node:path";
3
+ import { BaseCollector } from "./base.js";
4
+ export class FontsCollector extends BaseCollector {
5
+ id = "fonts";
6
+ label = "Installed Fonts";
7
+ category = "environment";
8
+ async collect() {
9
+ return this.timed(async (result) => {
10
+ const fontsDir = join(this.homeDir, "Library", "Fonts");
11
+ try {
12
+ const entries = await readdir(fontsDir, { withFileTypes: true });
13
+ const items = entries
14
+ .filter((entry) => entry.isFile())
15
+ .map((entry) => {
16
+ const ext = extname(entry.name);
17
+ return {
18
+ name: entry.name.slice(0, ext ? -ext.length : undefined),
19
+ meta: {
20
+ type: "font",
21
+ format: ext.replace(/^\./, "").toLowerCase() || "unknown",
22
+ },
23
+ };
24
+ })
25
+ .sort((a, b) => a.name.localeCompare(b.name));
26
+ result.lists.push(...items);
27
+ }
28
+ catch (err) {
29
+ if (err.code !== "ENOENT") {
30
+ result.errors.push(`Failed to read fonts directory: ${err.message}`);
31
+ }
32
+ }
33
+ });
34
+ }
35
+ }
36
+ //# sourceMappingURL=fonts.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fonts.js","sourceRoot":"","sources":["../../src/collectors/fonts.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,MAAM,kBAAkB,CAAC;AAC3C,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAO1C,MAAM,OAAO,cAAe,SAAQ,aAAa;IACtC,EAAE,GAAG,OAAO,CAAC;IACb,KAAK,GAAG,iBAAiB,CAAC;IAC1B,QAAQ,GAAsB,aAAa,CAAC;IAErD,KAAK,CAAC,OAAO;QACX,OAAO,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,MAAM,EAAE,EAAE;YACjC,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,SAAS,EAAE,OAAO,CAAC,CAAC;YACxD,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,QAAQ,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;gBACjE,MAAM,KAAK,GAAwB,OAAO;qBACvC,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC;qBACjC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE;oBACb,MAAM,GAAG,GAAG,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;oBAChC,OAAO;wBACL,IAAI,EAAE,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,SAAS,CAAC;wBACxD,IAAI,EAAE;4BACJ,IAAI,EAAE,MAAM;4BACZ,MAAM,EAAE,GAAG,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,WAAW,EAAE,IAAI,SAAS;yBAC1D;qBACF,CAAC;gBACJ,CAAC,CAAC;qBACD,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;gBAEhD,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,KAAK,CAAC,CAAC;YAC9B,CAAC;YAAC,OAAO,GAAG,EAAE,CAAC;gBACb,IAAK,GAA6B,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;oBACrD,MAAM,CAAC,MAAM,CAAC,IAAI,CAChB,mCAAoC,GAAa,CAAC,OAAO,EAAE,CAC5D,CAAC;gBACJ,CAAC;YACH,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;CACF"}
@@ -11,6 +11,7 @@ export declare class HomebrewCollector extends BaseCollector {
11
11
  /** Overridable for testing — executes a shell command and returns stdout */
12
12
  _execCommand: (cmd: string) => Promise<string>;
13
13
  collect(): Promise<CollectorResult>;
14
- private listPackages;
14
+ private collectList;
15
+ private markPinnedPackages;
15
16
  }
16
17
  //# sourceMappingURL=homebrew.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"homebrew.d.ts","sourceRoot":"","sources":["../../src/collectors/homebrew.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,KAAK,EACV,iBAAiB,EACjB,eAAe,EAEhB,MAAM,aAAa,CAAC;AAIrB;;;GAGG;AACH,qBAAa,iBAAkB,SAAQ,aAAa;IAClD,QAAQ,CAAC,EAAE,cAAc;IACzB,QAAQ,CAAC,KAAK,uBAAuB;IACrC,QAAQ,CAAC,QAAQ,EAAE,iBAAiB,CAAiB;IAErD,4EAA4E;IAC5E,YAAY,GAAU,KAAK,MAAM,KAAG,OAAO,CAAC,MAAM,CAAC,CAGjD;IAEI,OAAO,IAAI,OAAO,CAAC,eAAe,CAAC;YAoB3B,YAAY;CAmB3B"}
1
+ {"version":3,"file":"homebrew.d.ts","sourceRoot":"","sources":["../../src/collectors/homebrew.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,KAAK,EACV,iBAAiB,EACjB,eAAe,EAEhB,MAAM,aAAa,CAAC;AAgDrB;;;GAGG;AACH,qBAAa,iBAAkB,SAAQ,aAAa;IAClD,QAAQ,CAAC,EAAE,cAAc;IACzB,QAAQ,CAAC,KAAK,uBAAuB;IACrC,QAAQ,CAAC,QAAQ,EAAE,iBAAiB,CAAiB;IAErD,4EAA4E;IAC5E,YAAY,GAAU,KAAK,MAAM,KAAG,OAAO,CAAC,MAAM,CAAC,CAGjD;IAEI,OAAO,IAAI,OAAO,CAAC,eAAe,CAAC;YA+B3B,WAAW;YAiBX,kBAAkB;CAajC"}
@@ -2,6 +2,44 @@ import { exec } from "node:child_process";
2
2
  import { promisify } from "node:util";
3
3
  import { BaseCollector } from "./base.js";
4
4
  const execAsync = promisify(exec);
5
+ function parseVersionedItems(output, type) {
6
+ return output
7
+ .split("\n")
8
+ .map((line) => line.trim())
9
+ .filter((line) => line.length > 0)
10
+ .map((line) => {
11
+ const [name, ...versions] = line.split(/\s+/);
12
+ if (!name) {
13
+ return { name: "", meta: { type } };
14
+ }
15
+ return {
16
+ name,
17
+ ...(versions.length > 0 ? { version: versions.join(" ") } : {}),
18
+ meta: { type },
19
+ };
20
+ });
21
+ }
22
+ function parseTapItems(output) {
23
+ return output
24
+ .split("\n")
25
+ .map((line) => line.trim())
26
+ .filter((line) => line.length > 0)
27
+ .map((name) => ({ name, meta: { type: "tap" } }));
28
+ }
29
+ function applyPinnedPackages(items, output) {
30
+ const pinned = new Set(output
31
+ .split("\n")
32
+ .map((line) => line.trim())
33
+ .filter((line) => line.length > 0));
34
+ for (const item of items) {
35
+ if (pinned.has(item.name)) {
36
+ item.meta = {
37
+ ...item.meta,
38
+ pinned: "true",
39
+ };
40
+ }
41
+ }
42
+ }
5
43
  /**
6
44
  * Collects installed Homebrew packages (formulae + casks).
7
45
  * List-only: no binary content is collected.
@@ -17,27 +55,33 @@ export class HomebrewCollector extends BaseCollector {
17
55
  };
18
56
  async collect() {
19
57
  return this.timed(async (result) => {
20
- // Collect formulae
21
- const formulae = await this.listPackages("brew list --formula", "formula", result);
22
- result.lists.push(...formulae);
23
- // Collect casks
24
- const casks = await this.listPackages("brew list --cask", "cask", result);
25
- result.lists.push(...casks);
58
+ const formulae = await this.collectList("brew list --formula --versions", "formula", result, parseVersionedItems);
59
+ const casks = await this.collectList("brew list --cask --versions", "cask", result, parseVersionedItems);
60
+ const items = [...formulae, ...casks];
61
+ const taps = await this.collectList("brew tap", "tap", result, (output) => parseTapItems(output));
62
+ items.push(...taps);
63
+ await this.markPinnedPackages(items, result);
64
+ result.lists.push(...items);
26
65
  });
27
66
  }
28
- async listPackages(cmd, type, result) {
67
+ async collectList(cmd, type, result, parser) {
29
68
  try {
30
69
  const output = await this._execCommand(cmd);
31
- return output
32
- .split("\n")
33
- .map((line) => line.trim())
34
- .filter((line) => line.length > 0)
35
- .map((name) => ({ name, meta: { type } }));
70
+ return parser(output, type);
36
71
  }
37
72
  catch (err) {
38
73
  result.errors.push(`Failed to run '${cmd}': ${err.message}`);
39
74
  return [];
40
75
  }
41
76
  }
77
+ async markPinnedPackages(items, result) {
78
+ try {
79
+ const output = await this._execCommand("brew list --pinned");
80
+ applyPinnedPackages(items, output);
81
+ }
82
+ catch (err) {
83
+ result.errors.push(`Failed to run 'brew list --pinned': ${err.message}`);
84
+ }
85
+ }
42
86
  }
43
87
  //# sourceMappingURL=homebrew.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"homebrew.js","sourceRoot":"","sources":["../../src/collectors/homebrew.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,oBAAoB,CAAC;AAC1C,OAAO,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AACtC,OAAO,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAO1C,MAAM,SAAS,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;AAElC;;;GAGG;AACH,MAAM,OAAO,iBAAkB,SAAQ,aAAa;IACzC,EAAE,GAAG,UAAU,CAAC;IAChB,KAAK,GAAG,mBAAmB,CAAC;IAC5B,QAAQ,GAAsB,aAAa,CAAC;IAErD,4EAA4E;IAC5E,YAAY,GAAG,KAAK,EAAE,GAAW,EAAmB,EAAE;QACpD,MAAM,EAAE,MAAM,EAAE,GAAG,MAAM,SAAS,CAAC,GAAG,CAAC,CAAC;QACxC,OAAO,MAAM,CAAC;IAChB,CAAC,CAAC;IAEF,KAAK,CAAC,OAAO;QACX,OAAO,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,MAAM,EAAE,EAAE;YACjC,mBAAmB;YACnB,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,YAAY,CACtC,qBAAqB,EACrB,SAAS,EACT,MAAM,CACP,CAAC;YACF,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,QAAQ,CAAC,CAAC;YAE/B,gBAAgB;YAChB,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,YAAY,CACnC,kBAAkB,EAClB,MAAM,EACN,MAAM,CACP,CAAC;YACF,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,KAAK,CAAC,CAAC;QAC9B,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,KAAK,CAAC,YAAY,CACxB,GAAW,EACX,IAAY,EACZ,MAAuB;QAEvB,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;YAC5C,OAAO,MAAM;iBACV,KAAK,CAAC,IAAI,CAAC;iBACX,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;iBAC1B,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC;iBACjC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,EAAE,IAAI,EAAE,EAAE,CAAC,CAAC,CAAC;QAC/C,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,CAAC,MAAM,CAAC,IAAI,CAChB,kBAAkB,GAAG,MAAO,GAAa,CAAC,OAAO,EAAE,CACpD,CAAC;YACF,OAAO,EAAE,CAAC;QACZ,CAAC;IACH,CAAC;CACF"}
1
+ {"version":3,"file":"homebrew.js","sourceRoot":"","sources":["../../src/collectors/homebrew.ts"],"names":[],"mappings":"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"}
@@ -4,6 +4,13 @@ export { OpenCodeConfigCollector } from "./opencode-config.js";
4
4
  export { ShellConfigCollector } from "./shell-config.js";
5
5
  export { HomebrewCollector } from "./homebrew.js";
6
6
  export { ApplicationsCollector } from "./applications.js";
7
+ export { VSCodeCollector } from "./vscode.js";
8
+ export { DockerCollector } from "./docker.js";
9
+ export { FontsCollector } from "./fonts.js";
10
+ export { DevToolchainCollector } from "./dev-toolchain.js";
11
+ export { CloudCLICollector } from "./cloud-cli.js";
12
+ export { MacOSDefaultsCollector } from "./macos-defaults.js";
13
+ export { LaunchAgentsCollector } from "./launch-agents.js";
7
14
  import type { Collector } from "@otter/core";
8
15
  /**
9
16
  * Options for creating the default set of collectors.
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/collectors/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAC3D,OAAO,EAAE,uBAAuB,EAAE,MAAM,sBAAsB,CAAC;AAC/D,OAAO,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAC;AACzD,OAAO,EAAE,iBAAiB,EAAE,MAAM,eAAe,CAAC;AAClD,OAAO,EAAE,qBAAqB,EAAE,MAAM,mBAAmB,CAAC;AAE1D,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,aAAa,CAAC;AAW7C;;GAEG;AACH,MAAM,WAAW,gBAAgB;IAC/B,wEAAwE;IACxE,IAAI,CAAC,EAAE,OAAO,CAAC;IACf,0EAA0E;IAC1E,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAED;;GAEG;AACH,wBAAgB,uBAAuB,CACrC,OAAO,GAAE,MAAkB,EAC3B,OAAO,GAAE,gBAAqB,GAC7B,SAAS,EAAE,CASb"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/collectors/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAC3D,OAAO,EAAE,uBAAuB,EAAE,MAAM,sBAAsB,CAAC;AAC/D,OAAO,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAC;AACzD,OAAO,EAAE,iBAAiB,EAAE,MAAM,eAAe,CAAC;AAClD,OAAO,EAAE,qBAAqB,EAAE,MAAM,mBAAmB,CAAC;AAC1D,OAAO,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;AAC9C,OAAO,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;AAC9C,OAAO,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAC5C,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAC3D,OAAO,EAAE,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AACnD,OAAO,EAAE,sBAAsB,EAAE,MAAM,qBAAqB,CAAC;AAC7D,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAE3D,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,aAAa,CAAC;AAkB7C;;GAEG;AACH,MAAM,WAAW,gBAAgB;IAC/B,wEAAwE;IACxE,IAAI,CAAC,EAAE,OAAO,CAAC;IACf,0EAA0E;IAC1E,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAED;;GAEG;AACH,wBAAgB,uBAAuB,CACrC,OAAO,GAAE,MAAkB,EAC3B,OAAO,GAAE,gBAAqB,GAC7B,SAAS,EAAE,CAgBb"}
@@ -4,11 +4,25 @@ export { OpenCodeConfigCollector } from "./opencode-config.js";
4
4
  export { ShellConfigCollector } from "./shell-config.js";
5
5
  export { HomebrewCollector } from "./homebrew.js";
6
6
  export { ApplicationsCollector } from "./applications.js";
7
+ export { VSCodeCollector } from "./vscode.js";
8
+ export { DockerCollector } from "./docker.js";
9
+ export { FontsCollector } from "./fonts.js";
10
+ export { DevToolchainCollector } from "./dev-toolchain.js";
11
+ export { CloudCLICollector } from "./cloud-cli.js";
12
+ export { MacOSDefaultsCollector } from "./macos-defaults.js";
13
+ export { LaunchAgentsCollector } from "./launch-agents.js";
7
14
  import { ClaudeConfigCollector } from "./claude-config.js";
8
15
  import { OpenCodeConfigCollector } from "./opencode-config.js";
9
16
  import { ShellConfigCollector } from "./shell-config.js";
10
17
  import { HomebrewCollector } from "./homebrew.js";
11
18
  import { ApplicationsCollector } from "./applications.js";
19
+ import { VSCodeCollector } from "./vscode.js";
20
+ import { DockerCollector } from "./docker.js";
21
+ import { FontsCollector } from "./fonts.js";
22
+ import { DevToolchainCollector } from "./dev-toolchain.js";
23
+ import { CloudCLICollector } from "./cloud-cli.js";
24
+ import { MacOSDefaultsCollector } from "./macos-defaults.js";
25
+ import { LaunchAgentsCollector } from "./launch-agents.js";
12
26
  import { homedir } from "node:os";
13
27
  /** Default R2 public base URL for app icon assets */
14
28
  const DEFAULT_ICON_BASE_URL = "https://s.zhe.to/apps/otter";
@@ -23,6 +37,13 @@ export function createDefaultCollectors(homeDir = homedir(), options = {}) {
23
37
  new ShellConfigCollector(homeDir),
24
38
  new HomebrewCollector(homeDir),
25
39
  new ApplicationsCollector(homeDir, "/Applications", iconBaseUrl),
40
+ new VSCodeCollector(homeDir),
41
+ new DockerCollector(homeDir),
42
+ new FontsCollector(homeDir),
43
+ new DevToolchainCollector(homeDir),
44
+ new CloudCLICollector(homeDir),
45
+ new MacOSDefaultsCollector(homeDir),
46
+ new LaunchAgentsCollector(homeDir),
26
47
  ];
27
48
  }
28
49
  //# sourceMappingURL=index.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/collectors/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAC3D,OAAO,EAAE,uBAAuB,EAAE,MAAM,sBAAsB,CAAC;AAC/D,OAAO,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAC;AACzD,OAAO,EAAE,iBAAiB,EAAE,MAAM,eAAe,CAAC;AAClD,OAAO,EAAE,qBAAqB,EAAE,MAAM,mBAAmB,CAAC;AAG1D,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAC3D,OAAO,EAAE,uBAAuB,EAAE,MAAM,sBAAsB,CAAC;AAC/D,OAAO,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAC;AACzD,OAAO,EAAE,iBAAiB,EAAE,MAAM,eAAe,CAAC;AAClD,OAAO,EAAE,qBAAqB,EAAE,MAAM,mBAAmB,CAAC;AAC1D,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAElC,qDAAqD;AACrD,MAAM,qBAAqB,GAAG,6BAA6B,CAAC;AAY5D;;GAEG;AACH,MAAM,UAAU,uBAAuB,CACrC,UAAkB,OAAO,EAAE,EAC3B,UAA4B,EAAE;IAE9B,MAAM,WAAW,GAAG,OAAO,CAAC,WAAW,IAAI,qBAAqB,CAAC;IACjE,OAAO;QACL,IAAI,qBAAqB,CAAC,OAAO,EAAE,EAAE,IAAI,EAAE,OAAO,CAAC,IAAI,EAAE,CAAC;QAC1D,IAAI,uBAAuB,CAAC,OAAO,CAAC;QACpC,IAAI,oBAAoB,CAAC,OAAO,CAAC;QACjC,IAAI,iBAAiB,CAAC,OAAO,CAAC;QAC9B,IAAI,qBAAqB,CAAC,OAAO,EAAE,eAAe,EAAE,WAAW,CAAC;KACjE,CAAC;AACJ,CAAC"}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/collectors/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAC3D,OAAO,EAAE,uBAAuB,EAAE,MAAM,sBAAsB,CAAC;AAC/D,OAAO,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAC;AACzD,OAAO,EAAE,iBAAiB,EAAE,MAAM,eAAe,CAAC;AAClD,OAAO,EAAE,qBAAqB,EAAE,MAAM,mBAAmB,CAAC;AAC1D,OAAO,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;AAC9C,OAAO,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;AAC9C,OAAO,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAC5C,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAC3D,OAAO,EAAE,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AACnD,OAAO,EAAE,sBAAsB,EAAE,MAAM,qBAAqB,CAAC;AAC7D,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAG3D,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAC3D,OAAO,EAAE,uBAAuB,EAAE,MAAM,sBAAsB,CAAC;AAC/D,OAAO,EAAE,oBAAoB,EAAE,MAAM,mBAAmB,CAAC;AACzD,OAAO,EAAE,iBAAiB,EAAE,MAAM,eAAe,CAAC;AAClD,OAAO,EAAE,qBAAqB,EAAE,MAAM,mBAAmB,CAAC;AAC1D,OAAO,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;AAC9C,OAAO,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;AAC9C,OAAO,EAAE,cAAc,EAAE,MAAM,YAAY,CAAC;AAC5C,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAC3D,OAAO,EAAE,iBAAiB,EAAE,MAAM,gBAAgB,CAAC;AACnD,OAAO,EAAE,sBAAsB,EAAE,MAAM,qBAAqB,CAAC;AAC7D,OAAO,EAAE,qBAAqB,EAAE,MAAM,oBAAoB,CAAC;AAC3D,OAAO,EAAE,OAAO,EAAE,MAAM,SAAS,CAAC;AAElC,qDAAqD;AACrD,MAAM,qBAAqB,GAAG,6BAA6B,CAAC;AAY5D;;GAEG;AACH,MAAM,UAAU,uBAAuB,CACrC,UAAkB,OAAO,EAAE,EAC3B,UAA4B,EAAE;IAE9B,MAAM,WAAW,GAAG,OAAO,CAAC,WAAW,IAAI,qBAAqB,CAAC;IACjE,OAAO;QACL,IAAI,qBAAqB,CAAC,OAAO,EAAE,EAAE,IAAI,EAAE,OAAO,CAAC,IAAI,EAAE,CAAC;QAC1D,IAAI,uBAAuB,CAAC,OAAO,CAAC;QACpC,IAAI,oBAAoB,CAAC,OAAO,CAAC;QACjC,IAAI,iBAAiB,CAAC,OAAO,CAAC;QAC9B,IAAI,qBAAqB,CAAC,OAAO,EAAE,eAAe,EAAE,WAAW,CAAC;QAChE,IAAI,eAAe,CAAC,OAAO,CAAC;QAC5B,IAAI,eAAe,CAAC,OAAO,CAAC;QAC5B,IAAI,cAAc,CAAC,OAAO,CAAC;QAC3B,IAAI,qBAAqB,CAAC,OAAO,CAAC;QAClC,IAAI,iBAAiB,CAAC,OAAO,CAAC;QAC9B,IAAI,sBAAsB,CAAC,OAAO,CAAC;QACnC,IAAI,qBAAqB,CAAC,OAAO,CAAC;KACnC,CAAC;AACJ,CAAC"}
@@ -0,0 +1,10 @@
1
+ import { BaseCollector } from "./base.js";
2
+ import type { CollectorCategory, CollectorResult } from "@otter/core";
3
+ export declare class LaunchAgentsCollector extends BaseCollector {
4
+ readonly id = "launch-agents";
5
+ readonly label = "Launch Agents & Daemons";
6
+ readonly category: CollectorCategory;
7
+ _execCommand: (cmd: string) => Promise<string>;
8
+ collect(): Promise<CollectorResult>;
9
+ }
10
+ //# sourceMappingURL=launch-agents.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"launch-agents.d.ts","sourceRoot":"","sources":["../../src/collectors/launch-agents.ts"],"names":[],"mappings":"AAIA,OAAO,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,KAAK,EACV,iBAAiB,EACjB,eAAe,EAEhB,MAAM,aAAa,CAAC;AAKrB,qBAAa,qBAAsB,SAAQ,aAAa;IACtD,QAAQ,CAAC,EAAE,mBAAmB;IAC9B,QAAQ,CAAC,KAAK,6BAA6B;IAC3C,QAAQ,CAAC,QAAQ,EAAE,iBAAiB,CAAiB;IAErD,YAAY,GAAU,KAAK,MAAM,KAAG,OAAO,CAAC,MAAM,CAAC,CAGjD;IAEI,OAAO,IAAI,OAAO,CAAC,eAAe,CAAC;CAuC1C"}
@@ -0,0 +1,53 @@
1
+ import { exec } from "node:child_process";
2
+ import { readdir } from "node:fs/promises";
3
+ import { join } from "node:path";
4
+ import { promisify } from "node:util";
5
+ import { BaseCollector } from "./base.js";
6
+ import { redactSecrets } from "../utils/redact.js";
7
+ const execAsync = promisify(exec);
8
+ export class LaunchAgentsCollector extends BaseCollector {
9
+ id = "launch-agents";
10
+ label = "Launch Agents & Daemons";
11
+ category = "environment";
12
+ _execCommand = async (cmd) => {
13
+ const { stdout } = await execAsync(cmd);
14
+ return stdout;
15
+ };
16
+ async collect() {
17
+ return this.timed(async (result) => {
18
+ const agentsDir = join(this.homeDir, "Library", "LaunchAgents");
19
+ try {
20
+ const entries = await readdir(agentsDir, { withFileTypes: true });
21
+ result.lists.push(...entries
22
+ .filter((entry) => entry.isFile() && entry.name.endsWith(".plist"))
23
+ .map((entry) => ({
24
+ name: entry.name,
25
+ meta: { type: "user-agent" },
26
+ }))
27
+ .sort((a, b) => a.name.localeCompare(b.name)));
28
+ }
29
+ catch (err) {
30
+ if (err.code !== "ENOENT") {
31
+ result.errors.push(`Failed to read launch agents: ${err.message}`);
32
+ }
33
+ }
34
+ try {
35
+ const output = await this._execCommand("crontab -l");
36
+ const trimmed = output.trim();
37
+ if (trimmed.length > 0) {
38
+ const content = redactSecrets(trimmed, ".env");
39
+ const file = {
40
+ path: "crontab",
41
+ content,
42
+ sizeBytes: Buffer.byteLength(content, "utf-8"),
43
+ };
44
+ result.files.push(file);
45
+ }
46
+ }
47
+ catch {
48
+ // no crontab configured is fine
49
+ }
50
+ });
51
+ }
52
+ }
53
+ //# sourceMappingURL=launch-agents.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"launch-agents.js","sourceRoot":"","sources":["../../src/collectors/launch-agents.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,oBAAoB,CAAC;AAC1C,OAAO,EAAE,OAAO,EAAE,MAAM,kBAAkB,CAAC;AAC3C,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AACtC,OAAO,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAM1C,OAAO,EAAE,aAAa,EAAE,MAAM,oBAAoB,CAAC;AAEnD,MAAM,SAAS,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;AAElC,MAAM,OAAO,qBAAsB,SAAQ,aAAa;IAC7C,EAAE,GAAG,eAAe,CAAC;IACrB,KAAK,GAAG,yBAAyB,CAAC;IAClC,QAAQ,GAAsB,aAAa,CAAC;IAErD,YAAY,GAAG,KAAK,EAAE,GAAW,EAAmB,EAAE;QACpD,MAAM,EAAE,MAAM,EAAE,GAAG,MAAM,SAAS,CAAC,GAAG,CAAC,CAAC;QACxC,OAAO,MAAM,CAAC;IAChB,CAAC,CAAC;IAEF,KAAK,CAAC,OAAO;QACX,OAAO,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,MAAM,EAAE,EAAE;YACjC,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE,SAAS,EAAE,cAAc,CAAC,CAAC;YAChE,IAAI,CAAC;gBACH,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,SAAS,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;gBAClE,MAAM,CAAC,KAAK,CAAC,IAAI,CACf,GAAG,OAAO;qBACP,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,MAAM,EAAE,IAAI,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;qBAClE,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;oBACf,IAAI,EAAE,KAAK,CAAC,IAAI;oBAChB,IAAI,EAAE,EAAE,IAAI,EAAE,YAAY,EAAE;iBAC7B,CAAC,CAAC;qBACF,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAChD,CAAC;YACJ,CAAC;YAAC,OAAO,GAAG,EAAE,CAAC;gBACb,IAAK,GAA6B,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;oBACrD,MAAM,CAAC,MAAM,CAAC,IAAI,CAChB,iCAAkC,GAAa,CAAC,OAAO,EAAE,CAC1D,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,IAAI,CAAC;gBACH,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,YAAY,CAAC,CAAC;gBACrD,MAAM,OAAO,GAAG,MAAM,CAAC,IAAI,EAAE,CAAC;gBAC9B,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBACvB,MAAM,OAAO,GAAG,aAAa,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;oBAC/C,MAAM,IAAI,GAAkB;wBAC1B,IAAI,EAAE,SAAS;wBACf,OAAO;wBACP,SAAS,EAAE,MAAM,CAAC,UAAU,CAAC,OAAO,EAAE,OAAO,CAAC;qBAC/C,CAAC;oBACF,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBAC1B,CAAC;YACH,CAAC;YAAC,MAAM,CAAC;gBACP,gCAAgC;YAClC,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;CACF"}
@@ -0,0 +1,10 @@
1
+ import { BaseCollector } from "./base.js";
2
+ import type { CollectorCategory, CollectorResult } from "@otter/core";
3
+ export declare class MacOSDefaultsCollector extends BaseCollector {
4
+ readonly id = "macos-defaults";
5
+ readonly label = "macOS System Preferences";
6
+ readonly category: CollectorCategory;
7
+ _execCommand: (cmd: string) => Promise<string>;
8
+ collect(): Promise<CollectorResult>;
9
+ }
10
+ //# sourceMappingURL=macos-defaults.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"macos-defaults.d.ts","sourceRoot":"","sources":["../../src/collectors/macos-defaults.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,KAAK,EACV,iBAAiB,EACjB,eAAe,EAEhB,MAAM,aAAa,CAAC;AAcrB,qBAAa,sBAAuB,SAAQ,aAAa;IACvD,QAAQ,CAAC,EAAE,oBAAoB;IAC/B,QAAQ,CAAC,KAAK,8BAA8B;IAC5C,QAAQ,CAAC,QAAQ,EAAE,iBAAiB,CAAiB;IAErD,YAAY,GAAU,KAAK,MAAM,KAAG,OAAO,CAAC,MAAM,CAAC,CAGjD;IAEI,OAAO,IAAI,OAAO,CAAC,eAAe,CAAC;CAiC1C"}
@@ -0,0 +1,53 @@
1
+ import { exec } from "node:child_process";
2
+ import { promisify } from "node:util";
3
+ import { BaseCollector } from "./base.js";
4
+ const execAsync = promisify(exec);
5
+ const DOMAINS = [
6
+ "com.apple.dock",
7
+ "com.apple.finder",
8
+ "com.apple.AppleMultitouchTrackpad",
9
+ "com.apple.driver.AppleBluetoothMultitouch.trackpad",
10
+ "NSGlobalDomain",
11
+ "com.apple.symbolichotkeys",
12
+ "com.apple.screencapture",
13
+ ];
14
+ export class MacOSDefaultsCollector extends BaseCollector {
15
+ id = "macos-defaults";
16
+ label = "macOS System Preferences";
17
+ category = "environment";
18
+ _execCommand = async (cmd) => {
19
+ const { stdout } = await execAsync(cmd);
20
+ return stdout;
21
+ };
22
+ async collect() {
23
+ return this.timed(async (result) => {
24
+ for (const domain of DOMAINS) {
25
+ try {
26
+ const content = await this._execCommand(`defaults export ${domain} -`);
27
+ const file = {
28
+ path: `macos-defaults/${domain}.plist`,
29
+ content,
30
+ sizeBytes: Buffer.byteLength(content, "utf-8"),
31
+ };
32
+ result.files.push(file);
33
+ }
34
+ catch (err) {
35
+ result.errors.push(`Failed to export defaults domain ${domain}: ${err.message}`);
36
+ }
37
+ }
38
+ try {
39
+ const output = await this._execCommand("osascript -e 'tell application \"System Events\" to get the name of every login item'");
40
+ const items = output
41
+ .split(",")
42
+ .map((name) => name.trim())
43
+ .filter((name) => name.length > 0)
44
+ .map((name) => ({ name, meta: { type: "login-item" } }));
45
+ result.lists.push(...items);
46
+ }
47
+ catch {
48
+ // headless environments may fail; ignore
49
+ }
50
+ });
51
+ }
52
+ }
53
+ //# sourceMappingURL=macos-defaults.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"macos-defaults.js","sourceRoot":"","sources":["../../src/collectors/macos-defaults.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,oBAAoB,CAAC;AAC1C,OAAO,EAAE,SAAS,EAAE,MAAM,WAAW,CAAC;AACtC,OAAO,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAO1C,MAAM,SAAS,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;AAElC,MAAM,OAAO,GAAG;IACd,gBAAgB;IAChB,kBAAkB;IAClB,mCAAmC;IACnC,oDAAoD;IACpD,gBAAgB;IAChB,2BAA2B;IAC3B,yBAAyB;CAC1B,CAAC;AAEF,MAAM,OAAO,sBAAuB,SAAQ,aAAa;IAC9C,EAAE,GAAG,gBAAgB,CAAC;IACtB,KAAK,GAAG,0BAA0B,CAAC;IACnC,QAAQ,GAAsB,aAAa,CAAC;IAErD,YAAY,GAAG,KAAK,EAAE,GAAW,EAAmB,EAAE;QACpD,MAAM,EAAE,MAAM,EAAE,GAAG,MAAM,SAAS,CAAC,GAAG,CAAC,CAAC;QACxC,OAAO,MAAM,CAAC;IAChB,CAAC,CAAC;IAEF,KAAK,CAAC,OAAO;QACX,OAAO,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,MAAM,EAAE,EAAE;YACjC,KAAK,MAAM,MAAM,IAAI,OAAO,EAAE,CAAC;gBAC7B,IAAI,CAAC;oBACH,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,mBAAmB,MAAM,IAAI,CAAC,CAAC;oBACvE,MAAM,IAAI,GAAkB;wBAC1B,IAAI,EAAE,kBAAkB,MAAM,QAAQ;wBACtC,OAAO;wBACP,SAAS,EAAE,MAAM,CAAC,UAAU,CAAC,OAAO,EAAE,OAAO,CAAC;qBAC/C,CAAC;oBACF,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBAC1B,CAAC;gBAAC,OAAO,GAAG,EAAE,CAAC;oBACb,MAAM,CAAC,MAAM,CAAC,IAAI,CAChB,oCAAoC,MAAM,KAAM,GAAa,CAAC,OAAO,EAAE,CACxE,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,IAAI,CAAC;gBACH,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,YAAY,CACpC,uFAAuF,CACxF,CAAC;gBACF,MAAM,KAAK,GAAG,MAAM;qBACjB,KAAK,CAAC,GAAG,CAAC;qBACV,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;qBAC1B,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC;qBACjC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,EAAE,IAAI,EAAE,IAAI,EAAE,EAAE,IAAI,EAAE,YAAY,EAAE,EAAE,CAAC,CAAC,CAAC;gBAC3D,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,KAAK,CAAC,CAAC;YAC9B,CAAC;YAAC,MAAM,CAAC;gBACP,yCAAyC;YAC3C,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;CACF"}
@@ -0,0 +1,15 @@
1
+ import { BaseCollector } from "./base.js";
2
+ import type { CollectorCategory, CollectorResult } from "@otter/core";
3
+ export declare class VSCodeCollector extends BaseCollector {
4
+ readonly id = "vscode";
5
+ readonly label = "VS Code / Cursor Configuration";
6
+ readonly category: CollectorCategory;
7
+ private readonly editors;
8
+ constructor(homeDir: string);
9
+ _execCommand: (cmd: string) => Promise<string>;
10
+ collect(): Promise<CollectorResult>;
11
+ private collectExtensions;
12
+ private collectExtensionsFromDir;
13
+ private collectEditorFiles;
14
+ }
15
+ //# sourceMappingURL=vscode.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"vscode.d.ts","sourceRoot":"","sources":["../../src/collectors/vscode.ts"],"names":[],"mappings":"AAIA,OAAO,EAAE,aAAa,EAAE,MAAM,WAAW,CAAC;AAC1C,OAAO,KAAK,EACV,iBAAiB,EACjB,eAAe,EAGhB,MAAM,aAAa,CAAC;AA6CrB,qBAAa,eAAgB,SAAQ,aAAa;IAChD,QAAQ,CAAC,EAAE,YAAY;IACvB,QAAQ,CAAC,KAAK,oCAAoC;IAClD,QAAQ,CAAC,QAAQ,EAAE,iBAAiB,CAAY;IAEhD,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAiB;gBAE7B,OAAO,EAAE,MAAM;IAkB3B,YAAY,GAAU,KAAK,MAAM,KAAG,OAAO,CAAC,MAAM,CAAC,CAGjD;IAEI,OAAO,IAAI,OAAO,CAAC,eAAe,CAAC;YAY3B,iBAAiB;YAcjB,wBAAwB;YAyBxB,kBAAkB;CA4BjC"}
@@ -0,0 +1,118 @@
1
+ import { exec } from "node:child_process";
2
+ import { readdir } from "node:fs/promises";
3
+ import { join } from "node:path";
4
+ import { promisify } from "node:util";
5
+ import { BaseCollector } from "./base.js";
6
+ const execAsync = promisify(exec);
7
+ function parseCliExtensions(output, editor) {
8
+ return output
9
+ .split("\n")
10
+ .map((line) => line.trim())
11
+ .filter((line) => line.length > 0)
12
+ .map((line) => {
13
+ const atIndex = line.lastIndexOf("@");
14
+ if (atIndex === -1) {
15
+ return {
16
+ name: line,
17
+ meta: { type: "vscode-extension", editor },
18
+ };
19
+ }
20
+ return {
21
+ name: line.slice(0, atIndex),
22
+ version: line.slice(atIndex + 1),
23
+ meta: { type: "vscode-extension", editor },
24
+ };
25
+ });
26
+ }
27
+ function parseExtensionDirName(name) {
28
+ const match = name.match(/^(.*)-([0-9][A-Za-z0-9.+_-]*)$/);
29
+ if (!match) {
30
+ return name.length > 0 ? { name } : null;
31
+ }
32
+ return {
33
+ name: match[1],
34
+ version: match[2],
35
+ };
36
+ }
37
+ export class VSCodeCollector extends BaseCollector {
38
+ id = "vscode";
39
+ label = "VS Code / Cursor Configuration";
40
+ category = "config";
41
+ editors;
42
+ constructor(homeDir) {
43
+ super(homeDir);
44
+ this.editors = [
45
+ {
46
+ editor: "vscode",
47
+ cli: "code",
48
+ extensionsDir: join(homeDir, ".vscode", "extensions"),
49
+ userDir: join(homeDir, "Library", "Application Support", "Code", "User"),
50
+ },
51
+ {
52
+ editor: "cursor",
53
+ cli: "cursor",
54
+ extensionsDir: join(homeDir, ".cursor", "extensions"),
55
+ userDir: join(homeDir, "Library", "Application Support", "Cursor", "User"),
56
+ },
57
+ ];
58
+ }
59
+ _execCommand = async (cmd) => {
60
+ const { stdout } = await execAsync(cmd);
61
+ return stdout;
62
+ };
63
+ async collect() {
64
+ return this.timed(async (result) => {
65
+ for (const editor of this.editors) {
66
+ const items = await this.collectExtensions(editor, result);
67
+ result.lists.push(...items);
68
+ const files = await this.collectEditorFiles(editor, result);
69
+ result.files.push(...files);
70
+ }
71
+ });
72
+ }
73
+ async collectExtensions(editor, result) {
74
+ try {
75
+ const output = await this._execCommand(`${editor.cli} --list-extensions --show-versions`);
76
+ return parseCliExtensions(output, editor.editor);
77
+ }
78
+ catch {
79
+ return this.collectExtensionsFromDir(editor, result);
80
+ }
81
+ }
82
+ async collectExtensionsFromDir(editor, result) {
83
+ try {
84
+ const entries = await readdir(editor.extensionsDir, { withFileTypes: true });
85
+ return entries
86
+ .filter((entry) => entry.isDirectory())
87
+ .map((entry) => parseExtensionDirName(entry.name))
88
+ .filter((item) => item !== null)
89
+ .map((item) => ({
90
+ ...item,
91
+ meta: { type: "vscode-extension", editor: editor.editor },
92
+ }))
93
+ .sort((a, b) => a.name.localeCompare(b.name));
94
+ }
95
+ catch (err) {
96
+ if (err.code !== "ENOENT") {
97
+ result.errors.push(`Failed to read ${editor.editor} extensions: ${err.message}`);
98
+ }
99
+ return [];
100
+ }
101
+ }
102
+ async collectEditorFiles(editor, result) {
103
+ const files = [];
104
+ const settings = await this.safeReadFile(join(editor.userDir, "settings.json"), result, { redact: true });
105
+ if (settings)
106
+ files.push(settings);
107
+ const keybindings = await this.safeReadFile(join(editor.userDir, "keybindings.json"), result);
108
+ if (keybindings)
109
+ files.push(keybindings);
110
+ const snippets = await this.collectDir(join(editor.userDir, "snippets"), result, {
111
+ maxFileSize: 128 * 1024,
112
+ filter: (filePath) => filePath.endsWith(".json") || filePath.endsWith(".code-snippets"),
113
+ });
114
+ files.push(...snippets);
115
+ return files;
116
+ }
117
+ }
118
+ //# sourceMappingURL=vscode.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"vscode.js","sourceRoot":"","sources":["../../src/collectors/vscode.ts"],"names":[],"mappings":"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"}
@@ -1,5 +1,7 @@
1
1
  import type { Collector, CollectorResult, Snapshot } from "@otter/core";
2
2
  export interface ScanOptions {
3
+ /** Called when a collector is about to start */
4
+ onStart?: (collectorId: string, label: string) => void;
3
5
  /** Called after each collector finishes */
4
6
  onProgress?: (collectorId: string, result: CollectorResult) => void;
5
7
  }
@@ -1 +1 @@
1
- {"version":3,"file":"scan.d.ts","sourceRoot":"","sources":["../../src/commands/scan.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,eAAe,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAGxE,MAAM,WAAW,WAAW;IAC1B,2CAA2C;IAC3C,UAAU,CAAC,EAAE,CAAC,WAAW,EAAE,MAAM,EAAE,MAAM,EAAE,eAAe,KAAK,IAAI,CAAC;CACrE;AAED;;;GAGG;AACH,wBAAsB,WAAW,CAC/B,UAAU,EAAE,SAAS,EAAE,EACvB,OAAO,GAAE,WAAgB,GACxB,OAAO,CAAC,QAAQ,CAAC,CA6BnB"}
1
+ {"version":3,"file":"scan.d.ts","sourceRoot":"","sources":["../../src/commands/scan.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,eAAe,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAGxE,MAAM,WAAW,WAAW;IAC1B,gDAAgD;IAChD,OAAO,CAAC,EAAE,CAAC,WAAW,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,KAAK,IAAI,CAAC;IACvD,2CAA2C;IAC3C,UAAU,CAAC,EAAE,CAAC,WAAW,EAAE,MAAM,EAAE,MAAM,EAAE,eAAe,KAAK,IAAI,CAAC;CACrE;AAED;;;GAGG;AACH,wBAAsB,WAAW,CAC/B,UAAU,EAAE,SAAS,EAAE,EACvB,OAAO,GAAE,WAAgB,GACxB,OAAO,CAAC,QAAQ,CAAC,CA+BnB"}
@@ -7,6 +7,7 @@ export async function executeScan(collectors, options = {}) {
7
7
  // Run collectors sequentially to allow progress reporting
8
8
  const results = [];
9
9
  for (const collector of collectors) {
10
+ options.onStart?.(collector.id, collector.label);
10
11
  let result;
11
12
  try {
12
13
  result = await collector.collect();
@@ -21,6 +22,7 @@ export async function executeScan(collectors, options = {}) {
21
22
  errors: [
22
23
  `Collector '${collector.id}' crashed: ${err.message}`,
23
24
  ],
25
+ skipped: [],
24
26
  durationMs: 0,
25
27
  };
26
28
  }
@@ -1 +1 @@
1
- {"version":3,"file":"scan.js","sourceRoot":"","sources":["../../src/commands/scan.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AAOvD;;;GAGG;AACH,MAAM,CAAC,KAAK,UAAU,WAAW,CAC/B,UAAuB,EACvB,UAAuB,EAAE;IAEzB,0DAA0D;IAC1D,MAAM,OAAO,GAAsB,EAAE,CAAC;IACtC,KAAK,MAAM,SAAS,IAAI,UAAU,EAAE,CAAC;QACnC,IAAI,MAAuB,CAAC;QAC5B,IAAI,CAAC;YACH,MAAM,GAAG,MAAM,SAAS,CAAC,OAAO,EAAE,CAAC;QACrC,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,GAAG;gBACP,EAAE,EAAE,SAAS,CAAC,EAAE;gBAChB,KAAK,EAAE,SAAS,CAAC,KAAK;gBACtB,QAAQ,EAAE,SAAS,CAAC,QAAQ;gBAC5B,KAAK,EAAE,EAAE;gBACT,KAAK,EAAE,EAAE;gBACT,MAAM,EAAE;oBACN,cAAc,SAAS,CAAC,EAAE,cAAe,GAAa,CAAC,OAAO,EAAE;iBACjE;gBACD,UAAU,EAAE,CAAC;aACd,CAAC;QACJ,CAAC;QACD,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACrB,OAAO,CAAC,UAAU,EAAE,CAAC,SAAS,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC;IAC7C,CAAC;IAED,4CAA4C;IAC5C,0EAA0E;IAC1E,MAAM,QAAQ,GAAG,MAAM,aAAa,CAAC,EAAE,CAAC,CAAC;IACzC,QAAQ,CAAC,UAAU,GAAG,OAAO,CAAC;IAC9B,OAAO,QAAQ,CAAC;AAClB,CAAC"}
1
+ {"version":3,"file":"scan.js","sourceRoot":"","sources":["../../src/commands/scan.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AASvD;;;GAGG;AACH,MAAM,CAAC,KAAK,UAAU,WAAW,CAC/B,UAAuB,EACvB,UAAuB,EAAE;IAEzB,0DAA0D;IAC1D,MAAM,OAAO,GAAsB,EAAE,CAAC;IACtC,KAAK,MAAM,SAAS,IAAI,UAAU,EAAE,CAAC;QACnC,OAAO,CAAC,OAAO,EAAE,CAAC,SAAS,CAAC,EAAE,EAAE,SAAS,CAAC,KAAK,CAAC,CAAC;QACjD,IAAI,MAAuB,CAAC;QAC5B,IAAI,CAAC;YACH,MAAM,GAAG,MAAM,SAAS,CAAC,OAAO,EAAE,CAAC;QACrC,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,MAAM,GAAG;gBACP,EAAE,EAAE,SAAS,CAAC,EAAE;gBAChB,KAAK,EAAE,SAAS,CAAC,KAAK;gBACtB,QAAQ,EAAE,SAAS,CAAC,QAAQ;gBAC5B,KAAK,EAAE,EAAE;gBACT,KAAK,EAAE,EAAE;gBACT,MAAM,EAAE;oBACN,cAAc,SAAS,CAAC,EAAE,cAAe,GAAa,CAAC,OAAO,EAAE;iBACjE;gBACD,OAAO,EAAE,EAAE;gBACX,UAAU,EAAE,CAAC;aACd,CAAC;QACJ,CAAC;QACD,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QACrB,OAAO,CAAC,UAAU,EAAE,CAAC,SAAS,CAAC,EAAE,EAAE,MAAM,CAAC,CAAC;IAC7C,CAAC;IAED,4CAA4C;IAC5C,0EAA0E;IAC1E,MAAM,QAAQ,GAAG,MAAM,aAAa,CAAC,EAAE,CAAC,CAAC;IACzC,QAAQ,CAAC,UAAU,GAAG,OAAO,CAAC;IAC9B,OAAO,QAAQ,CAAC;AAClB,CAAC"}
@@ -1,13 +1,6 @@
1
1
  import type { Snapshot } from "@otter/core";
2
2
  import type { SnapshotMeta } from "../storage/local.js";
3
- /**
4
- * Format a file size in bytes to a human-readable string.
5
- */
6
- declare function formatSize(bytes: number): string;
7
- /**
8
- * Format an ISO 8601 date string to a short "YYYY-MM-DD HH:MM" format.
9
- */
10
- declare function formatDate(iso: string): string;
3
+ import { formatSize, formatDate } from "../ui.js";
11
4
  /**
12
5
  * Format the snapshot list output for the terminal.
13
6
  * Pure logic function, returns lines to print.
@@ -45,7 +38,7 @@ export interface SnapshotDiffResult {
45
38
  */
46
39
  export declare function diffSnapshots(oldSnap: Snapshot, newSnap: Snapshot): SnapshotDiffResult;
47
40
  /**
48
- * Format a diff result for terminal output.
41
+ * Format a diff result for terminal output using tree views.
49
42
  */
50
43
  export declare function formatSnapshotDiff(diff: SnapshotDiffResult): string;
51
44
  export { formatSize, formatDate };
@@ -1 +1 @@
1
- {"version":3,"file":"snapshot.d.ts","sourceRoot":"","sources":["../../src/commands/snapshot.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAC5C,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AAExD;;GAEG;AACH,iBAAS,UAAU,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,CAQzC;AAED;;GAEG;AACH,iBAAS,UAAU,CAAC,GAAG,EAAE,MAAM,GAAG,MAAM,CAIvC;AAED;;;GAGG;AACH,wBAAgB,kBAAkB,CAAC,KAAK,EAAE,YAAY,EAAE,GAAG,MAAM,CAiBhE;AAMD;;GAEG;AACH,wBAAgB,oBAAoB,CAAC,QAAQ,EAAE,QAAQ,GAAG,MAAM,CAwC/D;AAMD,wEAAwE;AACxE,MAAM,WAAW,SAAS;IACxB,IAAI,EAAE,OAAO,GAAG,SAAS,GAAG,SAAS,CAAC;IACtC,KAAK,EAAE,MAAM,CAAC;CACf;AAED,yCAAyC;AACzC,MAAM,WAAW,aAAa;IAC5B,WAAW,EAAE,MAAM,CAAC;IACpB,cAAc,EAAE,MAAM,CAAC;IACvB,KAAK,EAAE,SAAS,EAAE,CAAC;IACnB,KAAK,EAAE,SAAS,EAAE,CAAC;CACpB;AAED,6CAA6C;AAC7C,MAAM,WAAW,kBAAkB;IACjC,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,MAAM,CAAC;IACd,eAAe,EAAE,MAAM,EAAE,CAAC;IAC1B,iBAAiB,EAAE,MAAM,EAAE,CAAC;IAC5B,UAAU,EAAE,aAAa,EAAE,CAAC;CAC7B;AAED;;;;;GAKG;AACH,wBAAgB,aAAa,CAC3B,OAAO,EAAE,QAAQ,EACjB,OAAO,EAAE,QAAQ,GAChB,kBAAkB,CA4CpB;AAmDD;;GAEG;AACH,wBAAgB,kBAAkB,CAAC,IAAI,EAAE,kBAAkB,GAAG,MAAM,CAgDnE;AAED,OAAO,EAAE,UAAU,EAAE,UAAU,EAAE,CAAC"}
1
+ {"version":3,"file":"snapshot.d.ts","sourceRoot":"","sources":["../../src/commands/snapshot.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAC5C,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,qBAAqB,CAAC;AACxD,OAAO,EACL,UAAU,EACV,UAAU,EAMX,MAAM,UAAU,CAAC;AAIlB;;;GAGG;AACH,wBAAgB,kBAAkB,CAAC,KAAK,EAAE,YAAY,EAAE,GAAG,MAAM,CAwBhE;AAID;;GAEG;AACH,wBAAgB,oBAAoB,CAAC,QAAQ,EAAE,QAAQ,GAAG,MAAM,CA4D/D;AAID,wEAAwE;AACxE,MAAM,WAAW,SAAS;IACxB,IAAI,EAAE,OAAO,GAAG,SAAS,GAAG,SAAS,CAAC;IACtC,KAAK,EAAE,MAAM,CAAC;CACf;AAED,yCAAyC;AACzC,MAAM,WAAW,aAAa;IAC5B,WAAW,EAAE,MAAM,CAAC;IACpB,cAAc,EAAE,MAAM,CAAC;IACvB,KAAK,EAAE,SAAS,EAAE,CAAC;IACnB,KAAK,EAAE,SAAS,EAAE,CAAC;CACpB;AAED,6CAA6C;AAC7C,MAAM,WAAW,kBAAkB;IACjC,KAAK,EAAE,MAAM,CAAC;IACd,KAAK,EAAE,MAAM,CAAC;IACd,eAAe,EAAE,MAAM,EAAE,CAAC;IAC1B,iBAAiB,EAAE,MAAM,EAAE,CAAC;IAC5B,UAAU,EAAE,aAAa,EAAE,CAAC;CAC7B;AAED;;;;;GAKG;AACH,wBAAgB,aAAa,CAC3B,OAAO,EAAE,QAAQ,EACjB,OAAO,EAAE,QAAQ,GAChB,kBAAkB,CA+CpB;AAmDD;;GAEG;AACH,wBAAgB,kBAAkB,CAAC,IAAI,EAAE,kBAAkB,GAAG,MAAM,CAiEnE;AAED,OAAO,EAAE,UAAU,EAAE,UAAU,EAAE,CAAC"}