experimental-agent 0.7.1 → 0.8.0-alpha.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,199 @@
1
+ // src/framework/generate.ts
2
+ import * as fs from "fs";
3
+ import * as path from "path";
4
+ var TS_IMPORT_EXT = /\.(ts|mts)$/;
5
+ function generateRegistry(opts) {
6
+ const entries = opts.agents.map(
7
+ (agent) => inlineEntry({ agent, outputDir: opts.outputDir })
8
+ );
9
+ return [
10
+ "// Auto-generated by withAgents - do not edit",
11
+ 'import type { AgentRegistry } from "experimental-agent/framework";',
12
+ "",
13
+ "export const agents = {",
14
+ ...entries,
15
+ "} satisfies AgentRegistry;",
16
+ ""
17
+ ].join("\n");
18
+ }
19
+ function inlineEntry(opts) {
20
+ const config = buildConfig(opts.agent);
21
+ const configStr = JSON.stringify(config, null, 2).split("\n").map((line, i) => i === 0 ? line : ` ${line}`).join("\n");
22
+ const parts = [` config: ${configStr}`];
23
+ if (opts.agent.hasAgentConfig) {
24
+ const configPath = findAgentConfigPath(opts.agent.absolutePath);
25
+ const relativePath = relativeImportPath({
26
+ from: opts.outputDir,
27
+ to: configPath
28
+ });
29
+ parts.push(` import: () => import(${JSON.stringify(relativePath)})`);
30
+ }
31
+ return ` ${JSON.stringify(opts.agent.name)}: {
32
+ ${parts.join(",\n")}
33
+ },`;
34
+ }
35
+ function buildConfig(agent) {
36
+ const config = {};
37
+ if (agent.systemPrompt) {
38
+ config.system = agent.systemPrompt;
39
+ }
40
+ if (agent.skills.length > 0) {
41
+ config.skills = agent.skills;
42
+ }
43
+ return config;
44
+ }
45
+ function findAgentConfigPath(agentDir) {
46
+ const extensions = [".ts", ".js", ".mts", ".mjs"];
47
+ for (const ext of extensions) {
48
+ const candidate = path.join(agentDir, `agent${ext}`);
49
+ if (fs.existsSync(candidate)) {
50
+ return candidate;
51
+ }
52
+ }
53
+ return path.join(agentDir, "agent");
54
+ }
55
+ function relativeImportPath(opts) {
56
+ let rel = path.relative(opts.from, opts.to).replace(/\\/g, "/");
57
+ rel = rel.replace(TS_IMPORT_EXT, "");
58
+ if (!rel.startsWith(".")) {
59
+ rel = `./${rel}`;
60
+ }
61
+ return rel;
62
+ }
63
+
64
+ // src/framework/scan.ts
65
+ import * as fs2 from "fs";
66
+ import * as path2 from "path";
67
+ var AGENT_CONFIG_FILES = ["agent.ts", "agent.js", "agent.mts", "agent.mjs"];
68
+ var MD_EXT = /\.md$/;
69
+ var TS_EXT = /\.(ts|js|mts|mjs)$/;
70
+ function listDir(dir) {
71
+ if (!fs2.existsSync(dir)) {
72
+ return [];
73
+ }
74
+ return fs2.readdirSync(dir);
75
+ }
76
+ function readFile(filePath) {
77
+ return fs2.readFileSync(filePath, "utf-8");
78
+ }
79
+ function fileExists(filePath) {
80
+ return fs2.existsSync(filePath);
81
+ }
82
+ function isDirectory(filePath) {
83
+ return fs2.existsSync(filePath) && fs2.statSync(filePath).isDirectory();
84
+ }
85
+ function discoverAgentDirs(agentsRoot) {
86
+ const entries = listDir(agentsRoot);
87
+ const dirs = [];
88
+ for (const entry of entries) {
89
+ const absolutePath = path2.join(agentsRoot, entry);
90
+ if (!isDirectory(absolutePath)) {
91
+ continue;
92
+ }
93
+ if (entry.startsWith(".")) {
94
+ continue;
95
+ }
96
+ dirs.push({
97
+ name: entry,
98
+ absolutePath,
99
+ hasAgentConfig: hasAgentConfig(absolutePath)
100
+ });
101
+ }
102
+ return dirs;
103
+ }
104
+ function hasAgentConfig(agentDir) {
105
+ return AGENT_CONFIG_FILES.some((f) => fileExists(path2.join(agentDir, f)));
106
+ }
107
+ function getSkills(agentDir) {
108
+ const skillsDir = path2.join(agentDir, "skills");
109
+ const entries = listDir(skillsDir);
110
+ const skills = [];
111
+ const tsSkillImportPaths = [];
112
+ for (const entry of entries.sort()) {
113
+ const absolutePath = path2.join(skillsDir, entry);
114
+ if (MD_EXT.test(entry)) {
115
+ skills.push({
116
+ type: "host",
117
+ path: absolutePath
118
+ });
119
+ } else if (TS_EXT.test(entry)) {
120
+ tsSkillImportPaths.push(absolutePath);
121
+ }
122
+ }
123
+ return { skills, tsSkillImportPaths };
124
+ }
125
+ function getSystemPrompt(agentDir) {
126
+ const systemDir = path2.join(agentDir, "system");
127
+ const entries = listDir(systemDir);
128
+ const mdFiles = entries.filter((e) => MD_EXT.test(e)).sort();
129
+ if (mdFiles.length === 0) {
130
+ return void 0;
131
+ }
132
+ const parts = [];
133
+ for (const file of mdFiles) {
134
+ const content = readFile(path2.join(systemDir, file)).trim();
135
+ if (content) {
136
+ parts.push(content);
137
+ }
138
+ }
139
+ return parts.length > 0 ? parts.join("\n\n") : void 0;
140
+ }
141
+ function scanAgent(agentDir) {
142
+ const { skills, tsSkillImportPaths } = getSkills(agentDir.absolutePath);
143
+ const systemPrompt = getSystemPrompt(agentDir.absolutePath);
144
+ return { ...agentDir, skills, systemPrompt, tsSkillImportPaths };
145
+ }
146
+ function findAgentsRoot(cwd, userDefinedAgentsDir) {
147
+ if (userDefinedAgentsDir) {
148
+ return path2.resolve(cwd, userDefinedAgentsDir);
149
+ }
150
+ const candidates = [
151
+ path2.join(cwd, "src", "agents"),
152
+ path2.join(cwd, "agents")
153
+ ];
154
+ for (const candidate of candidates) {
155
+ if (isDirectory(candidate)) {
156
+ return candidate;
157
+ }
158
+ }
159
+ return null;
160
+ }
161
+
162
+ // src/framework/framework.ts
163
+ var Framework = class {
164
+ cwd;
165
+ agentsDir;
166
+ log;
167
+ constructor(opts) {
168
+ this.cwd = opts.cwd ?? process.cwd();
169
+ this.agentsDir = opts.agentsDir;
170
+ this.log = opts.debug ? console.log.bind(console, "[Framework]") : () => void 0;
171
+ }
172
+ /**
173
+ * Scan the agents directory and return metadata for every agent found.
174
+ * Each subfolder inside the agents directory becomes one agent.
175
+ * Returns an empty array when no agents directory exists.
176
+ */
177
+ discoverAgents() {
178
+ const root = findAgentsRoot(this.cwd, this.agentsDir);
179
+ if (!root) {
180
+ return [];
181
+ }
182
+ const dirs = discoverAgentDirs(root);
183
+ return dirs.map(scanAgent);
184
+ }
185
+ /**
186
+ * Generate the TypeScript source for an agent registry module.
187
+ *
188
+ * @param opts.agents - Scanned agents to include in the registry.
189
+ * @param opts.outputDir - Directory where the registry file will be written
190
+ * (used to compute relative import paths to each agent's config module).
191
+ * @returns The full source text of the generated `_registry.ts` file.
192
+ */
193
+ generateRegistrySource = generateRegistry;
194
+ };
195
+
196
+ export {
197
+ Framework
198
+ };
199
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vc3JjL2ZyYW1ld29yay9nZW5lcmF0ZS50cyIsICIuLi9zcmMvZnJhbWV3b3JrL3NjYW4udHMiLCAiLi4vc3JjL2ZyYW1ld29yay9mcmFtZXdvcmsudHMiXSwKICAic291cmNlc0NvbnRlbnQiOiBbImltcG9ydCAqIGFzIGZzIGZyb20gXCJub2RlOmZzXCI7XG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gXCJub2RlOnBhdGhcIjtcbmltcG9ydCB0eXBlIHsgU2Nhbm5lZEFnZW50IH0gZnJvbSBcIi4vc2NhblwiO1xuXG5jb25zdCBUU19JTVBPUlRfRVhUID0gL1xcLih0c3xtdHMpJC87XG5leHBvcnQgZnVuY3Rpb24gZ2VuZXJhdGVSZWdpc3RyeShvcHRzOiB7XG4gIGFnZW50czogU2Nhbm5lZEFnZW50W107XG4gIG91dHB1dERpcjogc3RyaW5nO1xufSk6IHN0cmluZyB7XG4gIGNvbnN0IGVudHJpZXMgPSBvcHRzLmFnZW50cy5tYXAoKGFnZW50KSA9PlxuICAgIGlubGluZUVudHJ5KHsgYWdlbnQsIG91dHB1dERpcjogb3B0cy5vdXRwdXREaXIgfSlcbiAgKTtcblxuICByZXR1cm4gW1xuICAgIFwiLy8gQXV0by1nZW5lcmF0ZWQgYnkgd2l0aEFnZW50cyAtIGRvIG5vdCBlZGl0XCIsXG4gICAgJ2ltcG9ydCB0eXBlIHsgQWdlbnRSZWdpc3RyeSB9IGZyb20gXCJleHBlcmltZW50YWwtYWdlbnQvZnJhbWV3b3JrXCI7JyxcbiAgICBcIlwiLFxuICAgIFwiZXhwb3J0IGNvbnN0IGFnZW50cyA9IHtcIixcbiAgICAuLi5lbnRyaWVzLFxuICAgIFwifSBzYXRpc2ZpZXMgQWdlbnRSZWdpc3RyeTtcIixcbiAgICBcIlwiLFxuICBdLmpvaW4oXCJcXG5cIik7XG59XG5cbmZ1bmN0aW9uIGlubGluZUVudHJ5KG9wdHM6IHtcbiAgYWdlbnQ6IFNjYW5uZWRBZ2VudDtcbiAgb3V0cHV0RGlyOiBzdHJpbmc7XG59KTogc3RyaW5nIHtcbiAgY29uc3QgY29uZmlnID0gYnVpbGRDb25maWcob3B0cy5hZ2VudCk7XG4gIGNvbnN0IGNvbmZpZ1N0ciA9IEpTT04uc3RyaW5naWZ5KGNvbmZpZywgbnVsbCwgMilcbiAgICAuc3BsaXQoXCJcXG5cIilcbiAgICAubWFwKChsaW5lLCBpKSA9PiAoaSA9PT0gMCA/IGxpbmUgOiBgICAgICAgJHtsaW5lfWApKVxuICAgIC5qb2luKFwiXFxuXCIpO1xuXG4gIGNvbnN0IHBhcnRzID0gW2AgICAgY29uZmlnOiAke2NvbmZpZ1N0cn1gXTtcblxuICBpZiAob3B0cy5hZ2VudC5oYXNBZ2VudENvbmZpZykge1xuICAgIGNvbnN0IGNvbmZpZ1BhdGggPSBmaW5kQWdlbnRDb25maWdQYXRoKG9wdHMuYWdlbnQuYWJzb2x1dGVQYXRoKTtcbiAgICBjb25zdCByZWxhdGl2ZVBhdGggPSByZWxhdGl2ZUltcG9ydFBhdGgoe1xuICAgICAgZnJvbTogb3B0cy5vdXRwdXREaXIsXG4gICAgICB0bzogY29uZmlnUGF0aCxcbiAgICB9KTtcbiAgICBwYXJ0cy5wdXNoKGAgICAgaW1wb3J0OiAoKSA9PiBpbXBvcnQoJHtKU09OLnN0cmluZ2lmeShyZWxhdGl2ZVBhdGgpfSlgKTtcbiAgfVxuXG4gIHJldHVybiBgICAke0pTT04uc3RyaW5naWZ5KG9wdHMuYWdlbnQubmFtZSl9OiB7XFxuJHtwYXJ0cy5qb2luKFwiLFxcblwiKX1cXG4gIH0sYDtcbn1cblxuZnVuY3Rpb24gYnVpbGRDb25maWcoYWdlbnQ6IFNjYW5uZWRBZ2VudCk6IFJlY29yZDxzdHJpbmcsIHVua25vd24+IHtcbiAgY29uc3QgY29uZmlnOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9IHt9O1xuXG4gIGlmIChhZ2VudC5zeXN0ZW1Qcm9tcHQpIHtcbiAgICBjb25maWcuc3lzdGVtID0gYWdlbnQuc3lzdGVtUHJvbXB0O1xuICB9XG5cbiAgaWYgKGFnZW50LnNraWxscy5sZW5ndGggPiAwKSB7XG4gICAgY29uZmlnLnNraWxscyA9IGFnZW50LnNraWxscztcbiAgfVxuXG4gIHJldHVybiBjb25maWc7XG59XG5cbmZ1bmN0aW9uIGZpbmRBZ2VudENvbmZpZ1BhdGgoYWdlbnREaXI6IHN0cmluZyk6IHN0cmluZyB7XG4gIGNvbnN0IGV4dGVuc2lvbnMgPSBbXCIudHNcIiwgXCIuanNcIiwgXCIubXRzXCIsIFwiLm1qc1wiXTtcbiAgZm9yIChjb25zdCBleHQgb2YgZXh0ZW5zaW9ucykge1xuICAgIGNvbnN0IGNhbmRpZGF0ZSA9IHBhdGguam9pbihhZ2VudERpciwgYGFnZW50JHtleHR9YCk7XG4gICAgaWYgKGZzLmV4aXN0c1N5bmMoY2FuZGlkYXRlKSkge1xuICAgICAgcmV0dXJuIGNhbmRpZGF0ZTtcbiAgICB9XG4gIH1cbiAgcmV0dXJuIHBhdGguam9pbihhZ2VudERpciwgXCJhZ2VudFwiKTtcbn1cblxuZnVuY3Rpb24gcmVsYXRpdmVJbXBvcnRQYXRoKG9wdHM6IHsgZnJvbTogc3RyaW5nOyB0bzogc3RyaW5nIH0pOiBzdHJpbmcge1xuICBsZXQgcmVsID0gcGF0aC5yZWxhdGl2ZShvcHRzLmZyb20sIG9wdHMudG8pLnJlcGxhY2UoL1xcXFwvZywgXCIvXCIpO1xuICByZWwgPSByZWwucmVwbGFjZShUU19JTVBPUlRfRVhULCBcIlwiKTtcbiAgaWYgKCFyZWwuc3RhcnRzV2l0aChcIi5cIikpIHtcbiAgICByZWwgPSBgLi8ke3JlbH1gO1xuICB9XG4gIHJldHVybiByZWw7XG59XG5cbiIsICJpbXBvcnQgKiBhcyBmcyBmcm9tIFwibm9kZTpmc1wiO1xuaW1wb3J0ICogYXMgcGF0aCBmcm9tIFwibm9kZTpwYXRoXCI7XG5pbXBvcnQgdHlwZSB7IEhvc3RTa2lsbElucHV0LCBTa2lsbElucHV0IH0gZnJvbSBcIi4uL3NraWxscy90eXBlc1wiO1xuXG5leHBvcnQgdHlwZSBBZ2VudERpciA9IHtcbiAgbmFtZTogc3RyaW5nO1xuICBhYnNvbHV0ZVBhdGg6IHN0cmluZztcbiAgaGFzQWdlbnRDb25maWc6IGJvb2xlYW47XG59O1xuXG5leHBvcnQgdHlwZSBTY2FubmVkQWdlbnQgPSBBZ2VudERpciAmIHtcbiAgc2tpbGxzOiBTa2lsbElucHV0W107XG4gIHN5c3RlbVByb21wdDogc3RyaW5nIHwgdW5kZWZpbmVkO1xuICB0c1NraWxsSW1wb3J0UGF0aHM6IHN0cmluZ1tdO1xufTtcblxuY29uc3QgQUdFTlRfQ09ORklHX0ZJTEVTID0gW1wiYWdlbnQudHNcIiwgXCJhZ2VudC5qc1wiLCBcImFnZW50Lm10c1wiLCBcImFnZW50Lm1qc1wiXTtcbmNvbnN0IE1EX0VYVCA9IC9cXC5tZCQvO1xuY29uc3QgVFNfRVhUID0gL1xcLih0c3xqc3xtdHN8bWpzKSQvO1xuXG5leHBvcnQgZnVuY3Rpb24gbGlzdERpcihkaXI6IHN0cmluZyk6IHN0cmluZ1tdIHtcbiAgaWYgKCFmcy5leGlzdHNTeW5jKGRpcikpIHtcbiAgICByZXR1cm4gW107XG4gIH1cbiAgcmV0dXJuIGZzLnJlYWRkaXJTeW5jKGRpcik7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiByZWFkRmlsZShmaWxlUGF0aDogc3RyaW5nKTogc3RyaW5nIHtcbiAgcmV0dXJuIGZzLnJlYWRGaWxlU3luYyhmaWxlUGF0aCwgXCJ1dGYtOFwiKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGZpbGVFeGlzdHMoZmlsZVBhdGg6IHN0cmluZyk6IGJvb2xlYW4ge1xuICByZXR1cm4gZnMuZXhpc3RzU3luYyhmaWxlUGF0aCk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBpc0RpcmVjdG9yeShmaWxlUGF0aDogc3RyaW5nKTogYm9vbGVhbiB7XG4gIHJldHVybiBmcy5leGlzdHNTeW5jKGZpbGVQYXRoKSAmJiBmcy5zdGF0U3luYyhmaWxlUGF0aCkuaXNEaXJlY3RvcnkoKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGRpc2NvdmVyQWdlbnREaXJzKGFnZW50c1Jvb3Q6IHN0cmluZyk6IEFnZW50RGlyW10ge1xuICBjb25zdCBlbnRyaWVzID0gbGlzdERpcihhZ2VudHNSb290KTtcbiAgY29uc3QgZGlyczogQWdlbnREaXJbXSA9IFtdO1xuXG4gIGZvciAoY29uc3QgZW50cnkgb2YgZW50cmllcykge1xuICAgIGNvbnN0IGFic29sdXRlUGF0aCA9IHBhdGguam9pbihhZ2VudHNSb290LCBlbnRyeSk7XG4gICAgaWYgKCFpc0RpcmVjdG9yeShhYnNvbHV0ZVBhdGgpKSB7XG4gICAgICBjb250aW51ZTtcbiAgICB9XG4gICAgaWYgKGVudHJ5LnN0YXJ0c1dpdGgoXCIuXCIpKSB7XG4gICAgICBjb250aW51ZTtcbiAgICB9XG5cbiAgICBkaXJzLnB1c2goe1xuICAgICAgbmFtZTogZW50cnksXG4gICAgICBhYnNvbHV0ZVBhdGgsXG4gICAgICBoYXNBZ2VudENvbmZpZzogaGFzQWdlbnRDb25maWcoYWJzb2x1dGVQYXRoKSxcbiAgICB9KTtcbiAgfVxuXG4gIHJldHVybiBkaXJzO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gaGFzQWdlbnRDb25maWcoYWdlbnREaXI6IHN0cmluZyk6IGJvb2xlYW4ge1xuICByZXR1cm4gQUdFTlRfQ09ORklHX0ZJTEVTLnNvbWUoKGYpID0+IGZpbGVFeGlzdHMocGF0aC5qb2luKGFnZW50RGlyLCBmKSkpO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0U2tpbGxzKGFnZW50RGlyOiBzdHJpbmcpOiB7XG4gIHNraWxsczogU2tpbGxJbnB1dFtdO1xuICB0c1NraWxsSW1wb3J0UGF0aHM6IHN0cmluZ1tdO1xufSB7XG4gIGNvbnN0IHNraWxsc0RpciA9IHBhdGguam9pbihhZ2VudERpciwgXCJza2lsbHNcIik7XG4gIGNvbnN0IGVudHJpZXMgPSBsaXN0RGlyKHNraWxsc0Rpcik7XG4gIGNvbnN0IHNraWxsczogU2tpbGxJbnB1dFtdID0gW107XG4gIGNvbnN0IHRzU2tpbGxJbXBvcnRQYXRoczogc3RyaW5nW10gPSBbXTtcblxuICBmb3IgKGNvbnN0IGVudHJ5IG9mIGVudHJpZXMuc29ydCgpKSB7XG4gICAgY29uc3QgYWJzb2x1dGVQYXRoID0gcGF0aC5qb2luKHNraWxsc0RpciwgZW50cnkpO1xuXG4gICAgaWYgKE1EX0VYVC50ZXN0KGVudHJ5KSkge1xuICAgICAgc2tpbGxzLnB1c2goe1xuICAgICAgICB0eXBlOiBcImhvc3RcIixcbiAgICAgICAgcGF0aDogYWJzb2x1dGVQYXRoLFxuICAgICAgfSBzYXRpc2ZpZXMgSG9zdFNraWxsSW5wdXQpO1xuICAgIH0gZWxzZSBpZiAoVFNfRVhULnRlc3QoZW50cnkpKSB7XG4gICAgICB0c1NraWxsSW1wb3J0UGF0aHMucHVzaChhYnNvbHV0ZVBhdGgpO1xuICAgIH1cbiAgfVxuXG4gIHJldHVybiB7IHNraWxscywgdHNTa2lsbEltcG9ydFBhdGhzIH07XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRTeXN0ZW1Qcm9tcHQoYWdlbnREaXI6IHN0cmluZyk6IHN0cmluZyB8IHVuZGVmaW5lZCB7XG4gIGNvbnN0IHN5c3RlbURpciA9IHBhdGguam9pbihhZ2VudERpciwgXCJzeXN0ZW1cIik7XG4gIGNvbnN0IGVudHJpZXMgPSBsaXN0RGlyKHN5c3RlbURpcik7XG4gIGNvbnN0IG1kRmlsZXMgPSBlbnRyaWVzLmZpbHRlcigoZSkgPT4gTURfRVhULnRlc3QoZSkpLnNvcnQoKTtcblxuICBpZiAobWRGaWxlcy5sZW5ndGggPT09IDApIHtcbiAgICByZXR1cm4gdW5kZWZpbmVkO1xuICB9XG5cbiAgY29uc3QgcGFydHM6IHN0cmluZ1tdID0gW107XG4gIGZvciAoY29uc3QgZmlsZSBvZiBtZEZpbGVzKSB7XG4gICAgY29uc3QgY29udGVudCA9IHJlYWRGaWxlKHBhdGguam9pbihzeXN0ZW1EaXIsIGZpbGUpKS50cmltKCk7XG4gICAgaWYgKGNvbnRlbnQpIHtcbiAgICAgIHBhcnRzLnB1c2goY29udGVudCk7XG4gICAgfVxuICB9XG5cbiAgcmV0dXJuIHBhcnRzLmxlbmd0aCA+IDAgPyBwYXJ0cy5qb2luKFwiXFxuXFxuXCIpIDogdW5kZWZpbmVkO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gc2NhbkFnZW50KGFnZW50RGlyOiBBZ2VudERpcik6IFNjYW5uZWRBZ2VudCB7XG4gIGNvbnN0IHsgc2tpbGxzLCB0c1NraWxsSW1wb3J0UGF0aHMgfSA9IGdldFNraWxscyhhZ2VudERpci5hYnNvbHV0ZVBhdGgpO1xuICBjb25zdCBzeXN0ZW1Qcm9tcHQgPSBnZXRTeXN0ZW1Qcm9tcHQoYWdlbnREaXIuYWJzb2x1dGVQYXRoKTtcblxuICByZXR1cm4geyAuLi5hZ2VudERpciwgc2tpbGxzLCBzeXN0ZW1Qcm9tcHQsIHRzU2tpbGxJbXBvcnRQYXRocyB9O1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZmluZEFnZW50c1Jvb3QoXG4gIGN3ZDogc3RyaW5nLFxuICB1c2VyRGVmaW5lZEFnZW50c0Rpcjogc3RyaW5nIHwgdW5kZWZpbmVkXG4pOiBzdHJpbmcgfCBudWxsIHtcbiAgaWYgKHVzZXJEZWZpbmVkQWdlbnRzRGlyKSB7XG4gICAgcmV0dXJuIHBhdGgucmVzb2x2ZShjd2QsIHVzZXJEZWZpbmVkQWdlbnRzRGlyKTtcbiAgfVxuICBjb25zdCBjYW5kaWRhdGVzID0gW1xuICAgIHBhdGguam9pbihjd2QsIFwic3JjXCIsIFwiYWdlbnRzXCIpLFxuICAgIHBhdGguam9pbihjd2QsIFwiYWdlbnRzXCIpLFxuICBdO1xuXG4gIGZvciAoY29uc3QgY2FuZGlkYXRlIG9mIGNhbmRpZGF0ZXMpIHtcbiAgICBpZiAoaXNEaXJlY3RvcnkoY2FuZGlkYXRlKSkge1xuICAgICAgcmV0dXJuIGNhbmRpZGF0ZTtcbiAgICB9XG4gIH1cblxuICByZXR1cm4gbnVsbDtcbn1cbiIsICJpbXBvcnQgeyBnZW5lcmF0ZVJlZ2lzdHJ5IH0gZnJvbSBcIi4vZ2VuZXJhdGVcIjtcbmltcG9ydCB0eXBlIHsgU2Nhbm5lZEFnZW50IH0gZnJvbSBcIi4vc2NhblwiO1xuaW1wb3J0IHsgZGlzY292ZXJBZ2VudERpcnMsIGZpbmRBZ2VudHNSb290LCBzY2FuQWdlbnQgfSBmcm9tIFwiLi9zY2FuXCI7XG5cbi8qKlxuICogRnJhbWV3b3JrLWFnbm9zdGljIGNvcmUgZm9yIGZpbGVzeXN0ZW0tYmFzZWQgYWdlbnQgZGlzY292ZXJ5IGFuZFxuICogcmVnaXN0cnkgY29kZSBnZW5lcmF0aW9uLiBBZGFwdGVycyBsaWtlIHtAbGluayBOZXh0fSByZWNlaXZlIGFcbiAqIGBGcmFtZXdvcmtgIGluc3RhbmNlIGFuZCB1c2UgaXQgdG8gc2NhbiBhZ2VudHMgYW5kIHByb2R1Y2UgcmVnaXN0cnlcbiAqIHNvdXJjZSBjb2RlLlxuICovXG5leHBvcnQgY2xhc3MgRnJhbWV3b3JrIHtcbiAgcmVhZG9ubHkgY3dkOiBzdHJpbmc7XG4gIHJlYWRvbmx5IGFnZW50c0Rpcjogc3RyaW5nIHwgdW5kZWZpbmVkO1xuICByZWFkb25seSBsb2c6ICguLi5hcmdzOiB1bmtub3duW10pID0+IHZvaWQ7XG5cbiAgY29uc3RydWN0b3Iob3B0czoge1xuICAgIC8qKlxuICAgICAqIFByb2plY3Qgcm9vdCBkaXJlY3RvcnkuXG4gICAgICogQGRlZmF1bHRWYWx1ZSBgcHJvY2Vzcy5jd2QoKWBcbiAgICAgKi9cbiAgICBjd2Q/OiBzdHJpbmc7XG4gICAgLyoqXG4gICAgICogUGF0aCB0byB0aGUgYWdlbnRzIGRpcmVjdG9yeSwgcmVsYXRpdmUgdG8gYGN3ZGAuXG4gICAgICogV2hlbiBvbWl0dGVkLCBhdXRvLWRldGVjdGVkIGFzIHRoZSBmaXJzdCBleGlzdGluZyBkaXJlY3RvcnlcbiAgICAgKiBhbW9uZyBgc3JjL2FnZW50c2AgYW5kIGBhZ2VudHNgLlxuICAgICAqL1xuICAgIGFnZW50c0Rpcj86IHN0cmluZztcbiAgICAvKiogRW5hYmxlIHZlcmJvc2UgbG9nZ2luZyB0byBzdGRvdXQuICovXG4gICAgZGVidWc/OiBib29sZWFuO1xuICB9KSB7XG4gICAgdGhpcy5jd2QgPSBvcHRzLmN3ZCA/PyBwcm9jZXNzLmN3ZCgpO1xuICAgIHRoaXMuYWdlbnRzRGlyID0gb3B0cy5hZ2VudHNEaXI7XG4gICAgdGhpcy5sb2cgPSBvcHRzLmRlYnVnXG4gICAgICA/IGNvbnNvbGUubG9nLmJpbmQoY29uc29sZSwgXCJbRnJhbWV3b3JrXVwiKVxuICAgICAgOiAoKSA9PiB1bmRlZmluZWQ7XG4gIH1cblxuICAvKipcbiAgICogU2NhbiB0aGUgYWdlbnRzIGRpcmVjdG9yeSBhbmQgcmV0dXJuIG1ldGFkYXRhIGZvciBldmVyeSBhZ2VudCBmb3VuZC5cbiAgICogRWFjaCBzdWJmb2xkZXIgaW5zaWRlIHRoZSBhZ2VudHMgZGlyZWN0b3J5IGJlY29tZXMgb25lIGFnZW50LlxuICAgKiBSZXR1cm5zIGFuIGVtcHR5IGFycmF5IHdoZW4gbm8gYWdlbnRzIGRpcmVjdG9yeSBleGlzdHMuXG4gICAqL1xuICBkaXNjb3ZlckFnZW50cygpOiBTY2FubmVkQWdlbnRbXSB7XG4gICAgY29uc3Qgcm9vdCA9IGZpbmRBZ2VudHNSb290KHRoaXMuY3dkLCB0aGlzLmFnZW50c0Rpcik7XG4gICAgaWYgKCFyb290KSB7XG4gICAgICByZXR1cm4gW107XG4gICAgfVxuICAgIGNvbnN0IGRpcnMgPSBkaXNjb3ZlckFnZW50RGlycyhyb290KTtcbiAgICByZXR1cm4gZGlycy5tYXAoc2NhbkFnZW50KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZW5lcmF0ZSB0aGUgVHlwZVNjcmlwdCBzb3VyY2UgZm9yIGFuIGFnZW50IHJlZ2lzdHJ5IG1vZHVsZS5cbiAgICpcbiAgICogQHBhcmFtIG9wdHMuYWdlbnRzIC0gU2Nhbm5lZCBhZ2VudHMgdG8gaW5jbHVkZSBpbiB0aGUgcmVnaXN0cnkuXG4gICAqIEBwYXJhbSBvcHRzLm91dHB1dERpciAtIERpcmVjdG9yeSB3aGVyZSB0aGUgcmVnaXN0cnkgZmlsZSB3aWxsIGJlIHdyaXR0ZW5cbiAgICogICAodXNlZCB0byBjb21wdXRlIHJlbGF0aXZlIGltcG9ydCBwYXRocyB0byBlYWNoIGFnZW50J3MgY29uZmlnIG1vZHVsZSkuXG4gICAqIEByZXR1cm5zIFRoZSBmdWxsIHNvdXJjZSB0ZXh0IG9mIHRoZSBnZW5lcmF0ZWQgYF9yZWdpc3RyeS50c2AgZmlsZS5cbiAgICovXG4gIGdlbmVyYXRlUmVnaXN0cnlTb3VyY2UgPSBnZW5lcmF0ZVJlZ2lzdHJ5O1xufVxu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TyxLQUFLLENBQUMsQ0FBQyxFQUFFLEtBQUs7QUFFM0QsTUFBSSxRQUFRLFdBQVcsR0FBRztBQUN4QixXQUFPO0FBQUEsRUFDVDtBQUVBLFFBQU0sUUFBa0IsQ0FBQztBQUN6QixhQUFXLFFBQVEsU0FBUztBQUMxQixVQUFNLFVBQVUsU0FBYyxXQUFLLFdBQVcsSUFBSSxDQUFDLEVBQUUsS0FBSztBQUMxRCxRQUFJLFNBQVM7QUFDWCxZQUFNLEtBQUssT0FBTztBQUFBLElBQ3BCO0FBQUEsRUFDRjtBQUVBLFNBQU8sTUFBTSxTQUFTLElBQUksTUFBTSxLQUFLLE1BQU0sSUFBSTtBQUNqRDtBQUVPLFNBQVMsVUFBVSxVQUFrQztBQUMxRCxRQUFNLEVBQUUsUUFBUSxtQkFBbUIsSUFBSSxVQUFVLFNBQVMsWUFBWTtBQUN0RSxRQUFNLGVBQWUsZ0JBQWdCLFNBQVMsWUFBWTtBQUUxRCxTQUFPLEVBQUUsR0FBRyxVQUFVLFFBQVEsY0FBYyxtQkFBbUI7QUFDakU7QUFFTyxTQUFTLGVBQ2QsS0FDQSxzQkFDZTtBQUNmLE1BQUksc0JBQXNCO0FBQ3hCLFdBQVksY0FBUSxLQUFLLG9CQUFvQjtBQUFBLEVBQy9DO0FBQ0EsUUFBTSxhQUFhO0FBQUEsSUFDWixXQUFLLEtBQUssT0FBTyxRQUFRO0FBQUEsSUFDekIsV0FBSyxLQUFLLFFBQVE7QUFBQSxFQUN6QjtBQUVBLGFBQVcsYUFBYSxZQUFZO0FBQ2xDLFFBQUksWUFBWSxTQUFTLEdBQUc7QUFDMUIsYUFBTztBQUFBLElBQ1Q7QUFBQSxFQUNGO0FBRUEsU0FBTztBQUNUOzs7QUMvSE8sSUFBTSxZQUFOLE1BQWdCO0FBQUEsRUFDWjtBQUFBLEVBQ0E7QUFBQSxFQUNBO0FBQUEsRUFFVCxZQUFZLE1BY1Q7QUFDRCxTQUFLLE1BQU0sS0FBSyxPQUFPLFFBQVEsSUFBSTtBQUNuQyxTQUFLLFlBQVksS0FBSztBQUN0QixTQUFLLE1BQU0sS0FBSyxRQUNaLFFBQVEsSUFBSSxLQUFLLFNBQVMsYUFBYSxJQUN2QyxNQUFNO0FBQUEsRUFDWjtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQU9BLGlCQUFpQztBQUMvQixVQUFNLE9BQU8sZUFBZSxLQUFLLEtBQUssS0FBSyxTQUFTO0FBQ3BELFFBQUksQ0FBQyxNQUFNO0FBQ1QsYUFBTyxDQUFDO0FBQUEsSUFDVjtBQUNBLFVBQU0sT0FBTyxrQkFBa0IsSUFBSTtBQUNuQyxXQUFPLEtBQUssSUFBSSxTQUFTO0FBQUEsRUFDM0I7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFVQSx5QkFBeUI7QUFDM0I7IiwKICAibmFtZXMiOiBbImZzIiwgInBhdGgiXQp9Cg==