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,43 @@
1
+ import { F as Framework } from '../framework-c9fvEy0o.mjs';
2
+ import { NextConfig } from 'next';
3
+ import '../types-B3lbsOa7.mjs';
4
+
5
+ /** A Next.js config object or async config function. */
6
+ type NextConfigInput = NextConfig | ((phase: string, ctx: {
7
+ defaultConfig: NextConfig;
8
+ }) => Promise<NextConfig>);
9
+ /** Options for {@link withAgents}. */
10
+ type WithAgentsConfig = {
11
+ /**
12
+ * Path to the agents directory, relative to the project root.
13
+ * When omitted, auto-detected as the first existing directory
14
+ * among `src/agents` and `agents`.
15
+ */
16
+ agentsDir?: string;
17
+ /** Enable verbose logging to stdout. */
18
+ debug?: boolean;
19
+ };
20
+
21
+ /**
22
+ * Next.js config wrapper that discovers agents on the filesystem and
23
+ * generates the route handler + registry files needed to serve them.
24
+ *
25
+ * Call this in `next.config.ts`:
26
+ *
27
+ * @example
28
+ * ```ts
29
+ * import { withAgents } from "experimental-agent/next/agents";
30
+ *
31
+ * export default withAgents({ reactStrictMode: true });
32
+ * ```
33
+ *
34
+ * Debug logs:
35
+ * ```ts
36
+ * import { withAgents } from "experimental-agent/next/agents";
37
+ *
38
+ * export default withAgents({ reactStrictMode: true }, { debug: true });
39
+ * ```
40
+ */
41
+ declare function withAgents(config: NextConfigInput, agentsConfig?: Pick<ConstructorParameters<typeof Framework>[0], "agentsDir" | "debug">): NextConfigInput;
42
+
43
+ export { type NextConfigInput, type WithAgentsConfig, withAgents };
@@ -0,0 +1,43 @@
1
+ import { F as Framework } from '../framework-DH4Z1NLV.js';
2
+ import { NextConfig } from 'next';
3
+ import '../types-B3lbsOa7.js';
4
+
5
+ /** A Next.js config object or async config function. */
6
+ type NextConfigInput = NextConfig | ((phase: string, ctx: {
7
+ defaultConfig: NextConfig;
8
+ }) => Promise<NextConfig>);
9
+ /** Options for {@link withAgents}. */
10
+ type WithAgentsConfig = {
11
+ /**
12
+ * Path to the agents directory, relative to the project root.
13
+ * When omitted, auto-detected as the first existing directory
14
+ * among `src/agents` and `agents`.
15
+ */
16
+ agentsDir?: string;
17
+ /** Enable verbose logging to stdout. */
18
+ debug?: boolean;
19
+ };
20
+
21
+ /**
22
+ * Next.js config wrapper that discovers agents on the filesystem and
23
+ * generates the route handler + registry files needed to serve them.
24
+ *
25
+ * Call this in `next.config.ts`:
26
+ *
27
+ * @example
28
+ * ```ts
29
+ * import { withAgents } from "experimental-agent/next/agents";
30
+ *
31
+ * export default withAgents({ reactStrictMode: true });
32
+ * ```
33
+ *
34
+ * Debug logs:
35
+ * ```ts
36
+ * import { withAgents } from "experimental-agent/next/agents";
37
+ *
38
+ * export default withAgents({ reactStrictMode: true }, { debug: true });
39
+ * ```
40
+ */
41
+ declare function withAgents(config: NextConfigInput, agentsConfig?: Pick<ConstructorParameters<typeof Framework>[0], "agentsDir" | "debug">): NextConfigInput;
42
+
43
+ export { type NextConfigInput, type WithAgentsConfig, withAgents };
@@ -0,0 +1,364 @@
1
+ "use strict";
2
+ var __create = Object.create;
3
+ var __defProp = Object.defineProperty;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __getOwnPropNames = Object.getOwnPropertyNames;
6
+ var __getProtoOf = Object.getPrototypeOf;
7
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
8
+ var __export = (target, all) => {
9
+ for (var name in all)
10
+ __defProp(target, name, { get: all[name], enumerable: true });
11
+ };
12
+ var __copyProps = (to, from, except, desc) => {
13
+ if (from && typeof from === "object" || typeof from === "function") {
14
+ for (let key of __getOwnPropNames(from))
15
+ if (!__hasOwnProp.call(to, key) && key !== except)
16
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
17
+ }
18
+ return to;
19
+ };
20
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
21
+ // If the importer is in node compatibility mode or this is not an ESM
22
+ // file that has been converted to a CommonJS file using a Babel-
23
+ // compatible transform (i.e. "__esModule" has not been set), then set
24
+ // "default" to the CommonJS "module.exports" for node compatibility.
25
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
26
+ mod
27
+ ));
28
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
29
+
30
+ // src/framework-next/agents-plugin.ts
31
+ var agents_plugin_exports = {};
32
+ __export(agents_plugin_exports, {
33
+ withAgents: () => withAgents
34
+ });
35
+ module.exports = __toCommonJS(agents_plugin_exports);
36
+
37
+ // src/framework/generate.ts
38
+ var fs = __toESM(require("fs"));
39
+ var path = __toESM(require("path"));
40
+ var TS_IMPORT_EXT = /\.(ts|mts)$/;
41
+ function generateRegistry(opts) {
42
+ const entries = opts.agents.map(
43
+ (agent) => inlineEntry({ agent, outputDir: opts.outputDir })
44
+ );
45
+ return [
46
+ "// Auto-generated by withAgents - do not edit",
47
+ 'import type { AgentRegistry } from "experimental-agent/framework";',
48
+ "",
49
+ "export const agents = {",
50
+ ...entries,
51
+ "} satisfies AgentRegistry;",
52
+ ""
53
+ ].join("\n");
54
+ }
55
+ function inlineEntry(opts) {
56
+ const config = buildConfig(opts.agent);
57
+ const configStr = JSON.stringify(config, null, 2).split("\n").map((line, i) => i === 0 ? line : ` ${line}`).join("\n");
58
+ const parts = [` config: ${configStr}`];
59
+ if (opts.agent.hasAgentConfig) {
60
+ const configPath = findAgentConfigPath(opts.agent.absolutePath);
61
+ const relativePath = relativeImportPath({
62
+ from: opts.outputDir,
63
+ to: configPath
64
+ });
65
+ parts.push(` import: () => import(${JSON.stringify(relativePath)})`);
66
+ }
67
+ return ` ${JSON.stringify(opts.agent.name)}: {
68
+ ${parts.join(",\n")}
69
+ },`;
70
+ }
71
+ function buildConfig(agent) {
72
+ const config = {};
73
+ if (agent.systemPrompt) {
74
+ config.system = agent.systemPrompt;
75
+ }
76
+ if (agent.skills.length > 0) {
77
+ config.skills = agent.skills;
78
+ }
79
+ return config;
80
+ }
81
+ function findAgentConfigPath(agentDir) {
82
+ const extensions = [".ts", ".js", ".mts", ".mjs"];
83
+ for (const ext of extensions) {
84
+ const candidate = path.join(agentDir, `agent${ext}`);
85
+ if (fs.existsSync(candidate)) {
86
+ return candidate;
87
+ }
88
+ }
89
+ return path.join(agentDir, "agent");
90
+ }
91
+ function relativeImportPath(opts) {
92
+ let rel = path.relative(opts.from, opts.to).replace(/\\/g, "/");
93
+ rel = rel.replace(TS_IMPORT_EXT, "");
94
+ if (!rel.startsWith(".")) {
95
+ rel = `./${rel}`;
96
+ }
97
+ return rel;
98
+ }
99
+
100
+ // src/framework/scan.ts
101
+ var fs2 = __toESM(require("fs"));
102
+ var path2 = __toESM(require("path"));
103
+ var AGENT_CONFIG_FILES = ["agent.ts", "agent.js", "agent.mts", "agent.mjs"];
104
+ var MD_EXT = /\.md$/;
105
+ var TS_EXT = /\.(ts|js|mts|mjs)$/;
106
+ function listDir(dir) {
107
+ if (!fs2.existsSync(dir)) {
108
+ return [];
109
+ }
110
+ return fs2.readdirSync(dir);
111
+ }
112
+ function readFile(filePath) {
113
+ return fs2.readFileSync(filePath, "utf-8");
114
+ }
115
+ function fileExists(filePath) {
116
+ return fs2.existsSync(filePath);
117
+ }
118
+ function isDirectory(filePath) {
119
+ return fs2.existsSync(filePath) && fs2.statSync(filePath).isDirectory();
120
+ }
121
+ function discoverAgentDirs(agentsRoot) {
122
+ const entries = listDir(agentsRoot);
123
+ const dirs = [];
124
+ for (const entry of entries) {
125
+ const absolutePath = path2.join(agentsRoot, entry);
126
+ if (!isDirectory(absolutePath)) {
127
+ continue;
128
+ }
129
+ if (entry.startsWith(".")) {
130
+ continue;
131
+ }
132
+ dirs.push({
133
+ name: entry,
134
+ absolutePath,
135
+ hasAgentConfig: hasAgentConfig(absolutePath)
136
+ });
137
+ }
138
+ return dirs;
139
+ }
140
+ function hasAgentConfig(agentDir) {
141
+ return AGENT_CONFIG_FILES.some((f) => fileExists(path2.join(agentDir, f)));
142
+ }
143
+ function getSkills(agentDir) {
144
+ const skillsDir = path2.join(agentDir, "skills");
145
+ const entries = listDir(skillsDir);
146
+ const skills = [];
147
+ const tsSkillImportPaths = [];
148
+ for (const entry of entries.sort()) {
149
+ const absolutePath = path2.join(skillsDir, entry);
150
+ if (MD_EXT.test(entry)) {
151
+ skills.push({
152
+ type: "host",
153
+ path: absolutePath
154
+ });
155
+ } else if (TS_EXT.test(entry)) {
156
+ tsSkillImportPaths.push(absolutePath);
157
+ }
158
+ }
159
+ return { skills, tsSkillImportPaths };
160
+ }
161
+ function getSystemPrompt(agentDir) {
162
+ const systemDir = path2.join(agentDir, "system");
163
+ const entries = listDir(systemDir);
164
+ const mdFiles = entries.filter((e) => MD_EXT.test(e)).sort();
165
+ if (mdFiles.length === 0) {
166
+ return void 0;
167
+ }
168
+ const parts = [];
169
+ for (const file of mdFiles) {
170
+ const content = readFile(path2.join(systemDir, file)).trim();
171
+ if (content) {
172
+ parts.push(content);
173
+ }
174
+ }
175
+ return parts.length > 0 ? parts.join("\n\n") : void 0;
176
+ }
177
+ function scanAgent(agentDir) {
178
+ const { skills, tsSkillImportPaths } = getSkills(agentDir.absolutePath);
179
+ const systemPrompt = getSystemPrompt(agentDir.absolutePath);
180
+ return { ...agentDir, skills, systemPrompt, tsSkillImportPaths };
181
+ }
182
+ function findAgentsRoot(cwd, userDefinedAgentsDir) {
183
+ if (userDefinedAgentsDir) {
184
+ return path2.resolve(cwd, userDefinedAgentsDir);
185
+ }
186
+ const candidates = [
187
+ path2.join(cwd, "src", "agents"),
188
+ path2.join(cwd, "agents")
189
+ ];
190
+ for (const candidate of candidates) {
191
+ if (isDirectory(candidate)) {
192
+ return candidate;
193
+ }
194
+ }
195
+ return null;
196
+ }
197
+
198
+ // src/framework/framework.ts
199
+ var Framework = class {
200
+ cwd;
201
+ agentsDir;
202
+ log;
203
+ constructor(opts) {
204
+ this.cwd = opts.cwd ?? process.cwd();
205
+ this.agentsDir = opts.agentsDir;
206
+ this.log = opts.debug ? console.log.bind(console, "[Framework]") : () => void 0;
207
+ }
208
+ /**
209
+ * Scan the agents directory and return metadata for every agent found.
210
+ * Each subfolder inside the agents directory becomes one agent.
211
+ * Returns an empty array when no agents directory exists.
212
+ */
213
+ discoverAgents() {
214
+ const root = findAgentsRoot(this.cwd, this.agentsDir);
215
+ if (!root) {
216
+ return [];
217
+ }
218
+ const dirs = discoverAgentDirs(root);
219
+ return dirs.map(scanAgent);
220
+ }
221
+ /**
222
+ * Generate the TypeScript source for an agent registry module.
223
+ *
224
+ * @param opts.agents - Scanned agents to include in the registry.
225
+ * @param opts.outputDir - Directory where the registry file will be written
226
+ * (used to compute relative import paths to each agent's config module).
227
+ * @returns The full source text of the generated `_registry.ts` file.
228
+ */
229
+ generateRegistrySource = generateRegistry;
230
+ };
231
+
232
+ // src/framework-next/next.ts
233
+ var fs3 = __toESM(require("fs"));
234
+ var path3 = __toESM(require("path"));
235
+ var Next = class {
236
+ constructor(framework) {
237
+ this.framework = framework;
238
+ }
239
+ get appDir() {
240
+ const cwd = this.framework.cwd;
241
+ const srcAppDir = path3.join(cwd, "src", "app");
242
+ const rootAppDir = path3.join(cwd, "app");
243
+ if (this.hasAppLayout(srcAppDir)) {
244
+ return "src/app";
245
+ }
246
+ if (this.hasAppLayout(rootAppDir)) {
247
+ return "app";
248
+ }
249
+ if (fs3.existsSync(srcAppDir)) {
250
+ return "src/app";
251
+ }
252
+ return "app";
253
+ }
254
+ generateNextRouteHandlerSource(opts) {
255
+ return `// Auto-generated by withAgents - do not edit
256
+ // @ts-nocheck
257
+ // biome-ignore format, lint, lint/style, lint/suspicious, lint/complexity, lint/performance, lint/security, lint/correctness, lint/strict
258
+ // prettier-ignore
259
+
260
+ import { runtime } from "experimental-agent/framework-runtime";
261
+ import { handleRequest } from "experimental-agent";
262
+ import { agents } from ${JSON.stringify(opts.registryImportPath)};
263
+
264
+ async function workflow(agent, sessionId, ...args) {
265
+ "use workflow";
266
+ return agent.session(sessionId).send(...args);
267
+ }
268
+
269
+ async function handler(
270
+ req: Request,
271
+ ctx: { params: Promise<{ agent: string; path?: string[] }> }
272
+ ) {
273
+ const params = await ctx.params;
274
+ const name = params.agent;
275
+ const instance = await runtime.resolveAgent({ name, agents });
276
+
277
+ if (!instance) {
278
+ const available = Object.keys(agents).join(", ");
279
+ const body =
280
+ \`Agent "\${name}" not found.\\n\\n\` +
281
+ \`Available agents: \${available}\\n\\n\` +
282
+ \`hint: each folder inside agents/ becomes an agent.\\n\` +
283
+ \` Create agents/\${name}/ to add this agent.\\n\`;
284
+ return new Response(body, { status: 404 });
285
+ }
286
+
287
+ return handleRequest(instance, { basePath: \`/.well-known/agents/\${name}\`, workflow }).fetch(req);
288
+ }
289
+
290
+ export const GET = handler;
291
+ export const POST = handler;
292
+ `;
293
+ }
294
+ /**
295
+ * Pure codegen step: scan agents and return the generated source
296
+ * strings without touching the filesystem. Useful for adapters that
297
+ * need to control where and how files are written.
298
+ *
299
+ * Returns `null` when no agents are found.
300
+ */
301
+ filesToWrite() {
302
+ const agents = this.framework.discoverAgents();
303
+ if (agents.length === 0) {
304
+ return null;
305
+ }
306
+ this.framework.log(
307
+ "discovered agents:",
308
+ agents.map((a) => a.name)
309
+ );
310
+ const dir = path3.join(
311
+ this.framework.cwd,
312
+ this.appDir,
313
+ ".well-known",
314
+ "agents",
315
+ "[agent]",
316
+ "[[...path]]"
317
+ );
318
+ const registrySource = this.framework.generateRegistrySource({
319
+ agents,
320
+ outputDir: dir
321
+ });
322
+ const routeSource = this.generateNextRouteHandlerSource({
323
+ registryImportPath: "./_registry"
324
+ });
325
+ return { registrySource, routeSource, dir };
326
+ }
327
+ /**
328
+ * Scan for agents, generate source, and write the `_registry.ts` and
329
+ * `route.ts` files into the Next.js app directory.
330
+ */
331
+ writeFiles() {
332
+ const filesToWrite = this.filesToWrite();
333
+ if (!filesToWrite) {
334
+ console.warn("[withAgents] No agents found");
335
+ return;
336
+ }
337
+ const { registrySource, routeSource, dir } = filesToWrite;
338
+ fs3.mkdirSync(dir, { recursive: true });
339
+ fs3.writeFileSync(path3.join(dir, "_registry.ts"), registrySource);
340
+ fs3.writeFileSync(path3.join(dir, "route.ts"), routeSource);
341
+ const gitignorePath = path3.join(dir, ".gitignore");
342
+ if (!fs3.existsSync(gitignorePath)) {
343
+ fs3.writeFileSync(gitignorePath, "*\n");
344
+ }
345
+ }
346
+ hasAppLayout(dir) {
347
+ return ["layout.tsx", "layout.js", "layout.jsx"].some(
348
+ (f) => fs3.existsSync(path3.join(dir, f))
349
+ );
350
+ }
351
+ };
352
+
353
+ // src/framework-next/agents-plugin.ts
354
+ function withAgents(config, agentsConfig = {}) {
355
+ const framework = new Framework(agentsConfig);
356
+ const next = new Next(framework);
357
+ next.writeFiles();
358
+ return config;
359
+ }
360
+ // Annotate the CommonJS export names for ESM import in node:
361
+ 0 && (module.exports = {
362
+ withAgents
363
+ });
364
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/framework-next/agents-plugin.ts", "../../src/framework/generate.ts", "../../src/framework/scan.ts", "../../src/framework/framework.ts", "../../src/framework-next/next.ts"],
  "sourcesContent": ["import { Framework } from \"../framework/framework\";\nimport { Next, type NextConfigInput } from \"./next\";\n\nexport type { NextConfigInput, WithAgentsConfig } from \"./next\";\n\n/**\n * Next.js config wrapper that discovers agents on the filesystem and\n * generates the route handler + registry files needed to serve them.\n *\n * Call this in `next.config.ts`:\n *\n * @example\n * ```ts\n * import { withAgents } from \"experimental-agent/next/agents\";\n *\n * export default withAgents({ reactStrictMode: true });\n * ```\n *\n * Debug logs:\n * ```ts\n * import { withAgents } from \"experimental-agent/next/agents\";\n *\n * export default withAgents({ reactStrictMode: true }, { debug: true });\n * ```\n */\nexport function withAgents(\n  config: NextConfigInput,\n  agentsConfig: Pick<\n    ConstructorParameters<typeof Framework>[0],\n    \"agentsDir\" | \"debug\"\n  > = {}\n): NextConfigInput {\n  const framework = new Framework(agentsConfig);\n  const next = new Next(framework);\n  next.writeFiles();\n\n  return config;\n}\n", "import * as fs from \"node:fs\";\nimport * as path from \"node:path\";\nimport type { ScannedAgent } from \"./scan\";\n\nconst TS_IMPORT_EXT = /\\.(ts|mts)$/;\nexport function generateRegistry(opts: {\n  agents: ScannedAgent[];\n  outputDir: string;\n}): string {\n  const entries = opts.agents.map((agent) =>\n    inlineEntry({ agent, outputDir: opts.outputDir })\n  );\n\n  return [\n    \"// Auto-generated by withAgents - do not edit\",\n    'import type { AgentRegistry } from \"experimental-agent/framework\";',\n    \"\",\n    \"export const agents = {\",\n    ...entries,\n    \"} satisfies AgentRegistry;\",\n    \"\",\n  ].join(\"\\n\");\n}\n\nfunction inlineEntry(opts: {\n  agent: ScannedAgent;\n  outputDir: string;\n}): string {\n  const config = buildConfig(opts.agent);\n  const configStr = JSON.stringify(config, null, 2)\n    .split(\"\\n\")\n    .map((line, i) => (i === 0 ? line : `      ${line}`))\n    .join(\"\\n\");\n\n  const parts = [`    config: ${configStr}`];\n\n  if (opts.agent.hasAgentConfig) {\n    const configPath = findAgentConfigPath(opts.agent.absolutePath);\n    const relativePath = relativeImportPath({\n      from: opts.outputDir,\n      to: configPath,\n    });\n    parts.push(`    import: () => import(${JSON.stringify(relativePath)})`);\n  }\n\n  return `  ${JSON.stringify(opts.agent.name)}: {\\n${parts.join(\",\\n\")}\\n  },`;\n}\n\nfunction buildConfig(agent: ScannedAgent): Record<string, unknown> {\n  const config: Record<string, unknown> = {};\n\n  if (agent.systemPrompt) {\n    config.system = agent.systemPrompt;\n  }\n\n  if (agent.skills.length > 0) {\n    config.skills = agent.skills;\n  }\n\n  return config;\n}\n\nfunction findAgentConfigPath(agentDir: string): string {\n  const extensions = [\".ts\", \".js\", \".mts\", \".mjs\"];\n  for (const ext of extensions) {\n    const candidate = path.join(agentDir, `agent${ext}`);\n    if (fs.existsSync(candidate)) {\n      return candidate;\n    }\n  }\n  return path.join(agentDir, \"agent\");\n}\n\nfunction relativeImportPath(opts: { from: string; to: string }): string {\n  let rel = path.relative(opts.from, opts.to).replace(/\\\\/g, \"/\");\n  rel = rel.replace(TS_IMPORT_EXT, \"\");\n  if (!rel.startsWith(\".\")) {\n    rel = `./${rel}`;\n  }\n  return rel;\n}\n\n", "import * as fs from \"node:fs\";\nimport * as path from \"node:path\";\nimport type { HostSkillInput, SkillInput } from \"../skills/types\";\n\nexport type AgentDir = {\n  name: string;\n  absolutePath: string;\n  hasAgentConfig: boolean;\n};\n\nexport type ScannedAgent = AgentDir & {\n  skills: SkillInput[];\n  systemPrompt: string | undefined;\n  tsSkillImportPaths: string[];\n};\n\nconst AGENT_CONFIG_FILES = [\"agent.ts\", \"agent.js\", \"agent.mts\", \"agent.mjs\"];\nconst MD_EXT = /\\.md$/;\nconst TS_EXT = /\\.(ts|js|mts|mjs)$/;\n\nexport function listDir(dir: string): string[] {\n  if (!fs.existsSync(dir)) {\n    return [];\n  }\n  return fs.readdirSync(dir);\n}\n\nexport function readFile(filePath: string): string {\n  return fs.readFileSync(filePath, \"utf-8\");\n}\n\nexport function fileExists(filePath: string): boolean {\n  return fs.existsSync(filePath);\n}\n\nexport function isDirectory(filePath: string): boolean {\n  return fs.existsSync(filePath) && fs.statSync(filePath).isDirectory();\n}\n\nexport function discoverAgentDirs(agentsRoot: string): AgentDir[] {\n  const entries = listDir(agentsRoot);\n  const dirs: AgentDir[] = [];\n\n  for (const entry of entries) {\n    const absolutePath = path.join(agentsRoot, entry);\n    if (!isDirectory(absolutePath)) {\n      continue;\n    }\n    if (entry.startsWith(\".\")) {\n      continue;\n    }\n\n    dirs.push({\n      name: entry,\n      absolutePath,\n      hasAgentConfig: hasAgentConfig(absolutePath),\n    });\n  }\n\n  return dirs;\n}\n\nexport function hasAgentConfig(agentDir: string): boolean {\n  return AGENT_CONFIG_FILES.some((f) => fileExists(path.join(agentDir, f)));\n}\n\nexport function getSkills(agentDir: string): {\n  skills: SkillInput[];\n  tsSkillImportPaths: string[];\n} {\n  const skillsDir = path.join(agentDir, \"skills\");\n  const entries = listDir(skillsDir);\n  const skills: SkillInput[] = [];\n  const tsSkillImportPaths: string[] = [];\n\n  for (const entry of entries.sort()) {\n    const absolutePath = path.join(skillsDir, entry);\n\n    if (MD_EXT.test(entry)) {\n      skills.push({\n        type: \"host\",\n        path: absolutePath,\n      } satisfies HostSkillInput);\n    } else if (TS_EXT.test(entry)) {\n      tsSkillImportPaths.push(absolutePath);\n    }\n  }\n\n  return { skills, tsSkillImportPaths };\n}\n\nexport function getSystemPrompt(agentDir: string): string | undefined {\n  const systemDir = path.join(agentDir, \"system\");\n  const entries = listDir(systemDir);\n  const mdFiles = entries.filter((e) => MD_EXT.test(e)).sort();\n\n  if (mdFiles.length === 0) {\n    return undefined;\n  }\n\n  const parts: string[] = [];\n  for (const file of mdFiles) {\n    const content = readFile(path.join(systemDir, file)).trim();\n    if (content) {\n      parts.push(content);\n    }\n  }\n\n  return parts.length > 0 ? parts.join(\"\\n\\n\") : undefined;\n}\n\nexport function scanAgent(agentDir: AgentDir): ScannedAgent {\n  const { skills, tsSkillImportPaths } = getSkills(agentDir.absolutePath);\n  const systemPrompt = getSystemPrompt(agentDir.absolutePath);\n\n  return { ...agentDir, skills, systemPrompt, tsSkillImportPaths };\n}\n\nexport function findAgentsRoot(\n  cwd: string,\n  userDefinedAgentsDir: string | undefined\n): string | null {\n  if (userDefinedAgentsDir) {\n    return path.resolve(cwd, userDefinedAgentsDir);\n  }\n  const candidates = [\n    path.join(cwd, \"src\", \"agents\"),\n    path.join(cwd, \"agents\"),\n  ];\n\n  for (const candidate of candidates) {\n    if (isDirectory(candidate)) {\n      return candidate;\n    }\n  }\n\n  return null;\n}\n", "import { generateRegistry } from \"./generate\";\nimport type { ScannedAgent } from \"./scan\";\nimport { discoverAgentDirs, findAgentsRoot, scanAgent } from \"./scan\";\n\n/**\n * Framework-agnostic core for filesystem-based agent discovery and\n * registry code generation. Adapters like {@link Next} receive a\n * `Framework` instance and use it to scan agents and produce registry\n * source code.\n */\nexport class Framework {\n  readonly cwd: string;\n  readonly agentsDir: string | undefined;\n  readonly log: (...args: unknown[]) => void;\n\n  constructor(opts: {\n    /**\n     * Project root directory.\n     * @defaultValue `process.cwd()`\n     */\n    cwd?: string;\n    /**\n     * Path to the agents directory, relative to `cwd`.\n     * When omitted, auto-detected as the first existing directory\n     * among `src/agents` and `agents`.\n     */\n    agentsDir?: string;\n    /** Enable verbose logging to stdout. */\n    debug?: boolean;\n  }) {\n    this.cwd = opts.cwd ?? process.cwd();\n    this.agentsDir = opts.agentsDir;\n    this.log = opts.debug\n      ? console.log.bind(console, \"[Framework]\")\n      : () => undefined;\n  }\n\n  /**\n   * Scan the agents directory and return metadata for every agent found.\n   * Each subfolder inside the agents directory becomes one agent.\n   * Returns an empty array when no agents directory exists.\n   */\n  discoverAgents(): ScannedAgent[] {\n    const root = findAgentsRoot(this.cwd, this.agentsDir);\n    if (!root) {\n      return [];\n    }\n    const dirs = discoverAgentDirs(root);\n    return dirs.map(scanAgent);\n  }\n\n  /**\n   * Generate the TypeScript source for an agent registry module.\n   *\n   * @param opts.agents - Scanned agents to include in the registry.\n   * @param opts.outputDir - Directory where the registry file will be written\n   *   (used to compute relative import paths to each agent's config module).\n   * @returns The full source text of the generated `_registry.ts` file.\n   */\n  generateRegistrySource = generateRegistry;\n}\n", "import * as fs from \"node:fs\";\nimport * as path from \"node:path\";\nimport type { NextConfig } from \"next\";\nimport type { Framework } from \"../framework/framework\";\n\n/** A Next.js config object or async config function. */\nexport type NextConfigInput =\n  | NextConfig\n  | ((\n      phase: string,\n      ctx: { defaultConfig: NextConfig }\n    ) => Promise<NextConfig>);\n\n/** Options for {@link withAgents}. */\nexport type WithAgentsConfig = {\n  /**\n   * Path to the agents directory, relative to the project root.\n   * When omitted, auto-detected as the first existing directory\n   * among `src/agents` and `agents`.\n   */\n  agentsDir?: string;\n  /** Enable verbose logging to stdout. */\n  debug?: boolean;\n};\n\n/**\n * Next.js adapter for filesystem-based agent routing.\n *\n * Receives a {@link Framework} instance (composition) and uses it to\n * discover agents, generate a registry, and write the Next.js route\n * handler files that serve them.\n *\n * Most users should use the {@link withAgents} helper instead of\n * constructing this class directly.\n *\n * @example\n * ```ts\n * import { Framework } from \"experimental-agent/framework\";\n * import { Next } from \"experimental-agent/next/agents\";\n *\n * const framework = new Framework({ cwd: process.cwd() });\n * new Next(framework).generate();\n * ```\n */\nexport class Next {\n  constructor(private readonly framework: Framework) {}\n\n  private get appDir(): string {\n    const cwd = this.framework.cwd;\n    const srcAppDir = path.join(cwd, \"src\", \"app\");\n    const rootAppDir = path.join(cwd, \"app\");\n\n    if (this.hasAppLayout(srcAppDir)) {\n      return \"src/app\";\n    }\n    if (this.hasAppLayout(rootAppDir)) {\n      return \"app\";\n    }\n    if (fs.existsSync(srcAppDir)) {\n      return \"src/app\";\n    }\n    return \"app\";\n  }\n\n  generateNextRouteHandlerSource(opts: { registryImportPath: string }): string {\n    return `// Auto-generated by withAgents - do not edit\n// @ts-nocheck\n// biome-ignore format, lint, lint/style, lint/suspicious, lint/complexity, lint/performance, lint/security, lint/correctness, lint/strict\n// prettier-ignore\n\nimport { runtime } from \"experimental-agent/framework-runtime\";\nimport { handleRequest } from \"experimental-agent\";\nimport { agents } from ${JSON.stringify(opts.registryImportPath)};\n\nasync function workflow(agent, sessionId, ...args) {\n  \"use workflow\";\n  return agent.session(sessionId).send(...args);\n}\n\nasync function handler(\n  req: Request,\n  ctx: { params: Promise<{ agent: string; path?: string[] }> }\n) {\n  const params = await ctx.params;\n  const name = params.agent;\n  const instance = await runtime.resolveAgent({ name, agents });\n\n  if (!instance) {\n    const available = Object.keys(agents).join(\", \");\n    const body =\n      \\`Agent \"\\${name}\" not found.\\\\n\\\\n\\` +\n      \\`Available agents: \\${available}\\\\n\\\\n\\` +\n      \\`hint: each folder inside agents/ becomes an agent.\\\\n\\` +\n      \\`      Create agents/\\${name}/ to add this agent.\\\\n\\`;\n    return new Response(body, { status: 404 });\n  }\n\n  return handleRequest(instance, { basePath: \\`/.well-known/agents/\\${name}\\`, workflow }).fetch(req);\n}\n\nexport const GET = handler;\nexport const POST = handler;\n`;\n  }\n\n  /**\n   * Pure codegen step: scan agents and return the generated source\n   * strings without touching the filesystem. Useful for adapters that\n   * need to control where and how files are written.\n   *\n   * Returns `null` when no agents are found.\n   */\n  filesToWrite(): {\n    registrySource: string;\n    routeSource: string;\n    dir: string;\n  } | null {\n    const agents = this.framework.discoverAgents();\n\n    if (agents.length === 0) {\n      return null;\n    }\n\n    this.framework.log(\n      \"discovered agents:\",\n      agents.map((a) => a.name)\n    );\n\n    const dir = path.join(\n      this.framework.cwd,\n      this.appDir,\n      \".well-known\",\n      \"agents\",\n      \"[agent]\",\n      \"[[...path]]\"\n    );\n\n    const registrySource = this.framework.generateRegistrySource({\n      agents,\n      outputDir: dir,\n    });\n    const routeSource = this.generateNextRouteHandlerSource({\n      registryImportPath: \"./_registry\",\n    });\n\n    return { registrySource, routeSource, dir };\n  }\n\n  /**\n   * Scan for agents, generate source, and write the `_registry.ts` and\n   * `route.ts` files into the Next.js app directory.\n   */\n  writeFiles(): void {\n    const filesToWrite = this.filesToWrite();\n\n    if (!filesToWrite) {\n      console.warn(\"[withAgents] No agents found\");\n      return;\n    }\n\n    const { registrySource, routeSource, dir } = filesToWrite;\n\n    fs.mkdirSync(dir, { recursive: true });\n    fs.writeFileSync(path.join(dir, \"_registry.ts\"), registrySource);\n    fs.writeFileSync(path.join(dir, \"route.ts\"), routeSource);\n\n    const gitignorePath = path.join(dir, \".gitignore\");\n    if (!fs.existsSync(gitignorePath)) {\n      fs.writeFileSync(gitignorePath, \"*\\n\");\n    }\n  }\n\n  private hasAppLayout(dir: string): boolean {\n    return [\"layout.tsx\", \"layout.js\", \"layout.jsx\"].some((f) =>\n      fs.existsSync(path.join(dir, f))\n    );\n  }\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;;;ACAA,SAAoB;AACpB,WAAsB;AAGtB,IAAM,gBAAgB;AACf,SAAS,iBAAiB,MAGtB;AACT,QAAM,UAAU,KAAK,OAAO;AAAA,IAAI,CAAC,UAC/B,YAAY,EAAE,OAAO,WAAW,KAAK,UAAU,CAAC;AAAA,EAClD;AAEA,SAAO;AAAA,IACL;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA,GAAG;AAAA,IACH;AAAA,IACA;AAAA,EACF,EAAE,KAAK,IAAI;AACb;AAEA,SAAS,YAAY,MAGV;AACT,QAAM,SAAS,YAAY,KAAK,KAAK;AACrC,QAAM,YAAY,KAAK,UAAU,QAAQ,MAAM,CAAC,EAC7C,MAAM,IAAI,EACV,IAAI,CAAC,MAAM,MAAO,MAAM,IAAI,OAAO,SAAS,IAAI,EAAG,EACnD,KAAK,IAAI;AAEZ,QAAM,QAAQ,CAAC,eAAe,SAAS,EAAE;AAEzC,MAAI,KAAK,MAAM,gBAAgB;AAC7B,UAAM,aAAa,oBAAoB,KAAK,MAAM,YAAY;AAC9D,UAAM,eAAe,mBAAmB;AAAA,MACtC,MAAM,KAAK;AAAA,MACX,IAAI;AAAA,IACN,CAAC;AACD,UAAM,KAAK,4BAA4B,KAAK,UAAU,YAAY,CAAC,GAAG;AAAA,EACxE;AAEA,SAAO,KAAK,KAAK,UAAU,KAAK,MAAM,IAAI,CAAC;AAAA,EAAQ,MAAM,KAAK,KAAK,CAAC;AAAA;AACtE;AAEA,SAAS,YAAY,OAA8C;AACjE,QAAM,SAAkC,CAAC;AAEzC,MAAI,MAAM,cAAc;AACtB,WAAO,SAAS,MAAM;AAAA,EACxB;AAEA,MAAI,MAAM,OAAO,SAAS,GAAG;AAC3B,WAAO,SAAS,MAAM;AAAA,EACxB;AAEA,SAAO;AACT;AAEA,SAAS,oBAAoB,UAA0B;AACrD,QAAM,aAAa,CAAC,OAAO,OAAO,QAAQ,MAAM;AAChD,aAAW,OAAO,YAAY;AAC5B,UAAM,YAAiB,UAAK,UAAU,QAAQ,GAAG,EAAE;AACnD,QAAO,cAAW,SAAS,GAAG;AAC5B,aAAO;AAAA,IACT;AAAA,EACF;AACA,SAAY,UAAK,UAAU,OAAO;AACpC;AAEA,SAAS,mBAAmB,MAA4C;AACtE,MAAI,MAAW,cAAS,KAAK,MAAM,KAAK,EAAE,EAAE,QAAQ,OAAO,GAAG;AAC9D,QAAM,IAAI,QAAQ,eAAe,EAAE;AACnC,MAAI,CAAC,IAAI,WAAW,GAAG,GAAG;AACxB,UAAM,KAAK,GAAG;AAAA,EAChB;AACA,SAAO;AACT;;;AChFA,IAAAA,MAAoB;AACpB,IAAAC,QAAsB;AAetB,IAAM,qBAAqB,CAAC,YAAY,YAAY,aAAa,WAAW;AAC5E,IAAM,SAAS;AACf,IAAM,SAAS;AAER,SAAS,QAAQ,KAAuB;AAC7C,MAAI,CAAI,eAAW,GAAG,GAAG;AACvB,WAAO,CAAC;AAAA,EACV;AACA,SAAU,gBAAY,GAAG;AAC3B;AAEO,SAAS,SAAS,UAA0B;AACjD,SAAU,iBAAa,UAAU,OAAO;AAC1C;AAEO,SAAS,WAAW,UAA2B;AACpD,SAAU,eAAW,QAAQ;AAC/B;AAEO,SAAS,YAAY,UAA2B;AACrD,SAAU,eAAW,QAAQ,KAAQ,aAAS,QAAQ,EAAE,YAAY;AACtE;AAEO,SAAS,kBAAkB,YAAgC;AAChE,QAAM,UAAU,QAAQ,UAAU;AAClC,QAAM,OAAmB,CAAC;AAE1B,aAAW,SAAS,SAAS;AAC3B,UAAM,eAAoB,WAAK,YAAY,KAAK;AAChD,QAAI,CAAC,YAAY,YAAY,GAAG;AAC9B;AAAA,IACF;AACA,QAAI,MAAM,WAAW,GAAG,GAAG;AACzB;AAAA,IACF;AAEA,SAAK,KAAK;AAAA,MACR,MAAM;AAAA,MACN;AAAA,MACA,gBAAgB,eAAe,YAAY;AAAA,IAC7C,CAAC;AAAA,EACH;AAEA,SAAO;AACT;AAEO,SAAS,eAAe,UAA2B;AACxD,SAAO,mBAAmB,KAAK,CAAC,MAAM,WAAgB,WAAK,UAAU,CAAC,CAAC,CAAC;AAC1E;AAEO,SAAS,UAAU,UAGxB;AACA,QAAM,YAAiB,WAAK,UAAU,QAAQ;AAC9C,QAAM,UAAU,QAAQ,SAAS;AACjC,QAAM,SAAuB,CAAC;AAC9B,QAAM,qBAA+B,CAAC;AAEtC,aAAW,SAAS,QAAQ,KAAK,GAAG;AAClC,UAAM,eAAoB,WAAK,WAAW,KAAK;AAE/C,QAAI,OAAO,KAAK,KAAK,GAAG;AACtB,aAAO,KAAK;AAAA,QACV,MAAM;AAAA,QACN,MAAM;AAAA,MACR,CAA0B;AAAA,IAC5B,WAAW,OAAO,KAAK,KAAK,GAAG;AAC7B,yBAAmB,KAAK,YAAY;AAAA,IACtC;AAAA,EACF;AAEA,SAAO,EAAE,QAAQ,mBAAmB;AACtC;AAEO,SAAS,gBAAgB,UAAsC;AACpE,QAAM,YAAiB,WAAK,UAAU,QAAQ;AAC9C,QAAM,UAAU,QAAQ,SAAS;AACjC,QAAM,UAAU,QAAQ,OAAO,CAAC,MAAM,OAAO,KAAK,CAAC,CAAC,EAAE,KAAK;AAE3D,MAAI,QAAQ,WAAW,GAAG;AACxB,WAAO;AAAA,EACT;AAEA,QAAM,QAAkB,CAAC;AACzB,aAAW,QAAQ,SAAS;AAC1B,UAAM,UAAU,SAAc,WAAK,WAAW,IAAI,CAAC,EAAE,KAAK;AAC1D,QAAI,SAAS;AACX,YAAM,KAAK,OAAO;AAAA,IACpB;AAAA,EACF;AAEA,SAAO,MAAM,SAAS,IAAI,MAAM,KAAK,MAAM,IAAI;AACjD;AAEO,SAAS,UAAU,UAAkC;AAC1D,QAAM,EAAE,QAAQ,mBAAmB,IAAI,UAAU,SAAS,YAAY;AACtE,QAAM,eAAe,gBAAgB,SAAS,YAAY;AAE1D,SAAO,EAAE,GAAG,UAAU,QAAQ,cAAc,mBAAmB;AACjE;AAEO,SAAS,eACd,KACA,sBACe;AACf,MAAI,sBAAsB;AACxB,WAAY,cAAQ,KAAK,oBAAoB;AAAA,EAC/C;AACA,QAAM,aAAa;AAAA,IACZ,WAAK,KAAK,OAAO,QAAQ;AAAA,IACzB,WAAK,KAAK,QAAQ;AAAA,EACzB;AAEA,aAAW,aAAa,YAAY;AAClC,QAAI,YAAY,SAAS,GAAG;AAC1B,aAAO;AAAA,IACT;AAAA,EACF;AAEA,SAAO;AACT;;;AC/HO,IAAM,YAAN,MAAgB;AAAA,EACZ;AAAA,EACA;AAAA,EACA;AAAA,EAET,YAAY,MAcT;AACD,SAAK,MAAM,KAAK,OAAO,QAAQ,IAAI;AACnC,SAAK,YAAY,KAAK;AACtB,SAAK,MAAM,KAAK,QACZ,QAAQ,IAAI,KAAK,SAAS,aAAa,IACvC,MAAM;AAAA,EACZ;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,iBAAiC;AAC/B,UAAM,OAAO,eAAe,KAAK,KAAK,KAAK,SAAS;AACpD,QAAI,CAAC,MAAM;AACT,aAAO,CAAC;AAAA,IACV;AACA,UAAM,OAAO,kBAAkB,IAAI;AACnC,WAAO,KAAK,IAAI,SAAS;AAAA,EAC3B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAUA,yBAAyB;AAC3B;;;AC5DA,IAAAC,MAAoB;AACpB,IAAAC,QAAsB;AA2Cf,IAAM,OAAN,MAAW;AAAA,EAChB,YAA6B,WAAsB;AAAtB;AAAA,EAAuB;AAAA,EAEpD,IAAY,SAAiB;AAC3B,UAAM,MAAM,KAAK,UAAU;AAC3B,UAAM,YAAiB,WAAK,KAAK,OAAO,KAAK;AAC7C,UAAM,aAAkB,WAAK,KAAK,KAAK;AAEvC,QAAI,KAAK,aAAa,SAAS,GAAG;AAChC,aAAO;AAAA,IACT;AACA,QAAI,KAAK,aAAa,UAAU,GAAG;AACjC,aAAO;AAAA,IACT;AACA,QAAO,eAAW,SAAS,GAAG;AAC5B,aAAO;AAAA,IACT;AACA,WAAO;AAAA,EACT;AAAA,EAEA,+BAA+B,MAA8C;AAC3E,WAAO;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,yBAOc,KAAK,UAAU,KAAK,kBAAkB,CAAC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EA+B9D;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,eAIS;AACP,UAAM,SAAS,KAAK,UAAU,eAAe;AAE7C,QAAI,OAAO,WAAW,GAAG;AACvB,aAAO;AAAA,IACT;AAEA,SAAK,UAAU;AAAA,MACb;AAAA,MACA,OAAO,IAAI,CAAC,MAAM,EAAE,IAAI;AAAA,IAC1B;AAEA,UAAM,MAAW;AAAA,MACf,KAAK,UAAU;AAAA,MACf,KAAK;AAAA,MACL;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAEA,UAAM,iBAAiB,KAAK,UAAU,uBAAuB;AAAA,MAC3D;AAAA,MACA,WAAW;AAAA,IACb,CAAC;AACD,UAAM,cAAc,KAAK,+BAA+B;AAAA,MACtD,oBAAoB;AAAA,IACtB,CAAC;AAED,WAAO,EAAE,gBAAgB,aAAa,IAAI;AAAA,EAC5C;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,aAAmB;AACjB,UAAM,eAAe,KAAK,aAAa;AAEvC,QAAI,CAAC,cAAc;AACjB,cAAQ,KAAK,8BAA8B;AAC3C;AAAA,IACF;AAEA,UAAM,EAAE,gBAAgB,aAAa,IAAI,IAAI;AAE7C,IAAG,cAAU,KAAK,EAAE,WAAW,KAAK,CAAC;AACrC,IAAG,kBAAmB,WAAK,KAAK,cAAc,GAAG,cAAc;AAC/D,IAAG,kBAAmB,WAAK,KAAK,UAAU,GAAG,WAAW;AAExD,UAAM,gBAAqB,WAAK,KAAK,YAAY;AACjD,QAAI,CAAI,eAAW,aAAa,GAAG;AACjC,MAAG,kBAAc,eAAe,KAAK;AAAA,IACvC;AAAA,EACF;AAAA,EAEQ,aAAa,KAAsB;AACzC,WAAO,CAAC,cAAc,aAAa,YAAY,EAAE;AAAA,MAAK,CAAC,MAClD,eAAgB,WAAK,KAAK,CAAC,CAAC;AAAA,IACjC;AAAA,EACF;AACF;;;AJxJO,SAAS,WACd,QACA,eAGI,CAAC,GACY;AACjB,QAAM,YAAY,IAAI,UAAU,YAAY;AAC5C,QAAM,OAAO,IAAI,KAAK,SAAS;AAC/B,OAAK,WAAW;AAEhB,SAAO;AACT;",
  "names": ["fs", "path", "fs", "path"]
}

@@ -0,0 +1,137 @@
1
+ import {
2
+ Framework
3
+ } from "../chunk-DO4HKPHR.mjs";
4
+ import "../chunk-BJTO5JO5.mjs";
5
+
6
+ // src/framework-next/next.ts
7
+ import * as fs from "fs";
8
+ import * as path from "path";
9
+ var Next = class {
10
+ constructor(framework) {
11
+ this.framework = framework;
12
+ }
13
+ get appDir() {
14
+ const cwd = this.framework.cwd;
15
+ const srcAppDir = path.join(cwd, "src", "app");
16
+ const rootAppDir = path.join(cwd, "app");
17
+ if (this.hasAppLayout(srcAppDir)) {
18
+ return "src/app";
19
+ }
20
+ if (this.hasAppLayout(rootAppDir)) {
21
+ return "app";
22
+ }
23
+ if (fs.existsSync(srcAppDir)) {
24
+ return "src/app";
25
+ }
26
+ return "app";
27
+ }
28
+ generateNextRouteHandlerSource(opts) {
29
+ return `// Auto-generated by withAgents - do not edit
30
+ // @ts-nocheck
31
+ // biome-ignore format, lint, lint/style, lint/suspicious, lint/complexity, lint/performance, lint/security, lint/correctness, lint/strict
32
+ // prettier-ignore
33
+
34
+ import { runtime } from "experimental-agent/framework-runtime";
35
+ import { handleRequest } from "experimental-agent";
36
+ import { agents } from ${JSON.stringify(opts.registryImportPath)};
37
+
38
+ async function workflow(agent, sessionId, ...args) {
39
+ "use workflow";
40
+ return agent.session(sessionId).send(...args);
41
+ }
42
+
43
+ async function handler(
44
+ req: Request,
45
+ ctx: { params: Promise<{ agent: string; path?: string[] }> }
46
+ ) {
47
+ const params = await ctx.params;
48
+ const name = params.agent;
49
+ const instance = await runtime.resolveAgent({ name, agents });
50
+
51
+ if (!instance) {
52
+ const available = Object.keys(agents).join(", ");
53
+ const body =
54
+ \`Agent "\${name}" not found.\\n\\n\` +
55
+ \`Available agents: \${available}\\n\\n\` +
56
+ \`hint: each folder inside agents/ becomes an agent.\\n\` +
57
+ \` Create agents/\${name}/ to add this agent.\\n\`;
58
+ return new Response(body, { status: 404 });
59
+ }
60
+
61
+ return handleRequest(instance, { basePath: \`/.well-known/agents/\${name}\`, workflow }).fetch(req);
62
+ }
63
+
64
+ export const GET = handler;
65
+ export const POST = handler;
66
+ `;
67
+ }
68
+ /**
69
+ * Pure codegen step: scan agents and return the generated source
70
+ * strings without touching the filesystem. Useful for adapters that
71
+ * need to control where and how files are written.
72
+ *
73
+ * Returns `null` when no agents are found.
74
+ */
75
+ filesToWrite() {
76
+ const agents = this.framework.discoverAgents();
77
+ if (agents.length === 0) {
78
+ return null;
79
+ }
80
+ this.framework.log(
81
+ "discovered agents:",
82
+ agents.map((a) => a.name)
83
+ );
84
+ const dir = path.join(
85
+ this.framework.cwd,
86
+ this.appDir,
87
+ ".well-known",
88
+ "agents",
89
+ "[agent]",
90
+ "[[...path]]"
91
+ );
92
+ const registrySource = this.framework.generateRegistrySource({
93
+ agents,
94
+ outputDir: dir
95
+ });
96
+ const routeSource = this.generateNextRouteHandlerSource({
97
+ registryImportPath: "./_registry"
98
+ });
99
+ return { registrySource, routeSource, dir };
100
+ }
101
+ /**
102
+ * Scan for agents, generate source, and write the `_registry.ts` and
103
+ * `route.ts` files into the Next.js app directory.
104
+ */
105
+ writeFiles() {
106
+ const filesToWrite = this.filesToWrite();
107
+ if (!filesToWrite) {
108
+ console.warn("[withAgents] No agents found");
109
+ return;
110
+ }
111
+ const { registrySource, routeSource, dir } = filesToWrite;
112
+ fs.mkdirSync(dir, { recursive: true });
113
+ fs.writeFileSync(path.join(dir, "_registry.ts"), registrySource);
114
+ fs.writeFileSync(path.join(dir, "route.ts"), routeSource);
115
+ const gitignorePath = path.join(dir, ".gitignore");
116
+ if (!fs.existsSync(gitignorePath)) {
117
+ fs.writeFileSync(gitignorePath, "*\n");
118
+ }
119
+ }
120
+ hasAppLayout(dir) {
121
+ return ["layout.tsx", "layout.js", "layout.jsx"].some(
122
+ (f) => fs.existsSync(path.join(dir, f))
123
+ );
124
+ }
125
+ };
126
+
127
+ // src/framework-next/agents-plugin.ts
128
+ function withAgents(config, agentsConfig = {}) {
129
+ const framework = new Framework(agentsConfig);
130
+ const next = new Next(framework);
131
+ next.writeFiles();
132
+ return config;
133
+ }
134
+ export {
135
+ withAgents
136
+ };
137
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/framework-next/next.ts", "../../src/framework-next/agents-plugin.ts"],
  "sourcesContent": ["import * as fs from \"node:fs\";\nimport * as path from \"node:path\";\nimport type { NextConfig } from \"next\";\nimport type { Framework } from \"../framework/framework\";\n\n/** A Next.js config object or async config function. */\nexport type NextConfigInput =\n  | NextConfig\n  | ((\n      phase: string,\n      ctx: { defaultConfig: NextConfig }\n    ) => Promise<NextConfig>);\n\n/** Options for {@link withAgents}. */\nexport type WithAgentsConfig = {\n  /**\n   * Path to the agents directory, relative to the project root.\n   * When omitted, auto-detected as the first existing directory\n   * among `src/agents` and `agents`.\n   */\n  agentsDir?: string;\n  /** Enable verbose logging to stdout. */\n  debug?: boolean;\n};\n\n/**\n * Next.js adapter for filesystem-based agent routing.\n *\n * Receives a {@link Framework} instance (composition) and uses it to\n * discover agents, generate a registry, and write the Next.js route\n * handler files that serve them.\n *\n * Most users should use the {@link withAgents} helper instead of\n * constructing this class directly.\n *\n * @example\n * ```ts\n * import { Framework } from \"experimental-agent/framework\";\n * import { Next } from \"experimental-agent/next/agents\";\n *\n * const framework = new Framework({ cwd: process.cwd() });\n * new Next(framework).generate();\n * ```\n */\nexport class Next {\n  constructor(private readonly framework: Framework) {}\n\n  private get appDir(): string {\n    const cwd = this.framework.cwd;\n    const srcAppDir = path.join(cwd, \"src\", \"app\");\n    const rootAppDir = path.join(cwd, \"app\");\n\n    if (this.hasAppLayout(srcAppDir)) {\n      return \"src/app\";\n    }\n    if (this.hasAppLayout(rootAppDir)) {\n      return \"app\";\n    }\n    if (fs.existsSync(srcAppDir)) {\n      return \"src/app\";\n    }\n    return \"app\";\n  }\n\n  generateNextRouteHandlerSource(opts: { registryImportPath: string }): string {\n    return `// Auto-generated by withAgents - do not edit\n// @ts-nocheck\n// biome-ignore format, lint, lint/style, lint/suspicious, lint/complexity, lint/performance, lint/security, lint/correctness, lint/strict\n// prettier-ignore\n\nimport { runtime } from \"experimental-agent/framework-runtime\";\nimport { handleRequest } from \"experimental-agent\";\nimport { agents } from ${JSON.stringify(opts.registryImportPath)};\n\nasync function workflow(agent, sessionId, ...args) {\n  \"use workflow\";\n  return agent.session(sessionId).send(...args);\n}\n\nasync function handler(\n  req: Request,\n  ctx: { params: Promise<{ agent: string; path?: string[] }> }\n) {\n  const params = await ctx.params;\n  const name = params.agent;\n  const instance = await runtime.resolveAgent({ name, agents });\n\n  if (!instance) {\n    const available = Object.keys(agents).join(\", \");\n    const body =\n      \\`Agent \"\\${name}\" not found.\\\\n\\\\n\\` +\n      \\`Available agents: \\${available}\\\\n\\\\n\\` +\n      \\`hint: each folder inside agents/ becomes an agent.\\\\n\\` +\n      \\`      Create agents/\\${name}/ to add this agent.\\\\n\\`;\n    return new Response(body, { status: 404 });\n  }\n\n  return handleRequest(instance, { basePath: \\`/.well-known/agents/\\${name}\\`, workflow }).fetch(req);\n}\n\nexport const GET = handler;\nexport const POST = handler;\n`;\n  }\n\n  /**\n   * Pure codegen step: scan agents and return the generated source\n   * strings without touching the filesystem. Useful for adapters that\n   * need to control where and how files are written.\n   *\n   * Returns `null` when no agents are found.\n   */\n  filesToWrite(): {\n    registrySource: string;\n    routeSource: string;\n    dir: string;\n  } | null {\n    const agents = this.framework.discoverAgents();\n\n    if (agents.length === 0) {\n      return null;\n    }\n\n    this.framework.log(\n      \"discovered agents:\",\n      agents.map((a) => a.name)\n    );\n\n    const dir = path.join(\n      this.framework.cwd,\n      this.appDir,\n      \".well-known\",\n      \"agents\",\n      \"[agent]\",\n      \"[[...path]]\"\n    );\n\n    const registrySource = this.framework.generateRegistrySource({\n      agents,\n      outputDir: dir,\n    });\n    const routeSource = this.generateNextRouteHandlerSource({\n      registryImportPath: \"./_registry\",\n    });\n\n    return { registrySource, routeSource, dir };\n  }\n\n  /**\n   * Scan for agents, generate source, and write the `_registry.ts` and\n   * `route.ts` files into the Next.js app directory.\n   */\n  writeFiles(): void {\n    const filesToWrite = this.filesToWrite();\n\n    if (!filesToWrite) {\n      console.warn(\"[withAgents] No agents found\");\n      return;\n    }\n\n    const { registrySource, routeSource, dir } = filesToWrite;\n\n    fs.mkdirSync(dir, { recursive: true });\n    fs.writeFileSync(path.join(dir, \"_registry.ts\"), registrySource);\n    fs.writeFileSync(path.join(dir, \"route.ts\"), routeSource);\n\n    const gitignorePath = path.join(dir, \".gitignore\");\n    if (!fs.existsSync(gitignorePath)) {\n      fs.writeFileSync(gitignorePath, \"*\\n\");\n    }\n  }\n\n  private hasAppLayout(dir: string): boolean {\n    return [\"layout.tsx\", \"layout.js\", \"layout.jsx\"].some((f) =>\n      fs.existsSync(path.join(dir, f))\n    );\n  }\n}\n", "import { Framework } from \"../framework/framework\";\nimport { Next, type NextConfigInput } from \"./next\";\n\nexport type { NextConfigInput, WithAgentsConfig } from \"./next\";\n\n/**\n * Next.js config wrapper that discovers agents on the filesystem and\n * generates the route handler + registry files needed to serve them.\n *\n * Call this in `next.config.ts`:\n *\n * @example\n * ```ts\n * import { withAgents } from \"experimental-agent/next/agents\";\n *\n * export default withAgents({ reactStrictMode: true });\n * ```\n *\n * Debug logs:\n * ```ts\n * import { withAgents } from \"experimental-agent/next/agents\";\n *\n * export default withAgents({ reactStrictMode: true }, { debug: true });\n * ```\n */\nexport function withAgents(\n  config: NextConfigInput,\n  agentsConfig: Pick<\n    ConstructorParameters<typeof Framework>[0],\n    \"agentsDir\" | \"debug\"\n  > = {}\n): NextConfigInput {\n  const framework = new Framework(agentsConfig);\n  const next = new Next(framework);\n  next.writeFiles();\n\n  return config;\n}\n"],
  "mappings": ";;;;;;AAAA,YAAY,QAAQ;AACpB,YAAY,UAAU;AA2Cf,IAAM,OAAN,MAAW;AAAA,EAChB,YAA6B,WAAsB;AAAtB;AAAA,EAAuB;AAAA,EAEpD,IAAY,SAAiB;AAC3B,UAAM,MAAM,KAAK,UAAU;AAC3B,UAAM,YAAiB,UAAK,KAAK,OAAO,KAAK;AAC7C,UAAM,aAAkB,UAAK,KAAK,KAAK;AAEvC,QAAI,KAAK,aAAa,SAAS,GAAG;AAChC,aAAO;AAAA,IACT;AACA,QAAI,KAAK,aAAa,UAAU,GAAG;AACjC,aAAO;AAAA,IACT;AACA,QAAO,cAAW,SAAS,GAAG;AAC5B,aAAO;AAAA,IACT;AACA,WAAO;AAAA,EACT;AAAA,EAEA,+BAA+B,MAA8C;AAC3E,WAAO;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,yBAOc,KAAK,UAAU,KAAK,kBAAkB,CAAC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EA+B9D;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,eAIS;AACP,UAAM,SAAS,KAAK,UAAU,eAAe;AAE7C,QAAI,OAAO,WAAW,GAAG;AACvB,aAAO;AAAA,IACT;AAEA,SAAK,UAAU;AAAA,MACb;AAAA,MACA,OAAO,IAAI,CAAC,MAAM,EAAE,IAAI;AAAA,IAC1B;AAEA,UAAM,MAAW;AAAA,MACf,KAAK,UAAU;AAAA,MACf,KAAK;AAAA,MACL;AAAA,MACA;AAAA,MACA;AAAA,MACA;AAAA,IACF;AAEA,UAAM,iBAAiB,KAAK,UAAU,uBAAuB;AAAA,MAC3D;AAAA,MACA,WAAW;AAAA,IACb,CAAC;AACD,UAAM,cAAc,KAAK,+BAA+B;AAAA,MACtD,oBAAoB;AAAA,IACtB,CAAC;AAED,WAAO,EAAE,gBAAgB,aAAa,IAAI;AAAA,EAC5C;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,aAAmB;AACjB,UAAM,eAAe,KAAK,aAAa;AAEvC,QAAI,CAAC,cAAc;AACjB,cAAQ,KAAK,8BAA8B;AAC3C;AAAA,IACF;AAEA,UAAM,EAAE,gBAAgB,aAAa,IAAI,IAAI;AAE7C,IAAG,aAAU,KAAK,EAAE,WAAW,KAAK,CAAC;AACrC,IAAG,iBAAmB,UAAK,KAAK,cAAc,GAAG,cAAc;AAC/D,IAAG,iBAAmB,UAAK,KAAK,UAAU,GAAG,WAAW;AAExD,UAAM,gBAAqB,UAAK,KAAK,YAAY;AACjD,QAAI,CAAI,cAAW,aAAa,GAAG;AACjC,MAAG,iBAAc,eAAe,KAAK;AAAA,IACvC;AAAA,EACF;AAAA,EAEQ,aAAa,KAAsB;AACzC,WAAO,CAAC,cAAc,aAAa,YAAY,EAAE;AAAA,MAAK,CAAC,MAClD,cAAgB,UAAK,KAAK,CAAC,CAAC;AAAA,IACjC;AAAA,EACF;AACF;;;ACxJO,SAAS,WACd,QACA,eAGI,CAAC,GACY;AACjB,QAAM,YAAY,IAAI,UAAU,YAAY;AAC5C,QAAM,OAAO,IAAI,KAAK,SAAS;AAC/B,OAAK,WAAW;AAEhB,SAAO;AACT;",
  "names": []
}
