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.
- package/dist/chunk-DO4HKPHR.mjs +199 -0
- package/dist/chunk-EH4L2CPW.mjs +3779 -0
- package/dist/chunk-LMYEJDMD.mjs +123 -0
- package/dist/{client-D4AiVqFt.d.ts → client-DIanpEQ_.d.ts} +8 -53
- package/dist/{client-DCtmZCY8.d.mts → client-PuttJd6O.d.mts} +8 -53
- package/dist/framework-DH4Z1NLV.d.ts +61 -0
- package/dist/framework-c9fvEy0o.d.mts +61 -0
- package/dist/framework-runtime.d.mts +37 -0
- package/dist/framework-runtime.d.ts +37 -0
- package/dist/framework-runtime.js +5431 -0
- package/dist/framework-runtime.mjs +107 -0
- package/dist/framework.d.mts +11 -0
- package/dist/framework.d.ts +11 -0
- package/dist/framework.js +235 -0
- package/dist/framework.mjs +8 -0
- package/dist/index.d.mts +30 -7
- package/dist/index.d.ts +30 -7
- package/dist/index.js +50 -4
- package/dist/index.mjs +159 -3863
- package/dist/next/agents.d.mts +43 -0
- package/dist/next/agents.d.ts +43 -0
- package/dist/next/agents.js +364 -0
- package/dist/next/agents.mjs +137 -0
- package/dist/next/loader.js +18 -2
- package/dist/next/loader.mjs +1 -1
- package/dist/next.js +18 -2
- package/dist/next.mjs +1 -1
- package/dist/react.d.mts +2 -1
- package/dist/react.d.ts +2 -1
- package/dist/resolve-BZ26gcyj.d.ts +34 -0
- package/dist/resolve-Cma6YFgx.d.mts +34 -0
- package/dist/sandbox.mjs +2 -2
- package/dist/types-B3lbsOa7.d.mts +53 -0
- package/dist/types-B3lbsOa7.d.ts +53 -0
- package/package.json +29 -13
- package/dist/chunk-ZIXWGYE5.mjs +0 -107
|
@@ -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==
|