@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.
- package/dist/cli.d.ts.map +1 -1
- package/dist/cli.js +162 -85
- package/dist/cli.js.map +1 -1
- package/dist/collectors/applications.d.ts +7 -2
- package/dist/collectors/applications.d.ts.map +1 -1
- package/dist/collectors/applications.js +55 -21
- package/dist/collectors/applications.js.map +1 -1
- package/dist/collectors/base.d.ts.map +1 -1
- package/dist/collectors/base.js +1 -0
- package/dist/collectors/base.js.map +1 -1
- package/dist/collectors/cloud-cli.d.ts +9 -0
- package/dist/collectors/cloud-cli.d.ts.map +1 -0
- package/dist/collectors/cloud-cli.js +61 -0
- package/dist/collectors/cloud-cli.js.map +1 -0
- package/dist/collectors/dev-toolchain.d.ts +21 -0
- package/dist/collectors/dev-toolchain.d.ts.map +1 -0
- package/dist/collectors/dev-toolchain.js +252 -0
- package/dist/collectors/dev-toolchain.js.map +1 -0
- package/dist/collectors/docker.d.ts +10 -0
- package/dist/collectors/docker.d.ts.map +1 -0
- package/dist/collectors/docker.js +57 -0
- package/dist/collectors/docker.js.map +1 -0
- package/dist/collectors/fonts.d.ts +9 -0
- package/dist/collectors/fonts.d.ts.map +1 -0
- package/dist/collectors/fonts.js +36 -0
- package/dist/collectors/fonts.js.map +1 -0
- package/dist/collectors/homebrew.d.ts +2 -1
- package/dist/collectors/homebrew.d.ts.map +1 -1
- package/dist/collectors/homebrew.js +56 -12
- package/dist/collectors/homebrew.js.map +1 -1
- package/dist/collectors/index.d.ts +7 -0
- package/dist/collectors/index.d.ts.map +1 -1
- package/dist/collectors/index.js +21 -0
- package/dist/collectors/index.js.map +1 -1
- package/dist/collectors/launch-agents.d.ts +10 -0
- package/dist/collectors/launch-agents.d.ts.map +1 -0
- package/dist/collectors/launch-agents.js +53 -0
- package/dist/collectors/launch-agents.js.map +1 -0
- package/dist/collectors/macos-defaults.d.ts +10 -0
- package/dist/collectors/macos-defaults.d.ts.map +1 -0
- package/dist/collectors/macos-defaults.js +53 -0
- package/dist/collectors/macos-defaults.js.map +1 -0
- package/dist/collectors/vscode.d.ts +15 -0
- package/dist/collectors/vscode.d.ts.map +1 -0
- package/dist/collectors/vscode.js +118 -0
- package/dist/collectors/vscode.js.map +1 -0
- package/dist/commands/scan.d.ts +2 -0
- package/dist/commands/scan.d.ts.map +1 -1
- package/dist/commands/scan.js +2 -0
- package/dist/commands/scan.js.map +1 -1
- package/dist/commands/snapshot.d.ts +2 -9
- package/dist/commands/snapshot.d.ts.map +1 -1
- package/dist/commands/snapshot.js +74 -64
- package/dist/commands/snapshot.js.map +1 -1
- package/dist/snapshot/builder.d.ts.map +1 -1
- package/dist/snapshot/builder.js +1 -0
- package/dist/snapshot/builder.js.map +1 -1
- package/dist/ui.d.ts +63 -0
- package/dist/ui.d.ts.map +1 -0
- package/dist/ui.js +167 -0
- package/dist/ui.js.map +1 -0
- package/dist/utils/icons.d.ts.map +1 -1
- package/dist/utils/icons.js +59 -7
- package/dist/utils/icons.js.map +1 -1
- 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
|
|
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;
|
|
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
|
-
|
|
21
|
-
const
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
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
|
|
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,
|
|
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;
|
|
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"}
|
package/dist/collectors/index.js
CHANGED
|
@@ -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;
|
|
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":"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;AAQ1C,MAAM,SAAS,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC;AASlC,SAAS,kBAAkB,CAAC,MAAc,EAAE,MAAc;IACxD,OAAO,MAAM;SACV,KAAK,CAAC,IAAI,CAAC;SACX,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;SAC1B,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC;SACjC,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE;QACZ,MAAM,OAAO,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC;QACtC,IAAI,OAAO,KAAK,CAAC,CAAC,EAAE,CAAC;YACnB,OAAO;gBACL,IAAI,EAAE,IAAI;gBACV,IAAI,EAAE,EAAE,IAAI,EAAE,kBAAkB,EAAE,MAAM,EAAE;aAC3C,CAAC;QACJ,CAAC;QAED,OAAO;YACL,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC,EAAE,OAAO,CAAC;YAC5B,OAAO,EAAE,IAAI,CAAC,KAAK,CAAC,OAAO,GAAG,CAAC,CAAC;YAChC,IAAI,EAAE,EAAE,IAAI,EAAE,kBAAkB,EAAE,MAAM,EAAE;SAC3C,CAAC;IACJ,CAAC,CAAC,CAAC;AACP,CAAC;AAED,SAAS,qBAAqB,CAAC,IAAY;IACzC,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,gCAAgC,CAAC,CAAC;IAC3D,IAAI,CAAC,KAAK,EAAE,CAAC;QACX,OAAO,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC;IAC3C,CAAC;IAED,OAAO;QACL,IAAI,EAAE,KAAK,CAAC,CAAC,CAAC;QACd,OAAO,EAAE,KAAK,CAAC,CAAC,CAAC;KAClB,CAAC;AACJ,CAAC;AAED,MAAM,OAAO,eAAgB,SAAQ,aAAa;IACvC,EAAE,GAAG,QAAQ,CAAC;IACd,KAAK,GAAG,gCAAgC,CAAC;IACzC,QAAQ,GAAsB,QAAQ,CAAC;IAE/B,OAAO,CAAiB;IAEzC,YAAY,OAAe;QACzB,KAAK,CAAC,OAAO,CAAC,CAAC;QACf,IAAI,CAAC,OAAO,GAAG;YACb;gBACE,MAAM,EAAE,QAAQ;gBAChB,GAAG,EAAE,MAAM;gBACX,aAAa,EAAE,IAAI,CAAC,OAAO,EAAE,SAAS,EAAE,YAAY,CAAC;gBACrD,OAAO,EAAE,IAAI,CAAC,OAAO,EAAE,SAAS,EAAE,qBAAqB,EAAE,MAAM,EAAE,MAAM,CAAC;aACzE;YACD;gBACE,MAAM,EAAE,QAAQ;gBAChB,GAAG,EAAE,QAAQ;gBACb,aAAa,EAAE,IAAI,CAAC,OAAO,EAAE,SAAS,EAAE,YAAY,CAAC;gBACrD,OAAO,EAAE,IAAI,CAAC,OAAO,EAAE,SAAS,EAAE,qBAAqB,EAAE,QAAQ,EAAE,MAAM,CAAC;aAC3E;SACF,CAAC;IACJ,CAAC;IAED,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,IAAI,CAAC,OAAO,EAAE,CAAC;gBAClC,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,iBAAiB,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;gBAC3D,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,KAAK,CAAC,CAAC;gBAE5B,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,kBAAkB,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;gBAC5D,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,KAAK,CAAC,CAAC;YAC9B,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,KAAK,CAAC,iBAAiB,CAC7B,MAAoB,EACpB,MAAuB;QAEvB,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,YAAY,CACpC,GAAG,MAAM,CAAC,GAAG,oCAAoC,CAClD,CAAC;YACF,OAAO,kBAAkB,CAAC,MAAM,EAAE,MAAM,CAAC,MAAM,CAAC,CAAC;QACnD,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,IAAI,CAAC,wBAAwB,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;QACvD,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,wBAAwB,CACpC,MAAoB,EACpB,MAAuB;QAEvB,IAAI,CAAC;YACH,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,MAAM,CAAC,aAAa,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;YAC7E,OAAO,OAAO;iBACX,MAAM,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC;iBACtC,GAAG,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC,qBAAqB,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;iBACjD,MAAM,CAAC,CAAC,IAAI,EAAoC,EAAE,CAAC,IAAI,KAAK,IAAI,CAAC;iBACjE,GAAG,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;gBACd,GAAG,IAAI;gBACP,IAAI,EAAE,EAAE,IAAI,EAAE,kBAAkB,EAAE,MAAM,EAAE,MAAM,CAAC,MAAM,EAAE;aAC1D,CAAC,CAAC;iBACF,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;QAClD,CAAC;QAAC,OAAO,GAAG,EAAE,CAAC;YACb,IAAK,GAA6B,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;gBACrD,MAAM,CAAC,MAAM,CAAC,IAAI,CAChB,kBAAkB,MAAM,CAAC,MAAM,gBAAiB,GAAa,CAAC,OAAO,EAAE,CACxE,CAAC;YACJ,CAAC;YACD,OAAO,EAAE,CAAC;QACZ,CAAC;IACH,CAAC;IAEO,KAAK,CAAC,kBAAkB,CAC9B,MAAoB,EACpB,MAAuB;QAEvB,MAAM,KAAK,GAAoB,EAAE,CAAC;QAElC,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,YAAY,CACtC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,eAAe,CAAC,EACrC,MAAM,EACN,EAAE,MAAM,EAAE,IAAI,EAAE,CACjB,CAAC;QACF,IAAI,QAAQ;YAAE,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAEnC,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,YAAY,CACzC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,kBAAkB,CAAC,EACxC,MAAM,CACP,CAAC;QACF,IAAI,WAAW;YAAE,KAAK,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAEzC,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,UAAU,CAAC,EAAE,MAAM,EAAE;YAC/E,WAAW,EAAE,GAAG,GAAG,IAAI;YACvB,MAAM,EAAE,CAAC,QAAQ,EAAE,EAAE,CACnB,QAAQ,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,QAAQ,CAAC,QAAQ,CAAC,gBAAgB,CAAC;SACpE,CAAC,CAAC;QACH,KAAK,CAAC,IAAI,CAAC,GAAG,QAAQ,CAAC,CAAC;QAExB,OAAO,KAAK,CAAC;IACf,CAAC;CACF"}
|
package/dist/commands/scan.d.ts
CHANGED
|
@@ -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,
|
|
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"}
|
package/dist/commands/scan.js
CHANGED
|
@@ -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;
|
|
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;
|
|
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"}
|