@sentry/dotagents 0.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/README.md +96 -0
- package/dist/cli/commands/add.d.ts +12 -0
- package/dist/cli/commands/add.d.ts.map +1 -0
- package/dist/cli/commands/add.js +127 -0
- package/dist/cli/commands/add.js.map +1 -0
- package/dist/cli/commands/init.d.ts +10 -0
- package/dist/cli/commands/init.d.ts.map +1 -0
- package/dist/cli/commands/init.js +81 -0
- package/dist/cli/commands/init.js.map +1 -0
- package/dist/cli/commands/init.test.d.ts +2 -0
- package/dist/cli/commands/init.test.d.ts.map +1 -0
- package/dist/cli/commands/init.test.js +71 -0
- package/dist/cli/commands/init.test.js.map +1 -0
- package/dist/cli/commands/install.d.ts +15 -0
- package/dist/cli/commands/install.d.ts.map +1 -0
- package/dist/cli/commands/install.js +139 -0
- package/dist/cli/commands/install.js.map +1 -0
- package/dist/cli/commands/install.test.d.ts +2 -0
- package/dist/cli/commands/install.test.d.ts.map +1 -0
- package/dist/cli/commands/install.test.js +94 -0
- package/dist/cli/commands/install.test.js.map +1 -0
- package/dist/cli/commands/list.d.ts +13 -0
- package/dist/cli/commands/list.d.ts.map +1 -0
- package/dist/cli/commands/list.js +82 -0
- package/dist/cli/commands/list.js.map +1 -0
- package/dist/cli/commands/list.test.d.ts +2 -0
- package/dist/cli/commands/list.test.d.ts.map +1 -0
- package/dist/cli/commands/list.test.js +96 -0
- package/dist/cli/commands/list.test.js.map +1 -0
- package/dist/cli/commands/remove.d.ts +10 -0
- package/dist/cli/commands/remove.d.ts.map +1 -0
- package/dist/cli/commands/remove.js +68 -0
- package/dist/cli/commands/remove.js.map +1 -0
- package/dist/cli/commands/sync.d.ts +16 -0
- package/dist/cli/commands/sync.d.ts.map +1 -0
- package/dist/cli/commands/sync.js +108 -0
- package/dist/cli/commands/sync.js.map +1 -0
- package/dist/cli/commands/sync.test.d.ts +2 -0
- package/dist/cli/commands/sync.test.d.ts.map +1 -0
- package/dist/cli/commands/sync.test.js +100 -0
- package/dist/cli/commands/sync.test.js.map +1 -0
- package/dist/cli/commands/update.d.ts +15 -0
- package/dist/cli/commands/update.d.ts.map +1 -0
- package/dist/cli/commands/update.js +113 -0
- package/dist/cli/commands/update.js.map +1 -0
- package/dist/cli/commands/update.test.d.ts +2 -0
- package/dist/cli/commands/update.test.d.ts.map +1 -0
- package/dist/cli/commands/update.test.js +90 -0
- package/dist/cli/commands/update.test.js.map +1 -0
- package/dist/cli/index.d.ts +2 -0
- package/dist/cli/index.d.ts.map +1 -0
- package/dist/cli/index.js +49 -0
- package/dist/cli/index.js.map +1 -0
- package/dist/config/index.d.ts +5 -0
- package/dist/config/index.d.ts.map +1 -0
- package/dist/config/index.js +4 -0
- package/dist/config/index.js.map +1 -0
- package/dist/config/loader.d.ts +6 -0
- package/dist/config/loader.d.ts.map +1 -0
- package/dist/config/loader.js +35 -0
- package/dist/config/loader.js.map +1 -0
- package/dist/config/loader.test.d.ts +2 -0
- package/dist/config/loader.test.d.ts.map +1 -0
- package/dist/config/loader.test.js +60 -0
- package/dist/config/loader.test.js.map +1 -0
- package/dist/config/schema.d.ts +37 -0
- package/dist/config/schema.d.ts.map +1 -0
- package/dist/config/schema.js +43 -0
- package/dist/config/schema.js.map +1 -0
- package/dist/config/schema.test.d.ts +2 -0
- package/dist/config/schema.test.d.ts.map +1 -0
- package/dist/config/schema.test.js +138 -0
- package/dist/config/schema.test.js.map +1 -0
- package/dist/config/writer.d.ts +17 -0
- package/dist/config/writer.d.ts.map +1 -0
- package/dist/config/writer.js +71 -0
- package/dist/config/writer.js.map +1 -0
- package/dist/config/writer.test.d.ts +2 -0
- package/dist/config/writer.test.d.ts.map +1 -0
- package/dist/config/writer.test.js +103 -0
- package/dist/config/writer.test.js.map +1 -0
- package/dist/gitignore/index.d.ts +2 -0
- package/dist/gitignore/index.d.ts.map +1 -0
- package/dist/gitignore/index.js +2 -0
- package/dist/gitignore/index.js.map +1 -0
- package/dist/gitignore/writer.d.ts +16 -0
- package/dist/gitignore/writer.d.ts.map +1 -0
- package/dist/gitignore/writer.js +40 -0
- package/dist/gitignore/writer.js.map +1 -0
- package/dist/gitignore/writer.test.d.ts +2 -0
- package/dist/gitignore/writer.test.d.ts.map +1 -0
- package/dist/gitignore/writer.test.js +99 -0
- package/dist/gitignore/writer.test.js.map +1 -0
- package/dist/index.d.ts +12 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +8 -0
- package/dist/index.js.map +1 -0
- package/dist/lockfile/index.d.ts +5 -0
- package/dist/lockfile/index.d.ts.map +1 -0
- package/dist/lockfile/index.js +4 -0
- package/dist/lockfile/index.js.map +1 -0
- package/dist/lockfile/loader.d.ts +10 -0
- package/dist/lockfile/loader.d.ts.map +1 -0
- package/dist/lockfile/loader.js +36 -0
- package/dist/lockfile/loader.js.map +1 -0
- package/dist/lockfile/schema.d.ts +42 -0
- package/dist/lockfile/schema.d.ts.map +1 -0
- package/dist/lockfile/schema.js +25 -0
- package/dist/lockfile/schema.js.map +1 -0
- package/dist/lockfile/schema.test.d.ts +2 -0
- package/dist/lockfile/schema.test.d.ts.map +1 -0
- package/dist/lockfile/schema.test.js +60 -0
- package/dist/lockfile/schema.test.js.map +1 -0
- package/dist/lockfile/writer.d.ts +7 -0
- package/dist/lockfile/writer.d.ts.map +1 -0
- package/dist/lockfile/writer.js +17 -0
- package/dist/lockfile/writer.js.map +1 -0
- package/dist/lockfile/writer.test.d.ts +2 -0
- package/dist/lockfile/writer.test.d.ts.map +1 -0
- package/dist/lockfile/writer.test.js +73 -0
- package/dist/lockfile/writer.test.js.map +1 -0
- package/dist/skills/discovery.d.ts +17 -0
- package/dist/skills/discovery.d.ts.map +1 -0
- package/dist/skills/discovery.js +156 -0
- package/dist/skills/discovery.js.map +1 -0
- package/dist/skills/discovery.test.d.ts +2 -0
- package/dist/skills/discovery.test.d.ts.map +1 -0
- package/dist/skills/discovery.test.js +110 -0
- package/dist/skills/discovery.test.js.map +1 -0
- package/dist/skills/index.d.ts +7 -0
- package/dist/skills/index.d.ts.map +1 -0
- package/dist/skills/index.js +4 -0
- package/dist/skills/index.js.map +1 -0
- package/dist/skills/loader.d.ts +14 -0
- package/dist/skills/loader.d.ts.map +1 -0
- package/dist/skills/loader.js +57 -0
- package/dist/skills/loader.js.map +1 -0
- package/dist/skills/loader.test.d.ts +2 -0
- package/dist/skills/loader.test.d.ts.map +1 -0
- package/dist/skills/loader.test.js +69 -0
- package/dist/skills/loader.test.js.map +1 -0
- package/dist/skills/resolver.d.ts +46 -0
- package/dist/skills/resolver.d.ts.map +1 -0
- package/dist/skills/resolver.integration.test.d.ts +2 -0
- package/dist/skills/resolver.integration.test.d.ts.map +1 -0
- package/dist/skills/resolver.integration.test.js +106 -0
- package/dist/skills/resolver.integration.test.js.map +1 -0
- package/dist/skills/resolver.js +82 -0
- package/dist/skills/resolver.js.map +1 -0
- package/dist/skills/resolver.test.d.ts +2 -0
- package/dist/skills/resolver.test.d.ts.map +1 -0
- package/dist/skills/resolver.test.js +36 -0
- package/dist/skills/resolver.test.js.map +1 -0
- package/dist/sources/cache.d.ts +23 -0
- package/dist/sources/cache.d.ts.map +1 -0
- package/dist/sources/cache.js +61 -0
- package/dist/sources/cache.js.map +1 -0
- package/dist/sources/git.d.ts +25 -0
- package/dist/sources/git.d.ts.map +1 -0
- package/dist/sources/git.js +72 -0
- package/dist/sources/git.js.map +1 -0
- package/dist/sources/index.d.ts +5 -0
- package/dist/sources/index.d.ts.map +1 -0
- package/dist/sources/index.js +4 -0
- package/dist/sources/index.js.map +1 -0
- package/dist/sources/local.d.ts +9 -0
- package/dist/sources/local.d.ts.map +1 -0
- package/dist/sources/local.js +32 -0
- package/dist/sources/local.js.map +1 -0
- package/dist/symlinks/index.d.ts +2 -0
- package/dist/symlinks/index.d.ts.map +1 -0
- package/dist/symlinks/index.js +2 -0
- package/dist/symlinks/index.js.map +1 -0
- package/dist/symlinks/manager.d.ts +20 -0
- package/dist/symlinks/manager.d.ts.map +1 -0
- package/dist/symlinks/manager.js +103 -0
- package/dist/symlinks/manager.js.map +1 -0
- package/dist/symlinks/manager.test.d.ts +2 -0
- package/dist/symlinks/manager.test.d.ts.map +1 -0
- package/dist/symlinks/manager.test.js +94 -0
- package/dist/symlinks/manager.test.js.map +1 -0
- package/dist/utils/exec.d.ts +16 -0
- package/dist/utils/exec.d.ts.map +1 -0
- package/dist/utils/exec.js +37 -0
- package/dist/utils/exec.js.map +1 -0
- package/dist/utils/fs.d.ts +6 -0
- package/dist/utils/fs.d.ts.map +1 -0
- package/dist/utils/fs.js +10 -0
- package/dist/utils/fs.js.map +1 -0
- package/dist/utils/hash.d.ts +16 -0
- package/dist/utils/hash.d.ts.map +1 -0
- package/dist/utils/hash.js +50 -0
- package/dist/utils/hash.js.map +1 -0
- package/dist/utils/hash.test.d.ts +2 -0
- package/dist/utils/hash.test.d.ts.map +1 -0
- package/dist/utils/hash.test.js +61 -0
- package/dist/utils/hash.test.js.map +1 -0
- package/dist/utils/index.d.ts +4 -0
- package/dist/utils/index.d.ts.map +1 -0
- package/dist/utils/index.js +4 -0
- package/dist/utils/index.js.map +1 -0
- package/package.json +56 -0
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach } from "vitest";
|
|
2
|
+
import { mkdtemp, mkdir, writeFile, rm } from "node:fs/promises";
|
|
3
|
+
import { existsSync } from "node:fs";
|
|
4
|
+
import { join } from "node:path";
|
|
5
|
+
import { tmpdir } from "node:os";
|
|
6
|
+
import { runInstall, InstallError } from "./install.js";
|
|
7
|
+
import { exec } from "../../utils/exec.js";
|
|
8
|
+
import { loadLockfile } from "../../lockfile/loader.js";
|
|
9
|
+
const SKILL_MD = (name) => `---
|
|
10
|
+
name: ${name}
|
|
11
|
+
description: Test skill ${name}
|
|
12
|
+
---
|
|
13
|
+
|
|
14
|
+
# ${name}
|
|
15
|
+
`;
|
|
16
|
+
describe("runInstall", () => {
|
|
17
|
+
let tmpDir;
|
|
18
|
+
let stateDir;
|
|
19
|
+
let projectRoot;
|
|
20
|
+
let repoDir;
|
|
21
|
+
beforeEach(async () => {
|
|
22
|
+
tmpDir = await mkdtemp(join(tmpdir(), "dotagents-install-"));
|
|
23
|
+
stateDir = join(tmpDir, "state");
|
|
24
|
+
projectRoot = join(tmpDir, "project");
|
|
25
|
+
repoDir = join(tmpDir, "repo");
|
|
26
|
+
process.env["DOTAGENTS_STATE_DIR"] = stateDir;
|
|
27
|
+
// Set up project
|
|
28
|
+
await mkdir(join(projectRoot, ".agents", "skills"), { recursive: true });
|
|
29
|
+
// Create a local git repo with skills
|
|
30
|
+
await mkdir(repoDir, { recursive: true });
|
|
31
|
+
await exec("git", ["init"], { cwd: repoDir });
|
|
32
|
+
await exec("git", ["config", "user.email", "test@test.com"], { cwd: repoDir });
|
|
33
|
+
await exec("git", ["config", "user.name", "Test"], { cwd: repoDir });
|
|
34
|
+
await mkdir(join(repoDir, "pdf"), { recursive: true });
|
|
35
|
+
await writeFile(join(repoDir, "pdf", "SKILL.md"), SKILL_MD("pdf"));
|
|
36
|
+
await writeFile(join(repoDir, "pdf", "prompt.md"), "Process PDFs");
|
|
37
|
+
await mkdir(join(repoDir, "skills", "review"), { recursive: true });
|
|
38
|
+
await writeFile(join(repoDir, "skills", "review", "SKILL.md"), SKILL_MD("review"));
|
|
39
|
+
await exec("git", ["add", "."], { cwd: repoDir });
|
|
40
|
+
await exec("git", ["commit", "-m", "initial"], { cwd: repoDir });
|
|
41
|
+
});
|
|
42
|
+
afterEach(async () => {
|
|
43
|
+
delete process.env["DOTAGENTS_STATE_DIR"];
|
|
44
|
+
await rm(tmpDir, { recursive: true });
|
|
45
|
+
});
|
|
46
|
+
it("installs a skill from a git source", async () => {
|
|
47
|
+
await writeFile(join(projectRoot, "agents.toml"), `version = 1\n\n[[skills]]\nname = "pdf"\nsource = "git:${repoDir}"\n`);
|
|
48
|
+
const result = await runInstall({ projectRoot });
|
|
49
|
+
expect(result.installed).toContain("pdf");
|
|
50
|
+
// Skill directory should exist
|
|
51
|
+
expect(existsSync(join(projectRoot, ".agents", "skills", "pdf", "SKILL.md"))).toBe(true);
|
|
52
|
+
expect(existsSync(join(projectRoot, ".agents", "skills", "pdf", "prompt.md"))).toBe(true);
|
|
53
|
+
});
|
|
54
|
+
it("creates agents.lock after install", async () => {
|
|
55
|
+
await writeFile(join(projectRoot, "agents.toml"), `version = 1\n\n[[skills]]\nname = "pdf"\nsource = "git:${repoDir}"\n`);
|
|
56
|
+
await runInstall({ projectRoot });
|
|
57
|
+
const lockfile = await loadLockfile(join(projectRoot, "agents.lock"));
|
|
58
|
+
expect(lockfile).not.toBeNull();
|
|
59
|
+
expect(lockfile.skills["pdf"]).toBeDefined();
|
|
60
|
+
expect(lockfile.skills["pdf"].integrity).toMatch(/^sha256-/);
|
|
61
|
+
});
|
|
62
|
+
it("installs multiple skills", async () => {
|
|
63
|
+
await writeFile(join(projectRoot, "agents.toml"), `version = 1\n\n[[skills]]\nname = "pdf"\nsource = "git:${repoDir}"\n\n[[skills]]\nname = "review"\nsource = "git:${repoDir}"\n`);
|
|
64
|
+
const result = await runInstall({ projectRoot });
|
|
65
|
+
expect(result.installed).toHaveLength(2);
|
|
66
|
+
expect(existsSync(join(projectRoot, ".agents", "skills", "pdf", "SKILL.md"))).toBe(true);
|
|
67
|
+
expect(existsSync(join(projectRoot, ".agents", "skills", "review", "SKILL.md"))).toBe(true);
|
|
68
|
+
});
|
|
69
|
+
it("regenerates .agents/.gitignore", async () => {
|
|
70
|
+
await writeFile(join(projectRoot, "agents.toml"), `version = 1\n\n[[skills]]\nname = "pdf"\nsource = "git:${repoDir}"\n`);
|
|
71
|
+
await runInstall({ projectRoot });
|
|
72
|
+
const { readFile } = await import("node:fs/promises");
|
|
73
|
+
const gitignore = await readFile(join(projectRoot, ".agents", ".gitignore"), "utf-8");
|
|
74
|
+
expect(gitignore).toContain("/skills/pdf/");
|
|
75
|
+
});
|
|
76
|
+
it("handles empty skills list", async () => {
|
|
77
|
+
await writeFile(join(projectRoot, "agents.toml"), "version = 1\n");
|
|
78
|
+
const result = await runInstall({ projectRoot });
|
|
79
|
+
expect(result.installed).toHaveLength(0);
|
|
80
|
+
});
|
|
81
|
+
it("fails with --frozen when no lockfile exists", async () => {
|
|
82
|
+
await writeFile(join(projectRoot, "agents.toml"), `version = 1\n\n[[skills]]\nname = "pdf"\nsource = "git:${repoDir}"\n`);
|
|
83
|
+
await expect(runInstall({ projectRoot, frozen: true })).rejects.toThrow(InstallError);
|
|
84
|
+
});
|
|
85
|
+
it("frozen mode passes when lockfile matches", async () => {
|
|
86
|
+
await writeFile(join(projectRoot, "agents.toml"), `version = 1\n\n[[skills]]\nname = "pdf"\nsource = "git:${repoDir}"\n`);
|
|
87
|
+
// First install to create lockfile
|
|
88
|
+
await runInstall({ projectRoot });
|
|
89
|
+
// Second install with --frozen
|
|
90
|
+
const result = await runInstall({ projectRoot, frozen: true });
|
|
91
|
+
expect(result.installed).toContain("pdf");
|
|
92
|
+
});
|
|
93
|
+
});
|
|
94
|
+
//# sourceMappingURL=install.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"install.test.js","sourceRoot":"","sources":["../../../src/cli/commands/install.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
export interface SkillStatus {
|
|
2
|
+
name: string;
|
|
3
|
+
source: string;
|
|
4
|
+
commit?: string;
|
|
5
|
+
status: "ok" | "modified" | "missing" | "unlocked";
|
|
6
|
+
}
|
|
7
|
+
export interface ListOptions {
|
|
8
|
+
projectRoot: string;
|
|
9
|
+
json?: boolean;
|
|
10
|
+
}
|
|
11
|
+
export declare function runList(opts: ListOptions): Promise<SkillStatus[]>;
|
|
12
|
+
export default function list(args: string[]): Promise<void>;
|
|
13
|
+
//# sourceMappingURL=list.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"list.d.ts","sourceRoot":"","sources":["../../../src/cli/commands/list.ts"],"names":[],"mappings":"AASA,MAAM,WAAW,WAAW;IAC1B,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;IACf,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,MAAM,EAAE,IAAI,GAAG,UAAU,GAAG,SAAS,GAAG,UAAU,CAAC;CACpD;AAED,MAAM,WAAW,WAAW;IAC1B,WAAW,EAAE,MAAM,CAAC;IACpB,IAAI,CAAC,EAAE,OAAO,CAAC;CAChB;AAED,wBAAsB,OAAO,CAAC,IAAI,EAAE,WAAW,GAAG,OAAO,CAAC,WAAW,EAAE,CAAC,CAuCvE;AAkBD,wBAA8B,IAAI,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CA6BhE"}
|
|
@@ -0,0 +1,82 @@
|
|
|
1
|
+
import { join } from "node:path";
|
|
2
|
+
import { parseArgs } from "node:util";
|
|
3
|
+
import chalk from "chalk";
|
|
4
|
+
import { loadConfig } from "../../config/loader.js";
|
|
5
|
+
import { loadLockfile } from "../../lockfile/loader.js";
|
|
6
|
+
import { isGitLocked } from "../../lockfile/schema.js";
|
|
7
|
+
import { hashDirectory } from "../../utils/hash.js";
|
|
8
|
+
import { existsSync } from "node:fs";
|
|
9
|
+
export async function runList(opts) {
|
|
10
|
+
const { projectRoot } = opts;
|
|
11
|
+
const configPath = join(projectRoot, "agents.toml");
|
|
12
|
+
const lockPath = join(projectRoot, "agents.lock");
|
|
13
|
+
const skillsDir = join(projectRoot, ".agents", "skills");
|
|
14
|
+
const config = await loadConfig(configPath);
|
|
15
|
+
const lockfile = await loadLockfile(lockPath);
|
|
16
|
+
const skillNames = config.skills.map((s) => s.name).sort();
|
|
17
|
+
const results = [];
|
|
18
|
+
for (const name of skillNames) {
|
|
19
|
+
const dep = config.skills.find((s) => s.name === name);
|
|
20
|
+
const locked = lockfile?.skills[name];
|
|
21
|
+
const installed = join(skillsDir, name);
|
|
22
|
+
if (!existsSync(installed)) {
|
|
23
|
+
results.push({ name, source: dep.source, status: "missing" });
|
|
24
|
+
continue;
|
|
25
|
+
}
|
|
26
|
+
if (!locked) {
|
|
27
|
+
results.push({ name, source: dep.source, status: "unlocked" });
|
|
28
|
+
continue;
|
|
29
|
+
}
|
|
30
|
+
// Check integrity
|
|
31
|
+
const integrity = await hashDirectory(installed);
|
|
32
|
+
const commit = isGitLocked(locked) ? locked.commit.slice(0, 8) : undefined;
|
|
33
|
+
if (integrity !== locked.integrity) {
|
|
34
|
+
results.push({ name, source: dep.source, commit, status: "modified" });
|
|
35
|
+
}
|
|
36
|
+
else {
|
|
37
|
+
results.push({ name, source: dep.source, commit, status: "ok" });
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
return results;
|
|
41
|
+
}
|
|
42
|
+
function formatStatus(s) {
|
|
43
|
+
const commit = s.commit ? chalk.dim(` (${s.commit})`) : "";
|
|
44
|
+
const source = chalk.dim(s.source);
|
|
45
|
+
switch (s.status) {
|
|
46
|
+
case "ok":
|
|
47
|
+
return ` ${chalk.green("✓")} ${s.name}${commit} ${source}`;
|
|
48
|
+
case "modified":
|
|
49
|
+
return ` ${chalk.yellow("~")} ${s.name}${commit} ${source} ${chalk.yellow("modified")}`;
|
|
50
|
+
case "missing":
|
|
51
|
+
return ` ${chalk.red("✗")} ${s.name} ${source} ${chalk.red("not installed")}`;
|
|
52
|
+
case "unlocked":
|
|
53
|
+
return ` ${chalk.yellow("?")} ${s.name} ${source} ${chalk.yellow("not in lockfile")}`;
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
export default async function list(args) {
|
|
57
|
+
const { values } = parseArgs({
|
|
58
|
+
args,
|
|
59
|
+
options: {
|
|
60
|
+
json: { type: "boolean" },
|
|
61
|
+
},
|
|
62
|
+
strict: true,
|
|
63
|
+
});
|
|
64
|
+
const { resolve } = await import("node:path");
|
|
65
|
+
const results = await runList({
|
|
66
|
+
projectRoot: resolve("."),
|
|
67
|
+
json: values["json"],
|
|
68
|
+
});
|
|
69
|
+
if (results.length === 0) {
|
|
70
|
+
console.log(chalk.dim("No skills declared in agents.toml."));
|
|
71
|
+
return;
|
|
72
|
+
}
|
|
73
|
+
if (values["json"]) {
|
|
74
|
+
console.log(JSON.stringify(results, null, 2));
|
|
75
|
+
return;
|
|
76
|
+
}
|
|
77
|
+
console.log(chalk.bold("Skills:"));
|
|
78
|
+
for (const s of results) {
|
|
79
|
+
console.log(formatStatus(s));
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
//# sourceMappingURL=list.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"list.js","sourceRoot":"","sources":["../../../src/cli/commands/list.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"list.test.d.ts","sourceRoot":"","sources":["../../../src/cli/commands/list.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,96 @@
|
|
|
1
|
+
import { describe, it, expect, beforeEach, afterEach } from "vitest";
|
|
2
|
+
import { mkdtemp, mkdir, writeFile, rm } from "node:fs/promises";
|
|
3
|
+
import { join } from "node:path";
|
|
4
|
+
import { tmpdir } from "node:os";
|
|
5
|
+
import { runList } from "./list.js";
|
|
6
|
+
import { writeLockfile } from "../../lockfile/writer.js";
|
|
7
|
+
import { hashDirectory } from "../../utils/hash.js";
|
|
8
|
+
const SKILL_MD = (name) => `---
|
|
9
|
+
name: ${name}
|
|
10
|
+
description: Test skill ${name}
|
|
11
|
+
---
|
|
12
|
+
`;
|
|
13
|
+
describe("runList", () => {
|
|
14
|
+
let tmpDir;
|
|
15
|
+
let projectRoot;
|
|
16
|
+
beforeEach(async () => {
|
|
17
|
+
tmpDir = await mkdtemp(join(tmpdir(), "dotagents-list-"));
|
|
18
|
+
projectRoot = join(tmpDir, "project");
|
|
19
|
+
await mkdir(join(projectRoot, ".agents", "skills"), { recursive: true });
|
|
20
|
+
});
|
|
21
|
+
afterEach(async () => {
|
|
22
|
+
await rm(tmpDir, { recursive: true });
|
|
23
|
+
});
|
|
24
|
+
it("returns empty array when no skills declared", async () => {
|
|
25
|
+
await writeFile(join(projectRoot, "agents.toml"), "version = 1\n");
|
|
26
|
+
const results = await runList({ projectRoot });
|
|
27
|
+
expect(results).toHaveLength(0);
|
|
28
|
+
});
|
|
29
|
+
it("reports missing skill when not installed", async () => {
|
|
30
|
+
await writeFile(join(projectRoot, "agents.toml"), `version = 1\n\n[[skills]]\nname = "pdf"\nsource = "org/repo"\n`);
|
|
31
|
+
const results = await runList({ projectRoot });
|
|
32
|
+
expect(results).toHaveLength(1);
|
|
33
|
+
expect(results[0].status).toBe("missing");
|
|
34
|
+
});
|
|
35
|
+
it("reports unlocked skill when no lockfile", async () => {
|
|
36
|
+
await writeFile(join(projectRoot, "agents.toml"), `version = 1\n\n[[skills]]\nname = "pdf"\nsource = "org/repo"\n`);
|
|
37
|
+
// Install the skill directory but no lockfile
|
|
38
|
+
const skillDir = join(projectRoot, ".agents", "skills", "pdf");
|
|
39
|
+
await mkdir(skillDir, { recursive: true });
|
|
40
|
+
await writeFile(join(skillDir, "SKILL.md"), SKILL_MD("pdf"));
|
|
41
|
+
const results = await runList({ projectRoot });
|
|
42
|
+
expect(results).toHaveLength(1);
|
|
43
|
+
expect(results[0].status).toBe("unlocked");
|
|
44
|
+
});
|
|
45
|
+
it("reports ok when integrity matches", async () => {
|
|
46
|
+
await writeFile(join(projectRoot, "agents.toml"), `version = 1\n\n[[skills]]\nname = "pdf"\nsource = "org/repo"\n`);
|
|
47
|
+
const skillDir = join(projectRoot, ".agents", "skills", "pdf");
|
|
48
|
+
await mkdir(skillDir, { recursive: true });
|
|
49
|
+
await writeFile(join(skillDir, "SKILL.md"), SKILL_MD("pdf"));
|
|
50
|
+
const integrity = await hashDirectory(skillDir);
|
|
51
|
+
await writeLockfile(join(projectRoot, "agents.lock"), {
|
|
52
|
+
version: 1,
|
|
53
|
+
skills: {
|
|
54
|
+
pdf: {
|
|
55
|
+
source: "org/repo",
|
|
56
|
+
resolved_url: "https://github.com/org/repo.git",
|
|
57
|
+
resolved_path: "pdf",
|
|
58
|
+
commit: "a".repeat(40),
|
|
59
|
+
integrity,
|
|
60
|
+
},
|
|
61
|
+
},
|
|
62
|
+
});
|
|
63
|
+
const results = await runList({ projectRoot });
|
|
64
|
+
expect(results).toHaveLength(1);
|
|
65
|
+
expect(results[0].status).toBe("ok");
|
|
66
|
+
expect(results[0].commit).toBe("aaaaaaaa");
|
|
67
|
+
});
|
|
68
|
+
it("reports modified when integrity differs", async () => {
|
|
69
|
+
await writeFile(join(projectRoot, "agents.toml"), `version = 1\n\n[[skills]]\nname = "pdf"\nsource = "org/repo"\n`);
|
|
70
|
+
const skillDir = join(projectRoot, ".agents", "skills", "pdf");
|
|
71
|
+
await mkdir(skillDir, { recursive: true });
|
|
72
|
+
await writeFile(join(skillDir, "SKILL.md"), SKILL_MD("pdf"));
|
|
73
|
+
await writeLockfile(join(projectRoot, "agents.lock"), {
|
|
74
|
+
version: 1,
|
|
75
|
+
skills: {
|
|
76
|
+
pdf: {
|
|
77
|
+
source: "org/repo",
|
|
78
|
+
resolved_url: "https://github.com/org/repo.git",
|
|
79
|
+
resolved_path: "pdf",
|
|
80
|
+
commit: "a".repeat(40),
|
|
81
|
+
integrity: "sha256-stale",
|
|
82
|
+
},
|
|
83
|
+
},
|
|
84
|
+
});
|
|
85
|
+
const results = await runList({ projectRoot });
|
|
86
|
+
expect(results).toHaveLength(1);
|
|
87
|
+
expect(results[0].status).toBe("modified");
|
|
88
|
+
});
|
|
89
|
+
it("sorts results by name", async () => {
|
|
90
|
+
await writeFile(join(projectRoot, "agents.toml"), `version = 1\n\n[[skills]]\nname = "z-skill"\nsource = "org/z"\n\n[[skills]]\nname = "a-skill"\nsource = "org/a"\n`);
|
|
91
|
+
const results = await runList({ projectRoot });
|
|
92
|
+
expect(results[0].name).toBe("a-skill");
|
|
93
|
+
expect(results[1].name).toBe("z-skill");
|
|
94
|
+
});
|
|
95
|
+
});
|
|
96
|
+
//# sourceMappingURL=list.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"list.test.js","sourceRoot":"","sources":["../../../src/cli/commands/list.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
export declare class RemoveError extends Error {
|
|
2
|
+
constructor(message: string);
|
|
3
|
+
}
|
|
4
|
+
export interface RemoveOptions {
|
|
5
|
+
projectRoot: string;
|
|
6
|
+
skillName: string;
|
|
7
|
+
}
|
|
8
|
+
export declare function runRemove(opts: RemoveOptions): Promise<void>;
|
|
9
|
+
export default function remove(args: string[]): Promise<void>;
|
|
10
|
+
//# sourceMappingURL=remove.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"remove.d.ts","sourceRoot":"","sources":["../../../src/cli/commands/remove.ts"],"names":[],"mappings":"AAUA,qBAAa,WAAY,SAAQ,KAAK;gBACxB,OAAO,EAAE,MAAM;CAI5B;AAED,MAAM,WAAW,aAAa;IAC5B,WAAW,EAAE,MAAM,CAAC;IACpB,SAAS,EAAE,MAAM,CAAC;CACnB;AAED,wBAAsB,SAAS,CAAC,IAAI,EAAE,aAAa,GAAG,OAAO,CAAC,IAAI,CAAC,CA8BlE;AAED,wBAA8B,MAAM,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CA0BlE"}
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import { join } from "node:path";
|
|
2
|
+
import { rm } from "node:fs/promises";
|
|
3
|
+
import { parseArgs } from "node:util";
|
|
4
|
+
import chalk from "chalk";
|
|
5
|
+
import { loadConfig } from "../../config/loader.js";
|
|
6
|
+
import { removeSkillFromConfig } from "../../config/writer.js";
|
|
7
|
+
import { loadLockfile } from "../../lockfile/loader.js";
|
|
8
|
+
import { writeLockfile } from "../../lockfile/writer.js";
|
|
9
|
+
import { updateAgentsGitignore } from "../../gitignore/writer.js";
|
|
10
|
+
export class RemoveError extends Error {
|
|
11
|
+
constructor(message) {
|
|
12
|
+
super(message);
|
|
13
|
+
this.name = "RemoveError";
|
|
14
|
+
}
|
|
15
|
+
}
|
|
16
|
+
export async function runRemove(opts) {
|
|
17
|
+
const { projectRoot, skillName } = opts;
|
|
18
|
+
const configPath = join(projectRoot, "agents.toml");
|
|
19
|
+
const lockPath = join(projectRoot, "agents.lock");
|
|
20
|
+
const agentsDir = join(projectRoot, ".agents");
|
|
21
|
+
const skillDir = join(agentsDir, "skills", skillName);
|
|
22
|
+
// Verify skill exists in config
|
|
23
|
+
const config = await loadConfig(configPath);
|
|
24
|
+
if (!config.skills.some((s) => s.name === skillName)) {
|
|
25
|
+
throw new RemoveError(`Skill "${skillName}" not found in agents.toml.`);
|
|
26
|
+
}
|
|
27
|
+
// 1. Remove from agents.toml
|
|
28
|
+
await removeSkillFromConfig(configPath, skillName);
|
|
29
|
+
// 2. Delete skill directory
|
|
30
|
+
await rm(skillDir, { recursive: true, force: true });
|
|
31
|
+
// 3. Update lockfile
|
|
32
|
+
const lockfile = await loadLockfile(lockPath);
|
|
33
|
+
if (lockfile) {
|
|
34
|
+
delete lockfile.skills[skillName];
|
|
35
|
+
await writeLockfile(lockPath, lockfile);
|
|
36
|
+
}
|
|
37
|
+
// 4. Regenerate gitignore
|
|
38
|
+
const updatedConfig = await loadConfig(configPath);
|
|
39
|
+
const managedNames = updatedConfig.skills.map((s) => s.name);
|
|
40
|
+
await updateAgentsGitignore(agentsDir, updatedConfig.gitignore, managedNames);
|
|
41
|
+
}
|
|
42
|
+
export default async function remove(args) {
|
|
43
|
+
const { positionals } = parseArgs({
|
|
44
|
+
args,
|
|
45
|
+
allowPositionals: true,
|
|
46
|
+
strict: true,
|
|
47
|
+
});
|
|
48
|
+
const skillName = positionals[0];
|
|
49
|
+
if (!skillName) {
|
|
50
|
+
console.error(chalk.red("Usage: dotagents remove <name>"));
|
|
51
|
+
process.exitCode = 1;
|
|
52
|
+
return;
|
|
53
|
+
}
|
|
54
|
+
const { resolve } = await import("node:path");
|
|
55
|
+
try {
|
|
56
|
+
await runRemove({ projectRoot: resolve("."), skillName });
|
|
57
|
+
console.log(chalk.green(`Removed skill: ${skillName}`));
|
|
58
|
+
}
|
|
59
|
+
catch (err) {
|
|
60
|
+
if (err instanceof RemoveError) {
|
|
61
|
+
console.error(chalk.red(err.message));
|
|
62
|
+
process.exitCode = 1;
|
|
63
|
+
return;
|
|
64
|
+
}
|
|
65
|
+
throw err;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
//# sourceMappingURL=remove.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"remove.js","sourceRoot":"","sources":["../../../src/cli/commands/remove.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,16 @@
|
|
|
1
|
+
export interface SyncIssue {
|
|
2
|
+
type: "orphan" | "modified" | "symlink" | "missing";
|
|
3
|
+
name: string;
|
|
4
|
+
message: string;
|
|
5
|
+
}
|
|
6
|
+
export interface SyncOptions {
|
|
7
|
+
projectRoot: string;
|
|
8
|
+
}
|
|
9
|
+
export interface SyncResult {
|
|
10
|
+
issues: SyncIssue[];
|
|
11
|
+
gitignoreUpdated: boolean;
|
|
12
|
+
symlinksRepaired: number;
|
|
13
|
+
}
|
|
14
|
+
export declare function runSync(opts: SyncOptions): Promise<SyncResult>;
|
|
15
|
+
export default function sync(): Promise<void>;
|
|
16
|
+
//# sourceMappingURL=sync.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sync.d.ts","sourceRoot":"","sources":["../../../src/cli/commands/sync.ts"],"names":[],"mappings":"AAUA,MAAM,WAAW,SAAS;IACxB,IAAI,EAAE,QAAQ,GAAG,UAAU,GAAG,SAAS,GAAG,SAAS,CAAC;IACpD,IAAI,EAAE,MAAM,CAAC;IACb,OAAO,EAAE,MAAM,CAAC;CACjB;AAED,MAAM,WAAW,WAAW;IAC1B,WAAW,EAAE,MAAM,CAAC;CACrB;AAED,MAAM,WAAW,UAAU;IACzB,MAAM,EAAE,SAAS,EAAE,CAAC;IACpB,gBAAgB,EAAE,OAAO,CAAC;IAC1B,gBAAgB,EAAE,MAAM,CAAC;CAC1B;AAED,wBAAsB,OAAO,CAAC,IAAI,EAAE,WAAW,GAAG,OAAO,CAAC,UAAU,CAAC,CA8EpE;AAED,wBAA8B,IAAI,IAAI,OAAO,CAAC,IAAI,CAAC,CAgClD"}
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
import { join } from "node:path";
|
|
2
|
+
import { existsSync } from "node:fs";
|
|
3
|
+
import { readdir } from "node:fs/promises";
|
|
4
|
+
import chalk from "chalk";
|
|
5
|
+
import { loadConfig } from "../../config/loader.js";
|
|
6
|
+
import { loadLockfile } from "../../lockfile/loader.js";
|
|
7
|
+
import { updateAgentsGitignore } from "../../gitignore/writer.js";
|
|
8
|
+
import { ensureSkillsSymlink, verifySymlinks } from "../../symlinks/manager.js";
|
|
9
|
+
import { hashDirectory } from "../../utils/hash.js";
|
|
10
|
+
export async function runSync(opts) {
|
|
11
|
+
const { projectRoot } = opts;
|
|
12
|
+
const configPath = join(projectRoot, "agents.toml");
|
|
13
|
+
const lockPath = join(projectRoot, "agents.lock");
|
|
14
|
+
const agentsDir = join(projectRoot, ".agents");
|
|
15
|
+
const skillsDir = join(agentsDir, "skills");
|
|
16
|
+
const config = await loadConfig(configPath);
|
|
17
|
+
const lockfile = await loadLockfile(lockPath);
|
|
18
|
+
const declaredNames = new Set(config.skills.map((s) => s.name));
|
|
19
|
+
const issues = [];
|
|
20
|
+
// 1. Regenerate .agents/.gitignore
|
|
21
|
+
await updateAgentsGitignore(agentsDir, config.gitignore, [...declaredNames]);
|
|
22
|
+
// 2. Check for orphaned skills (installed but not in agents.toml)
|
|
23
|
+
if (existsSync(skillsDir)) {
|
|
24
|
+
const entries = await readdir(skillsDir, { withFileTypes: true });
|
|
25
|
+
for (const entry of entries) {
|
|
26
|
+
if (!entry.isDirectory())
|
|
27
|
+
continue;
|
|
28
|
+
if (!declaredNames.has(entry.name)) {
|
|
29
|
+
issues.push({
|
|
30
|
+
type: "orphan",
|
|
31
|
+
name: entry.name,
|
|
32
|
+
message: `"${entry.name}" is installed but not in agents.toml`,
|
|
33
|
+
});
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
// 3. Check for missing skills (in agents.toml but not installed)
|
|
38
|
+
for (const name of declaredNames) {
|
|
39
|
+
if (!existsSync(join(skillsDir, name))) {
|
|
40
|
+
issues.push({
|
|
41
|
+
type: "missing",
|
|
42
|
+
name,
|
|
43
|
+
message: `"${name}" is in agents.toml but not installed. Run 'dotagents install'.`,
|
|
44
|
+
});
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
// 4. Verify integrity hashes against lockfile
|
|
48
|
+
if (lockfile) {
|
|
49
|
+
for (const [name, locked] of Object.entries(lockfile.skills)) {
|
|
50
|
+
const installed = join(skillsDir, name);
|
|
51
|
+
if (!existsSync(installed))
|
|
52
|
+
continue;
|
|
53
|
+
const integrity = await hashDirectory(installed);
|
|
54
|
+
if (integrity !== locked.integrity) {
|
|
55
|
+
issues.push({
|
|
56
|
+
type: "modified",
|
|
57
|
+
name,
|
|
58
|
+
message: `"${name}" has been locally modified (integrity mismatch)`,
|
|
59
|
+
});
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
// 5. Verify and repair symlinks
|
|
64
|
+
const targets = config.symlinks?.targets ?? [];
|
|
65
|
+
let symlinksRepaired = 0;
|
|
66
|
+
const symlinkIssues = await verifySymlinks(agentsDir, targets.map((t) => join(projectRoot, t)));
|
|
67
|
+
for (const issue of symlinkIssues) {
|
|
68
|
+
const targetDir = join(projectRoot, issue.target);
|
|
69
|
+
await ensureSkillsSymlink(agentsDir, targetDir);
|
|
70
|
+
symlinksRepaired++;
|
|
71
|
+
}
|
|
72
|
+
return {
|
|
73
|
+
issues,
|
|
74
|
+
gitignoreUpdated: config.gitignore,
|
|
75
|
+
symlinksRepaired,
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
export default async function sync() {
|
|
79
|
+
const { resolve } = await import("node:path");
|
|
80
|
+
const result = await runSync({ projectRoot: resolve(".") });
|
|
81
|
+
if (result.gitignoreUpdated) {
|
|
82
|
+
console.log(chalk.green("Regenerated .agents/.gitignore"));
|
|
83
|
+
}
|
|
84
|
+
else {
|
|
85
|
+
console.log(chalk.green("Removed .agents/.gitignore (skills checked into git)"));
|
|
86
|
+
}
|
|
87
|
+
if (result.symlinksRepaired > 0) {
|
|
88
|
+
console.log(chalk.green(`Repaired ${result.symlinksRepaired} symlink(s)`));
|
|
89
|
+
}
|
|
90
|
+
if (result.issues.length === 0) {
|
|
91
|
+
console.log(chalk.green("Everything in sync."));
|
|
92
|
+
return;
|
|
93
|
+
}
|
|
94
|
+
for (const issue of result.issues) {
|
|
95
|
+
switch (issue.type) {
|
|
96
|
+
case "orphan":
|
|
97
|
+
console.log(chalk.yellow(` warn: ${issue.message}`));
|
|
98
|
+
break;
|
|
99
|
+
case "modified":
|
|
100
|
+
console.log(chalk.yellow(` warn: ${issue.message}`));
|
|
101
|
+
break;
|
|
102
|
+
case "missing":
|
|
103
|
+
console.log(chalk.red(` error: ${issue.message}`));
|
|
104
|
+
break;
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
//# sourceMappingURL=sync.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sync.js","sourceRoot":"","sources":["../../../src/cli/commands/sync.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,IAAI,EAAE,MAAM,WAAW,CAAC;AACjC,OAAO,EAAE,UAAU,EAAE,MAAM,SAAS,CAAC;AACrC,OAAO,EAAE,OAAO,EAAE,MAAM,kBAAkB,CAAC;AAC3C,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,EAAE,UAAU,EAAE,MAAM,wBAAwB,CAAC;AACpD,OAAO,EAAE,YAAY,EAAE,MAAM,0BAA0B,CAAC;AACxD,OAAO,EAAE,qBAAqB,EAAE,MAAM,2BAA2B,CAAC;AAClE,OAAO,EAAE,mBAAmB,EAAE,cAAc,EAAE,MAAM,2BAA2B,CAAC;AAChF,OAAO,EAAE,aAAa,EAAE,MAAM,qBAAqB,CAAC;AAkBpD,MAAM,CAAC,KAAK,UAAU,OAAO,CAAC,IAAiB;IAC7C,MAAM,EAAE,WAAW,EAAE,GAAG,IAAI,CAAC;IAC7B,MAAM,UAAU,GAAG,IAAI,CAAC,WAAW,EAAE,aAAa,CAAC,CAAC;IACpD,MAAM,QAAQ,GAAG,IAAI,CAAC,WAAW,EAAE,aAAa,CAAC,CAAC;IAClD,MAAM,SAAS,GAAG,IAAI,CAAC,WAAW,EAAE,SAAS,CAAC,CAAC;IAC/C,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;IAE5C,MAAM,MAAM,GAAG,MAAM,UAAU,CAAC,UAAU,CAAC,CAAC;IAC5C,MAAM,QAAQ,GAAG,MAAM,YAAY,CAAC,QAAQ,CAAC,CAAC;IAC9C,MAAM,aAAa,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC;IAChE,MAAM,MAAM,GAAgB,EAAE,CAAC;IAE/B,mCAAmC;IACnC,MAAM,qBAAqB,CAAC,SAAS,EAAE,MAAM,CAAC,SAAS,EAAE,CAAC,GAAG,aAAa,CAAC,CAAC,CAAC;IAE7E,kEAAkE;IAClE,IAAI,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;QAC1B,MAAM,OAAO,GAAG,MAAM,OAAO,CAAC,SAAS,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;QAClE,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE,CAAC;YAC5B,IAAI,CAAC,KAAK,CAAC,WAAW,EAAE;gBAAE,SAAS;YACnC,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC;gBACnC,MAAM,CAAC,IAAI,CAAC;oBACV,IAAI,EAAE,QAAQ;oBACd,IAAI,EAAE,KAAK,CAAC,IAAI;oBAChB,OAAO,EAAE,IAAI,KAAK,CAAC,IAAI,uCAAuC;iBAC/D,CAAC,CAAC;YACL,CAAC;QACH,CAAC;IACH,CAAC;IAED,iEAAiE;IACjE,KAAK,MAAM,IAAI,IAAI,aAAa,EAAE,CAAC;QACjC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC,EAAE,CAAC;YACvC,MAAM,CAAC,IAAI,CAAC;gBACV,IAAI,EAAE,SAAS;gBACf,IAAI;gBACJ,OAAO,EAAE,IAAI,IAAI,iEAAiE;aACnF,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,8CAA8C;IAC9C,IAAI,QAAQ,EAAE,CAAC;QACb,KAAK,MAAM,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;YAC7D,MAAM,SAAS,GAAG,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;YACxC,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC;gBAAE,SAAS;YAErC,MAAM,SAAS,GAAG,MAAM,aAAa,CAAC,SAAS,CAAC,CAAC;YACjD,IAAI,SAAS,KAAK,MAAM,CAAC,SAAS,EAAE,CAAC;gBACnC,MAAM,CAAC,IAAI,CAAC;oBACV,IAAI,EAAE,UAAU;oBAChB,IAAI;oBACJ,OAAO,EAAE,IAAI,IAAI,kDAAkD;iBACpE,CAAC,CAAC;YACL,CAAC;QACH,CAAC;IACH,CAAC;IAED,gCAAgC;IAChC,MAAM,OAAO,GAAG,MAAM,CAAC,QAAQ,EAAE,OAAO,IAAI,EAAE,CAAC;IAC/C,IAAI,gBAAgB,GAAG,CAAC,CAAC;IAEzB,MAAM,aAAa,GAAG,MAAM,cAAc,CACxC,SAAS,EACT,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC,CACzC,CAAC;IAEF,KAAK,MAAM,KAAK,IAAI,aAAa,EAAE,CAAC;QAClC,MAAM,SAAS,GAAG,IAAI,CAAC,WAAW,EAAE,KAAK,CAAC,MAAM,CAAC,CAAC;QAClD,MAAM,mBAAmB,CAAC,SAAS,EAAE,SAAS,CAAC,CAAC;QAChD,gBAAgB,EAAE,CAAC;IACrB,CAAC;IAED,OAAO;QACL,MAAM;QACN,gBAAgB,EAAE,MAAM,CAAC,SAAS;QAClC,gBAAgB;KACjB,CAAC;AACJ,CAAC;AAED,MAAM,CAAC,OAAO,CAAC,KAAK,UAAU,IAAI;IAChC,MAAM,EAAE,OAAO,EAAE,GAAG,MAAM,MAAM,CAAC,WAAW,CAAC,CAAC;IAC9C,MAAM,MAAM,GAAG,MAAM,OAAO,CAAC,EAAE,WAAW,EAAE,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;IAE5D,IAAI,MAAM,CAAC,gBAAgB,EAAE,CAAC;QAC5B,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,gCAAgC,CAAC,CAAC,CAAC;IAC7D,CAAC;SAAM,CAAC;QACN,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,sDAAsD,CAAC,CAAC,CAAC;IACnF,CAAC;IAED,IAAI,MAAM,CAAC,gBAAgB,GAAG,CAAC,EAAE,CAAC;QAChC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,YAAY,MAAM,CAAC,gBAAgB,aAAa,CAAC,CAAC,CAAC;IAC7E,CAAC;IAED,IAAI,MAAM,CAAC,MAAM,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAC/B,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,qBAAqB,CAAC,CAAC,CAAC;QAChD,OAAO;IACT,CAAC;IAED,KAAK,MAAM,KAAK,IAAI,MAAM,CAAC,MAAM,EAAE,CAAC;QAClC,QAAQ,KAAK,CAAC,IAAI,EAAE,CAAC;YACnB,KAAK,QAAQ;gBACX,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,WAAW,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;gBACtD,MAAM;YACR,KAAK,UAAU;gBACb,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,MAAM,CAAC,WAAW,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;gBACtD,MAAM;YACR,KAAK,SAAS;gBACZ,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,YAAY,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;gBACpD,MAAM;QACV,CAAC;IACH,CAAC;AACH,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"sync.test.d.ts","sourceRoot":"","sources":["../../../src/cli/commands/sync.test.ts"],"names":[],"mappings":""}
|