experimental-agent 0.8.0-alpha.2 → 0.8.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/dist/next.mjs CHANGED
@@ -1,6 +1,6 @@
1
1
  import {
2
2
  package_default
3
- } from "./chunk-LMYEJDMD.mjs";
3
+ } from "./chunk-5N2UOGXY.mjs";
4
4
  import {
5
5
  __require
6
6
  } from "./chunk-BJTO5JO5.mjs";
@@ -59,7 +59,9 @@ function detectAgents(cwd, debug) {
59
59
  const content = fs.readFileSync(fullPath, "utf-8");
60
60
  if (content.includes(PACKAGE_NAME) && content.includes("agent(")) {
61
61
  const stripped = content.replace(/\/\/.*$/gm, "").replace(/\/\*[\s\S]*?\*\//g, "");
62
- const exportMatches = stripped.matchAll(AGENT_EXPORT_REGEX);
62
+ const exportMatches = Array.from(
63
+ stripped.matchAll(AGENT_EXPORT_REGEX)
64
+ );
63
65
  const seen = /* @__PURE__ */ new Set();
64
66
  for (const match of exportMatches) {
65
67
  if (seen.has(match[1])) {
@@ -244,4 +246,4 @@ function withAgent(configOrFn, agentConfig) {
244
246
  export {
245
247
  withAgent
246
248
  };
247
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/next/index.ts"],
  "sourcesContent": ["import * as crypto from \"node:crypto\";\nimport * as fs from \"node:fs\";\nimport * as path from \"node:path\";\nimport type { NextConfig } from \"next\";\nimport pkg from \"../../package.json\";\n\nexport type WithAgentConfig = {\n  /** Enable debug logging for the Next.js loader (agent detection, file generation). */\n  debug?: boolean;\n};\n\nconst PACKAGE_NAME = pkg.name;\nconst AGENT_PROTOCOL_VERSION = \"v1\";\nconst FILE_EXT_REGEX = /\\.(ts|tsx|js|jsx)$/;\nconst AGENT_EXPORT_REGEX = /export\\s+const\\s+(\\w+)\\s*=\\s*agent\\s*\\(/g;\nconst DEFAULT_WORKFLOW_DIRS = [\"pages\", \"app\", \"src/pages\", \"src/app\"];\n\ntype DetectedAgent = {\n  exportName: string;\n  filePath: string;\n  /** Unique identifier: exportName or exportName_hash if duplicates exist */\n  uniqueId: string;\n};\n\nfunction shortHash(str: string): string {\n  let hash = 0;\n  for (let i = 0; i < str.length; i++) {\n    const char = str.charCodeAt(i);\n    // biome-ignore lint/suspicious/noBitwiseOperators: intentional hash computation\n    hash = (hash << 5) - hash + char;\n    // biome-ignore lint/suspicious/noBitwiseOperators: convert to 32-bit integer\n    hash &= hash;\n  }\n  return Math.abs(hash).toString(36).slice(0, 6);\n}\n\nfunction hasNextAppLayout(dir: string): boolean {\n  return [\"layout.tsx\", \"layout.ts\", \"layout.js\", \"layout.jsx\"].some((f) =>\n    fs.existsSync(path.join(dir, f))\n  );\n}\n\n/**\n * Detects whether the project uses src/app or app directory structure\n * by checking which directory contains a Next.js layout file.\n * Returns the app directory path relative to cwd.\n */\nfunction getAppDir(cwd: string): string {\n  const srcAppDir = path.join(cwd, \"src\", \"app\");\n  const rootAppDir = path.join(cwd, \"app\");\n\n  if (hasNextAppLayout(srcAppDir)) {\n    return \"src/app\";\n  }\n  if (hasNextAppLayout(rootAppDir)) {\n    return \"app\";\n  }\n\n  /**\n   * Fallback when neither directory has a layout file yet\n   * (e.g. fresh project). Prefer src/app if the directory exists.\n   */\n  if (fs.existsSync(srcAppDir)) {\n    return \"src/app\";\n  }\n  return \"app\";\n}\n\n/**\n * Scans for agent exports in the project.\n */\nfunction detectAgents(cwd: string, debug: boolean): DetectedAgent[] {\n  const agents: DetectedAgent[] = [];\n\n  const dirsToScan = [path.join(cwd, \"src\"), cwd];\n\n  function scanDir(dir: string) {\n    if (!fs.existsSync(dir)) {\n      return;\n    }\n\n    const entries = fs.readdirSync(dir, { withFileTypes: true });\n    for (const entry of entries) {\n      const fullPath = path.join(dir, entry.name);\n      if (\n        entry.isDirectory() &&\n        !entry.name.startsWith(\".\") &&\n        entry.name !== \"node_modules\" &&\n        entry.name !== \"src\"\n      ) {\n        scanDir(fullPath);\n      } else if (entry.isFile() && FILE_EXT_REGEX.test(entry.name)) {\n        try {\n          const content = fs.readFileSync(fullPath, \"utf-8\");\n          if (content.includes(PACKAGE_NAME) && content.includes(\"agent(\")) {\n            // Strip single-line comments and block comments to avoid\n            // matching commented-out agent exports\n            const stripped = content\n              .replace(/\\/\\/.*$/gm, \"\")\n              .replace(/\\/\\*[\\s\\S]*?\\*\\//g, \"\");\n            const exportMatches = stripped.matchAll(AGENT_EXPORT_REGEX);\n            const seen = new Set<string>();\n            for (const match of exportMatches) {\n              if (seen.has(match[1])) {\n                continue;\n              }\n              seen.add(match[1]);\n              agents.push({\n                exportName: match[1],\n                filePath: fullPath,\n                uniqueId: \"\", // Will be set after all agents are found\n              });\n              if (debug) {\n                console.log(\n                  `[withAgent] Found agent: ${match[1]} in ${fullPath}`\n                );\n              }\n            }\n          }\n        } catch {\n          // Ignore read errors\n        }\n      }\n    }\n  }\n\n  for (const dir of dirsToScan) {\n    scanDir(dir);\n  }\n\n  // Assign unique IDs - add hash suffix if duplicate export names exist\n  const nameCounts = new Map<string, number>();\n  for (const agent of agents) {\n    nameCounts.set(\n      agent.exportName,\n      (nameCounts.get(agent.exportName) || 0) + 1\n    );\n  }\n\n  for (const agent of agents) {\n    const count = nameCounts.get(agent.exportName) ?? 0;\n    if (count > 1) {\n      agent.uniqueId = `${agent.exportName}_${shortHash(agent.filePath)}`;\n    } else {\n      agent.uniqueId = agent.exportName;\n    }\n  }\n\n  return agents;\n}\n\n/**\n * Generates all agent files in .well-known/agent/v1/:\n * - steps.js: RPC step functions with \"use step\"\n * - route.js: triggers workflow discovery\n */\nfunction generateAgentFiles(\n  agents: DetectedAgent[],\n  outputDir: string,\n  _cwd: string,\n  debug: boolean\n): void {\n  if (agents.length === 0) {\n    if (debug) {\n      console.log(\"[withAgent] No agents found, skipping generation\");\n    }\n    return;\n  }\n\n  if (!fs.existsSync(outputDir)) {\n    fs.mkdirSync(outputDir, { recursive: true });\n  }\n\n  /**\n   * Generate step functions that use dynamic import.\n   * Dynamic import ensures the agent module loads (including loader-injected init)\n   * when the step runs in workflow context, avoiding the \"not initialized\" error.\n   *\n   * Uses uniqueId for function names to handle duplicate export names across files.\n   */\n  const rpcFunctions = agents\n    .map((a) => {\n      // Calculate relative path from steps.js to the agent file\n      let agentRelPath = path\n        .relative(outputDir, a.filePath)\n        .replace(/\\\\/g, \"/\");\n\n      if (!agentRelPath.startsWith(\".\")) {\n        agentRelPath = `./${agentRelPath}`;\n      }\n\n      return `\nexport const ${a.uniqueId}Rpc = async (params) => {\n  \"use step\";\n  const { ${a.exportName} } = await import(\"${agentRelPath}\");\n  const res = await ${a.exportName}.handler(params);\n  if (res instanceof Response) {\n    return res.json();\n  }\n  return res;\n};`;\n    })\n    .join(\"\\n\");\n\n  /**\n   * Init functions wire up agent.rpc in the main app context.\n   * Uses exportName for the function name (loader matches by export name),\n   * but references uniqueId for the Rpc function.\n   */\n  const initFunctions = agents\n    .map(\n      (a) => `\nexport function __init${a.exportName[0].toUpperCase()}${a.exportName.slice(\n        1\n      )}(agent) {\n  agent.rpc = ${a.uniqueId}Rpc;\n}`\n    )\n    .join(\"\\n\");\n\n  const stepsContent = `// Auto-generated by withAgent - do not edit\n${rpcFunctions}\n${initFunctions}\n`;\n\n  const stepsPath = path.join(outputDir, \"steps.js\");\n  fs.writeFileSync(stepsPath, stepsContent);\n\n  /**\n   * Generate route.js that imports steps.js.\n   * This ensures workflow discovers the \"use step\" functions.\n   * No actual handler needed - storage RPC uses the step function directly.\n   */\n  const routeContent = `// Auto-generated by withAgent - triggers workflow discovery\nimport \"./steps.js\";\nexport function GET() {\n  return new Response(\"ok\");\n}\n`;\n\n  const routePath = path.join(outputDir, \"route.js\");\n  fs.writeFileSync(routePath, routeContent);\n\n  // Add .gitignore\n  const gitignorePath = path.join(outputDir, \".gitignore\");\n  if (!fs.existsSync(gitignorePath)) {\n    fs.writeFileSync(gitignorePath, \"*\\n\");\n  }\n\n  if (debug) {\n    console.log(`[withAgent] Generated agent files at ${outputDir}`);\n  }\n}\n\ntype TurbopackLoader =\n  | string\n  | { loader: string; options?: Record<string, unknown> };\n\ntype TurbopackRule = {\n  loaders?: TurbopackLoader[];\n  condition?: {\n    all?: unknown[];\n    any?: unknown[];\n    path?: RegExp;\n    content?: RegExp;\n    not?: unknown;\n  };\n};\n\ntype WorkflowConfigFn = (\n  phase: string,\n  ctx: { defaultConfig: NextConfig }\n) => Promise<NextConfig>;\n\ntype NextConfigInput =\n  | NextConfig\n  | ((\n      phase: string,\n      ctx: { defaultConfig: NextConfig }\n    ) => Promise<NextConfig>);\n\n/**\n * Next.js plugin that configures agent support.\n *\n * - Detects agents and generates RPC steps with \"use step\" in .well-known/agent/v1/\n * - Generates AGENT_SECRET for secure RPC calls\n * - Wraps withWorkflow for workflow support\n *\n * @example\n * ```ts\n * // next.config.ts\n * import { withAgent } from \"experimental-agent/next\";\n *\n * // Option 1: withAgent handles everything (recommended)\n * export default withAgent({});\n *\n * // Option 2: Compose with other plugins\n * export default withAgent(withSomeOtherPlugin({}));\n * ```\n */\nexport function withAgent(\n  configOrFn: NextConfigInput,\n  agentConfig?: WithAgentConfig\n): WorkflowConfigFn {\n  const debug = agentConfig?.debug ?? false;\n  const cwd = process.cwd();\n\n  // Detect app directory structure (src/app vs app)\n  const appDir = getAppDir(cwd);\n  const agentDir = `${appDir}/.well-known/agent/${AGENT_PROTOCOL_VERSION}`;\n\n  // Detect agents and generate files FIRST (before workflow discovers them)\n  const agents = detectAgents(cwd, debug);\n  const outputDir = path.join(cwd, agentDir);\n  generateAgentFiles(agents, outputDir, cwd, debug);\n\n  const agentFilePaths = agents.map((a) => a.filePath);\n  const loaderPath = require.resolve(\"./next/loader\");\n  const absoluteAppDir = path.join(cwd, appDir);\n  const loaderOptions = { appDir: absoluteAppDir, debug };\n\n  if (debug) {\n    console.log(\"[withAgent] loader path:\", loaderPath);\n    console.log(\"[withAgent] agent files:\", agentFilePaths);\n    console.log(\"[withAgent] agent dir:\", agentDir);\n  }\n\n  // Build regex to match agent file paths\n  const agentPathRegex =\n    agentFilePaths.length > 0\n      ? new RegExp(\n          `${agentFilePaths\n            .map((p) =>\n              path\n                .relative(cwd, p)\n                .replace(/\\\\/g, \"/\")\n                .replace(/[.*+?^${}()|[\\]\\\\]/g, \"\\\\$&\")\n            )\n            .join(\"|\")}$`\n        )\n      : null;\n\n  /**\n   * Always pass through withWorkflow - it handles both NextConfig and async functions.\n   * This ensures workflow is always configured, regardless of what the user passed.\n   */\n  let withWorkflow: (\n    config: NextConfigInput,\n    options?: { workflows?: { dirs?: string[] } }\n  ) => WorkflowConfigFn;\n  try {\n    withWorkflow = require(\"workflow/next\").withWorkflow;\n  } catch {\n    console.warn(\n      \"[withAgent] workflow/next not found. Make sure workflow is installed.\"\n    );\n    return typeof configOrFn === \"function\"\n      ? configOrFn\n      : async () => configOrFn;\n  }\n\n  const baseConfigFn = withWorkflow(configOrFn, {\n    workflows: {\n      dirs: [...DEFAULT_WORKFLOW_DIRS, agentDir],\n    },\n  });\n\n  /**\n   * Return a wrapper that runs AFTER workflow processes, then adds our loader.\n   * This is necessary because workflow overwrites turbopack rules - we need to\n   * modify them after workflow is done.\n   */\n  return async (phase, ctx) => {\n    const nextConfig = await baseConfigFn(phase, ctx);\n\n    // Set up AGENT_SECRET if not already set\n    if (!process.env.AGENT_SECRET) {\n      nextConfig.env = {\n        ...nextConfig.env,\n        AGENT_SECRET: crypto.randomUUID(),\n      };\n      if (debug) {\n        console.log(\"[withAgent] Generated AGENT_SECRET\");\n      }\n    }\n\n    if (!agentPathRegex || agentFilePaths.length === 0) {\n      return nextConfig;\n    }\n\n    // Modify turbopack rules to include our loader for agent files\n    const rules = (nextConfig.turbopack?.rules || {}) as Record<\n      string,\n      TurbopackRule\n    >;\n\n    for (const key of [\"*.ts\", \"*.tsx\", \"*.js\", \"*.jsx\"]) {\n      const existingRule = rules[key];\n      if (!existingRule) {\n        continue;\n      }\n\n      /**\n       * Workflow's rule has: condition.all = [{ not: path }, { content: /use step|.../ }]\n       * We want: run loader if workflow's conditions match OR it's an agent file.\n       * So we wrap in: condition.any = [existingCondition, { path: agentPathRegex }]\n       */\n      rules[key] = {\n        ...existingRule,\n        loaders: [\n          { loader: loaderPath, options: loaderOptions },\n          ...(existingRule.loaders || []),\n        ],\n        condition: {\n          any: [existingRule.condition, { path: agentPathRegex }],\n        },\n      };\n    }\n\n    // Also configure webpack for non-turbopack builds\n    const existingWebpack = nextConfig.webpack;\n    nextConfig.webpack = (webpackConfig, context) => {\n      webpackConfig.module.rules.push({\n        test: (resourcePath: string) => agentFilePaths.includes(resourcePath),\n        use: [{ loader: loaderPath, options: loaderOptions }],\n      });\n      return existingWebpack?.(webpackConfig, context) ?? webpackConfig;\n    };\n\n    if (debug) {\n      console.log(\"[withAgent] Added agent loader to turbopack rules\");\n    }\n\n    return nextConfig;\n  };\n}\n"],
  "mappings": "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",
  "names": []
}

249
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../src/next/index.ts"],
  "sourcesContent": ["import * as crypto from \"node:crypto\";\nimport * as fs from \"node:fs\";\nimport * as path from \"node:path\";\nimport type { NextConfig } from \"next\";\nimport pkg from \"../../package.json\";\n\nexport type WithAgentConfig = {\n  /** Enable debug logging for the Next.js loader (agent detection, file generation). */\n  debug?: boolean;\n};\n\nconst PACKAGE_NAME = pkg.name;\nconst AGENT_PROTOCOL_VERSION = \"v1\";\nconst FILE_EXT_REGEX = /\\.(ts|tsx|js|jsx)$/;\nconst AGENT_EXPORT_REGEX = /export\\s+const\\s+(\\w+)\\s*=\\s*agent\\s*\\(/g;\nconst DEFAULT_WORKFLOW_DIRS = [\"pages\", \"app\", \"src/pages\", \"src/app\"];\n\ntype DetectedAgent = {\n  exportName: string;\n  filePath: string;\n  /** Unique identifier: exportName or exportName_hash if duplicates exist */\n  uniqueId: string;\n};\n\nfunction shortHash(str: string): string {\n  let hash = 0;\n  for (let i = 0; i < str.length; i++) {\n    const char = str.charCodeAt(i);\n    // biome-ignore lint/suspicious/noBitwiseOperators: intentional hash computation\n    hash = (hash << 5) - hash + char;\n    // biome-ignore lint/suspicious/noBitwiseOperators: convert to 32-bit integer\n    hash &= hash;\n  }\n  return Math.abs(hash).toString(36).slice(0, 6);\n}\n\nfunction hasNextAppLayout(dir: string): boolean {\n  return [\"layout.tsx\", \"layout.ts\", \"layout.js\", \"layout.jsx\"].some((f) =>\n    fs.existsSync(path.join(dir, f))\n  );\n}\n\n/**\n * Detects whether the project uses src/app or app directory structure\n * by checking which directory contains a Next.js layout file.\n * Returns the app directory path relative to cwd.\n */\nfunction getAppDir(cwd: string): string {\n  const srcAppDir = path.join(cwd, \"src\", \"app\");\n  const rootAppDir = path.join(cwd, \"app\");\n\n  if (hasNextAppLayout(srcAppDir)) {\n    return \"src/app\";\n  }\n  if (hasNextAppLayout(rootAppDir)) {\n    return \"app\";\n  }\n\n  /**\n   * Fallback when neither directory has a layout file yet\n   * (e.g. fresh project). Prefer src/app if the directory exists.\n   */\n  if (fs.existsSync(srcAppDir)) {\n    return \"src/app\";\n  }\n  return \"app\";\n}\n\n/**\n * Scans for agent exports in the project.\n */\nfunction detectAgents(cwd: string, debug: boolean): DetectedAgent[] {\n  const agents: DetectedAgent[] = [];\n\n  const dirsToScan = [path.join(cwd, \"src\"), cwd];\n\n  function scanDir(dir: string) {\n    if (!fs.existsSync(dir)) {\n      return;\n    }\n\n    const entries = fs.readdirSync(dir, { withFileTypes: true });\n    for (const entry of entries) {\n      const fullPath = path.join(dir, entry.name);\n      if (\n        entry.isDirectory() &&\n        !entry.name.startsWith(\".\") &&\n        entry.name !== \"node_modules\" &&\n        entry.name !== \"src\"\n      ) {\n        scanDir(fullPath);\n      } else if (entry.isFile() && FILE_EXT_REGEX.test(entry.name)) {\n        try {\n          const content = fs.readFileSync(fullPath, \"utf-8\");\n          if (content.includes(PACKAGE_NAME) && content.includes(\"agent(\")) {\n            // Strip single-line comments and block comments to avoid\n            // matching commented-out agent exports\n            const stripped = content\n              .replace(/\\/\\/.*$/gm, \"\")\n              .replace(/\\/\\*[\\s\\S]*?\\*\\//g, \"\");\n            const exportMatches = Array.from(\n              stripped.matchAll(AGENT_EXPORT_REGEX)\n            );\n            const seen = new Set<string>();\n            for (const match of exportMatches) {\n              if (seen.has(match[1])) {\n                continue;\n              }\n              seen.add(match[1]);\n              agents.push({\n                exportName: match[1],\n                filePath: fullPath,\n                uniqueId: \"\", // Will be set after all agents are found\n              });\n              if (debug) {\n                console.log(\n                  `[withAgent] Found agent: ${match[1]} in ${fullPath}`\n                );\n              }\n            }\n          }\n        } catch {\n          // Ignore read errors\n        }\n      }\n    }\n  }\n\n  for (const dir of dirsToScan) {\n    scanDir(dir);\n  }\n\n  // Assign unique IDs - add hash suffix if duplicate export names exist\n  const nameCounts = new Map<string, number>();\n  for (const agent of agents) {\n    nameCounts.set(\n      agent.exportName,\n      (nameCounts.get(agent.exportName) || 0) + 1\n    );\n  }\n\n  for (const agent of agents) {\n    const count = nameCounts.get(agent.exportName) ?? 0;\n    if (count > 1) {\n      agent.uniqueId = `${agent.exportName}_${shortHash(agent.filePath)}`;\n    } else {\n      agent.uniqueId = agent.exportName;\n    }\n  }\n\n  return agents;\n}\n\n/**\n * Generates all agent files in .well-known/agent/v1/:\n * - steps.js: RPC step functions with \"use step\"\n * - route.js: triggers workflow discovery\n */\nfunction generateAgentFiles(\n  agents: DetectedAgent[],\n  outputDir: string,\n  _cwd: string,\n  debug: boolean\n): void {\n  if (agents.length === 0) {\n    if (debug) {\n      console.log(\"[withAgent] No agents found, skipping generation\");\n    }\n    return;\n  }\n\n  if (!fs.existsSync(outputDir)) {\n    fs.mkdirSync(outputDir, { recursive: true });\n  }\n\n  /**\n   * Generate step functions that use dynamic import.\n   * Dynamic import ensures the agent module loads (including loader-injected init)\n   * when the step runs in workflow context, avoiding the \"not initialized\" error.\n   *\n   * Uses uniqueId for function names to handle duplicate export names across files.\n   */\n  const rpcFunctions = agents\n    .map((a) => {\n      // Calculate relative path from steps.js to the agent file\n      let agentRelPath = path\n        .relative(outputDir, a.filePath)\n        .replace(/\\\\/g, \"/\");\n\n      if (!agentRelPath.startsWith(\".\")) {\n        agentRelPath = `./${agentRelPath}`;\n      }\n\n      return `\nexport const ${a.uniqueId}Rpc = async (params) => {\n  \"use step\";\n  const { ${a.exportName} } = await import(\"${agentRelPath}\");\n  const res = await ${a.exportName}.handler(params);\n  if (res instanceof Response) {\n    return res.json();\n  }\n  return res;\n};`;\n    })\n    .join(\"\\n\");\n\n  /**\n   * Init functions wire up agent.rpc in the main app context.\n   * Uses exportName for the function name (loader matches by export name),\n   * but references uniqueId for the Rpc function.\n   */\n  const initFunctions = agents\n    .map(\n      (a) => `\nexport function __init${a.exportName[0].toUpperCase()}${a.exportName.slice(\n        1\n      )}(agent) {\n  agent.rpc = ${a.uniqueId}Rpc;\n}`\n    )\n    .join(\"\\n\");\n\n  const stepsContent = `// Auto-generated by withAgent - do not edit\n${rpcFunctions}\n${initFunctions}\n`;\n\n  const stepsPath = path.join(outputDir, \"steps.js\");\n  fs.writeFileSync(stepsPath, stepsContent);\n\n  /**\n   * Generate route.js that imports steps.js.\n   * This ensures workflow discovers the \"use step\" functions.\n   * No actual handler needed - storage RPC uses the step function directly.\n   */\n  const routeContent = `// Auto-generated by withAgent - triggers workflow discovery\nimport \"./steps.js\";\nexport function GET() {\n  return new Response(\"ok\");\n}\n`;\n\n  const routePath = path.join(outputDir, \"route.js\");\n  fs.writeFileSync(routePath, routeContent);\n\n  // Add .gitignore\n  const gitignorePath = path.join(outputDir, \".gitignore\");\n  if (!fs.existsSync(gitignorePath)) {\n    fs.writeFileSync(gitignorePath, \"*\\n\");\n  }\n\n  if (debug) {\n    console.log(`[withAgent] Generated agent files at ${outputDir}`);\n  }\n}\n\ntype TurbopackLoader =\n  | string\n  | { loader: string; options?: Record<string, unknown> };\n\ntype TurbopackRule = {\n  loaders?: TurbopackLoader[];\n  condition?: {\n    all?: unknown[];\n    any?: unknown[];\n    path?: RegExp;\n    content?: RegExp;\n    not?: unknown;\n  };\n};\n\ntype WorkflowConfigFn = (\n  phase: string,\n  ctx: { defaultConfig: NextConfig }\n) => Promise<NextConfig>;\n\ntype NextConfigInput =\n  | NextConfig\n  | ((\n      phase: string,\n      ctx: { defaultConfig: NextConfig }\n    ) => Promise<NextConfig>);\n\n/**\n * Next.js plugin that configures agent support.\n *\n * - Detects agents and generates RPC steps with \"use step\" in .well-known/agent/v1/\n * - Generates AGENT_SECRET for secure RPC calls\n * - Wraps withWorkflow for workflow support\n *\n * @example\n * ```ts\n * // next.config.ts\n * import { withAgent } from \"experimental-agent/next\";\n *\n * // Option 1: withAgent handles everything (recommended)\n * export default withAgent({});\n *\n * // Option 2: Compose with other plugins\n * export default withAgent(withSomeOtherPlugin({}));\n * ```\n */\nexport function withAgent(\n  configOrFn: NextConfigInput,\n  agentConfig?: WithAgentConfig\n): WorkflowConfigFn {\n  const debug = agentConfig?.debug ?? false;\n  const cwd = process.cwd();\n\n  // Detect app directory structure (src/app vs app)\n  const appDir = getAppDir(cwd);\n  const agentDir = `${appDir}/.well-known/agent/${AGENT_PROTOCOL_VERSION}`;\n\n  // Detect agents and generate files FIRST (before workflow discovers them)\n  const agents = detectAgents(cwd, debug);\n  const outputDir = path.join(cwd, agentDir);\n  generateAgentFiles(agents, outputDir, cwd, debug);\n\n  const agentFilePaths = agents.map((a) => a.filePath);\n  const loaderPath = require.resolve(\"./next/loader\");\n  const absoluteAppDir = path.join(cwd, appDir);\n  const loaderOptions = { appDir: absoluteAppDir, debug };\n\n  if (debug) {\n    console.log(\"[withAgent] loader path:\", loaderPath);\n    console.log(\"[withAgent] agent files:\", agentFilePaths);\n    console.log(\"[withAgent] agent dir:\", agentDir);\n  }\n\n  // Build regex to match agent file paths\n  const agentPathRegex =\n    agentFilePaths.length > 0\n      ? new RegExp(\n          `${agentFilePaths\n            .map((p) =>\n              path\n                .relative(cwd, p)\n                .replace(/\\\\/g, \"/\")\n                .replace(/[.*+?^${}()|[\\]\\\\]/g, \"\\\\$&\")\n            )\n            .join(\"|\")}$`\n        )\n      : null;\n\n  /**\n   * Always pass through withWorkflow - it handles both NextConfig and async functions.\n   * This ensures workflow is always configured, regardless of what the user passed.\n   */\n  let withWorkflow: (\n    config: NextConfigInput,\n    options?: { workflows?: { dirs?: string[] } }\n  ) => WorkflowConfigFn;\n  try {\n    withWorkflow = require(\"workflow/next\").withWorkflow;\n  } catch {\n    console.warn(\n      \"[withAgent] workflow/next not found. Make sure workflow is installed.\"\n    );\n    return typeof configOrFn === \"function\"\n      ? configOrFn\n      : async () => configOrFn;\n  }\n\n  const baseConfigFn = withWorkflow(configOrFn, {\n    workflows: {\n      dirs: [...DEFAULT_WORKFLOW_DIRS, agentDir],\n    },\n  });\n\n  /**\n   * Return a wrapper that runs AFTER workflow processes, then adds our loader.\n   * This is necessary because workflow overwrites turbopack rules - we need to\n   * modify them after workflow is done.\n   */\n  return async (phase, ctx) => {\n    const nextConfig = await baseConfigFn(phase, ctx);\n\n    // Set up AGENT_SECRET if not already set\n    if (!process.env.AGENT_SECRET) {\n      nextConfig.env = {\n        ...nextConfig.env,\n        AGENT_SECRET: crypto.randomUUID(),\n      };\n      if (debug) {\n        console.log(\"[withAgent] Generated AGENT_SECRET\");\n      }\n    }\n\n    if (!agentPathRegex || agentFilePaths.length === 0) {\n      return nextConfig;\n    }\n\n    // Modify turbopack rules to include our loader for agent files\n    const rules = (nextConfig.turbopack?.rules || {}) as Record<\n      string,\n      TurbopackRule\n    >;\n\n    for (const key of [\"*.ts\", \"*.tsx\", \"*.js\", \"*.jsx\"]) {\n      const existingRule = rules[key];\n      if (!existingRule) {\n        continue;\n      }\n\n      /**\n       * Workflow's rule has: condition.all = [{ not: path }, { content: /use step|.../ }]\n       * We want: run loader if workflow's conditions match OR it's an agent file.\n       * So we wrap in: condition.any = [existingCondition, { path: agentPathRegex }]\n       */\n      rules[key] = {\n        ...existingRule,\n        loaders: [\n          { loader: loaderPath, options: loaderOptions },\n          ...(existingRule.loaders || []),\n        ],\n        condition: {\n          any: [existingRule.condition, { path: agentPathRegex }],\n        },\n      };\n    }\n\n    // Also configure webpack for non-turbopack builds\n    const existingWebpack = nextConfig.webpack;\n    nextConfig.webpack = (webpackConfig, context) => {\n      webpackConfig.module.rules.push({\n        test: (resourcePath: string) => agentFilePaths.includes(resourcePath),\n        use: [{ loader: loaderPath, options: loaderOptions }],\n      });\n      return existingWebpack?.(webpackConfig, context) ?? webpackConfig;\n    };\n\n    if (debug) {\n      console.log(\"[withAgent] Added agent loader to turbopack rules\");\n    }\n\n    return nextConfig;\n  };\n}\n"],
  "mappings": "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",
  "names": []
}

package/dist/react.d.mts CHANGED
@@ -1,5 +1,5 @@
1
1
  import * as _ai_sdk_react from '@ai-sdk/react';
2
- import { A as AnyAgent, a as AgentStatus } from './client-PuttJd6O.mjs';
2
+ import { A as AnyAgent, a as AgentStatus } from './client-BaItaOx5.mjs';
3
3
  import * as _tanstack_react_query from '@tanstack/react-query';
4
4
  import { QueryOptions } from '@tanstack/react-query';
5
5
  import '@vercel/sandbox';
package/dist/react.d.ts CHANGED
@@ -1,5 +1,5 @@
1
1
  import * as _ai_sdk_react from '@ai-sdk/react';
2
- import { A as AnyAgent, a as AgentStatus } from './client-DIanpEQ_.js';
2
+ import { A as AnyAgent, a as AgentStatus } from './client-BY6gceox.js';
3
3
  import * as _tanstack_react_query from '@tanstack/react-query';
4
4
  import { QueryOptions } from '@tanstack/react-query';
5
5
  import '@vercel/sandbox';
@@ -1,4 +1,5 @@
1
- import { A as AnyAgent } from './client-DIanpEQ_.js';
1
+ import { ToolSet } from 'ai';
2
+ import { A as AnyAgent } from './client-BY6gceox.js';
2
3
  import { S as SkillInput } from './types-B3lbsOa7.js';
3
4
 
4
5
  /**
@@ -8,6 +9,19 @@ import { S as SkillInput } from './types-B3lbsOa7.js';
8
9
  type FsAgentConfig = {
9
10
  system?: string;
10
11
  skills?: SkillInput[];
12
+ tools?: ToolSet;
13
+ };
14
+ type ModuleLoader = () => Promise<{
15
+ default?: unknown;
16
+ }>;
17
+ /**
18
+ * Dynamic import loaders for filesystem modules that can't be
19
+ * serialized into `config`. Keyed by category, then by name.
20
+ * Currently only `tools` is populated; `skills` and `system`
21
+ * will follow the same pattern for `.ts` modules.
22
+ */
23
+ type RegistryImports = {
24
+ tools?: Record<string, ModuleLoader>;
11
25
  };
12
26
  /**
13
27
  * A single entry in the generated agent registry.
@@ -16,9 +30,8 @@ type FsAgentConfig = {
16
30
  */
17
31
  type AgentRegistryEntry = {
18
32
  config: FsAgentConfig;
19
- import?: () => Promise<{
20
- default?: unknown;
21
- }>;
33
+ import?: ModuleLoader;
34
+ imports?: RegistryImports;
22
35
  };
23
36
  /**
24
37
  * Map of agent name to registry entry, produced by the generated
@@ -1,4 +1,5 @@
1
- import { A as AnyAgent } from './client-PuttJd6O.mjs';
1
+ import { ToolSet } from 'ai';
2
+ import { A as AnyAgent } from './client-BaItaOx5.mjs';
2
3
  import { S as SkillInput } from './types-B3lbsOa7.mjs';
3
4
 
4
5
  /**
@@ -8,6 +9,19 @@ import { S as SkillInput } from './types-B3lbsOa7.mjs';
8
9
  type FsAgentConfig = {
9
10
  system?: string;
10
11
  skills?: SkillInput[];
12
+ tools?: ToolSet;
13
+ };
14
+ type ModuleLoader = () => Promise<{
15
+ default?: unknown;
16
+ }>;
17
+ /**
18
+ * Dynamic import loaders for filesystem modules that can't be
19
+ * serialized into `config`. Keyed by category, then by name.
20
+ * Currently only `tools` is populated; `skills` and `system`
21
+ * will follow the same pattern for `.ts` modules.
22
+ */
23
+ type RegistryImports = {
24
+ tools?: Record<string, ModuleLoader>;
11
25
  };
12
26
  /**
13
27
  * A single entry in the generated agent registry.
@@ -16,9 +30,8 @@ type FsAgentConfig = {
16
30
  */
17
31
  type AgentRegistryEntry = {
18
32
  config: FsAgentConfig;
19
- import?: () => Promise<{
20
- default?: unknown;
21
- }>;
33
+ import?: ModuleLoader;
34
+ imports?: RegistryImports;
22
35
  };
23
36
  /**
24
37
  * Map of agent name to registry entry, produced by the generated
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "experimental-agent",
3
- "version": "0.8.0-alpha.2",
3
+ "version": "0.8.0",
4
4
  "main": "./dist/index.js",
5
5
  "module": "./dist/index.mjs",
6
6
  "types": "./dist/index.d.ts",
@@ -71,15 +71,6 @@
71
71
  "require": "./dist/react.js"
72
72
  }
73
73
  },
74
- "scripts": {
75
- "build": "tsup",
76
- "dev": "tsup --watch",
77
- "clean": "rm -rf .turbo && rm -rf node_modules && rm -rf dist",
78
- "release": "pnpm build && npm publish",
79
- "typecheck": "tsc --noEmit",
80
- "test": "vitest run",
81
- "test:watch": "vitest"
82
- },
83
74
  "dependencies": {
84
75
  "@elysiajs/openapi": "^1.4.14",
85
76
  "@swc/core": "^1.10.0",
@@ -96,7 +87,6 @@
96
87
  "zod": "^4.3.6"
97
88
  },
98
89
  "devDependencies": {
99
- "@agent/tsconfig": "workspace:*",
100
90
  "@ai-sdk/react": "^3.0.88",
101
91
  "@types/node": "^20.11.24",
102
92
  "@types/react": "^19.2.14",
@@ -106,12 +96,22 @@
106
96
  "tsup": "^8.0.2",
107
97
  "typescript": "5.5.4",
108
98
  "vitest": "^3.0.0",
109
- "workflow": "4.2.0-beta.70"
99
+ "workflow": "4.2.0-beta.70",
100
+ "@agent/tsconfig": "0.0.0"
110
101
  },
111
102
  "peerDependencies": {
112
103
  "ai": "^6.0.0"
113
104
  },
114
105
  "publishConfig": {
115
106
  "access": "public"
107
+ },
108
+ "scripts": {
109
+ "build": "tsup",
110
+ "dev": "tsup --watch",
111
+ "clean": "rm -rf .turbo && rm -rf node_modules && rm -rf dist",
112
+ "release": "pnpm build && npm publish",
113
+ "typecheck": "tsc --noEmit",
114
+ "test": "vitest run",
115
+ "test:watch": "vitest"
116
116
  }
117
- }
117
+ }
@@ -1,199 +0,0 @@
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,{
  "version": 3,
  "sources": ["../src/framework/generate.ts", "../src/framework/scan.ts", "../src/framework/framework.ts"],
  "sourcesContent": ["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"],
  "mappings": "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",
  "names": ["fs", "path"]
}

@@ -1,123 +0,0 @@
1
- // package.json
2
- var package_default = {
3
- name: "experimental-agent",
4
- version: "0.8.0-alpha.1",
5
- main: "./dist/index.js",
6
- module: "./dist/index.mjs",
7
- types: "./dist/index.d.ts",
8
- sideEffects: true,
9
- license: "MIT",
10
- files: [
11
- "dist/**"
12
- ],
13
- exports: {
14
- ".": {
15
- types: "./dist/index.d.ts",
16
- import: "./dist/index.mjs",
17
- require: "./dist/index.js"
18
- },
19
- "./storage": {
20
- types: "./dist/storage.d.ts",
21
- import: "./dist/storage.mjs",
22
- require: "./dist/storage.js"
23
- },
24
- "./sandbox": {
25
- types: "./dist/sandbox.d.ts",
26
- import: "./dist/sandbox.mjs",
27
- require: "./dist/sandbox.js"
28
- },
29
- "./next": {
30
- types: "./dist/next.d.ts",
31
- import: "./dist/next.mjs",
32
- require: "./dist/next.js"
33
- },
34
- "./next/loader": {
35
- types: "./dist/next/loader.d.ts",
36
- import: "./dist/next/loader.mjs",
37
- require: "./dist/next/loader.js"
38
- },
39
- "./next/agents": {
40
- types: "./dist/next/agents.d.ts",
41
- import: "./dist/next/agents.mjs",
42
- require: "./dist/next/agents.js"
43
- },
44
- "./framework": {
45
- types: "./dist/framework.d.ts",
46
- import: "./dist/framework.mjs",
47
- require: "./dist/framework.js"
48
- },
49
- "./framework-runtime": {
50
- types: "./dist/framework-runtime.d.ts",
51
- import: "./dist/framework-runtime.mjs",
52
- require: "./dist/framework-runtime.js"
53
- },
54
- "./client": {
55
- types: "./dist/client.d.ts",
56
- import: "./dist/client.mjs",
57
- require: "./dist/client.js"
58
- },
59
- "./lifecycle-workflow": {
60
- types: "./dist/lifecycle-workflow.d.ts",
61
- import: "./dist/lifecycle-workflow.mjs",
62
- require: "./dist/lifecycle-workflow.js"
63
- },
64
- "./lifecycle-workflow-steps": {
65
- types: "./dist/lifecycle-workflow-steps.d.ts",
66
- import: "./dist/lifecycle-workflow-steps.mjs",
67
- require: "./dist/lifecycle-workflow-steps.js"
68
- },
69
- "./react": {
70
- types: "./dist/react.d.ts",
71
- import: "./dist/react.mjs",
72
- require: "./dist/react.js"
73
- }
74
- },
75
- scripts: {
76
- build: "tsup",
77
- dev: "tsup --watch",
78
- clean: "rm -rf .turbo && rm -rf node_modules && rm -rf dist",
79
- release: "pnpm build && npm publish",
80
- typecheck: "tsc --noEmit",
81
- test: "vitest run",
82
- "test:watch": "vitest"
83
- },
84
- dependencies: {
85
- "@elysiajs/openapi": "^1.4.14",
86
- "@swc/core": "^1.10.0",
87
- "@tanstack/react-query": "^5.90.21",
88
- "@vercel/kv2": "^0.0.18",
89
- "@vercel/oidc": "^3.1.0",
90
- "@vercel/sandbox": "1.7.1",
91
- "@workflow/serde": "4.1.0-beta.2",
92
- "better-all": "^0.0.5",
93
- elysia: "^1.4.25",
94
- errore: "^0.14.0",
95
- "fast-deep-equal": "^3.1.3",
96
- ulid: "^3.0.2",
97
- zod: "^4.3.6"
98
- },
99
- devDependencies: {
100
- "@agent/tsconfig": "workspace:*",
101
- "@ai-sdk/react": "^3.0.88",
102
- "@types/node": "^20.11.24",
103
- "@types/react": "^19.2.14",
104
- dotenv: "^17.2.3",
105
- next: "^16.0.0",
106
- react: "^19.2.4",
107
- tsup: "^8.0.2",
108
- typescript: "5.5.4",
109
- vitest: "^3.0.0",
110
- workflow: "4.2.0-beta.70"
111
- },
112
- peerDependencies: {
113
- ai: "^6.0.0"
114
- },
115
- publishConfig: {
116
- access: "public"
117
- }
118
- };
119
-
120
- export {
121
- package_default
122
- };
123
- //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vcGFja2FnZS5qc29uIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyJ7XG4gIFwibmFtZVwiOiBcImV4cGVyaW1lbnRhbC1hZ2VudFwiLFxuICBcInZlcnNpb25cIjogXCIwLjguMC1hbHBoYS4xXCIsXG4gIFwibWFpblwiOiBcIi4vZGlzdC9pbmRleC5qc1wiLFxuICBcIm1vZHVsZVwiOiBcIi4vZGlzdC9pbmRleC5tanNcIixcbiAgXCJ0eXBlc1wiOiBcIi4vZGlzdC9pbmRleC5kLnRzXCIsXG4gIFwic2lkZUVmZmVjdHNcIjogdHJ1ZSxcbiAgXCJsaWNlbnNlXCI6IFwiTUlUXCIsXG4gIFwiZmlsZXNcIjogW1xuICAgIFwiZGlzdC8qKlwiXG4gIF0sXG4gIFwiZXhwb3J0c1wiOiB7XG4gICAgXCIuXCI6IHtcbiAgICAgIFwidHlwZXNcIjogXCIuL2Rpc3QvaW5kZXguZC50c1wiLFxuICAgICAgXCJpbXBvcnRcIjogXCIuL2Rpc3QvaW5kZXgubWpzXCIsXG4gICAgICBcInJlcXVpcmVcIjogXCIuL2Rpc3QvaW5kZXguanNcIlxuICAgIH0sXG4gICAgXCIuL3N0b3JhZ2VcIjoge1xuICAgICAgXCJ0eXBlc1wiOiBcIi4vZGlzdC9zdG9yYWdlLmQudHNcIixcbiAgICAgIFwiaW1wb3J0XCI6IFwiLi9kaXN0L3N0b3JhZ2UubWpzXCIsXG4gICAgICBcInJlcXVpcmVcIjogXCIuL2Rpc3Qvc3RvcmFnZS5qc1wiXG4gICAgfSxcbiAgICBcIi4vc2FuZGJveFwiOiB7XG4gICAgICBcInR5cGVzXCI6IFwiLi9kaXN0L3NhbmRib3guZC50c1wiLFxuICAgICAgXCJpbXBvcnRcIjogXCIuL2Rpc3Qvc2FuZGJveC5tanNcIixcbiAgICAgIFwicmVxdWlyZVwiOiBcIi4vZGlzdC9zYW5kYm94LmpzXCJcbiAgICB9LFxuICAgIFwiLi9uZXh0XCI6IHtcbiAgICAgIFwidHlwZXNcIjogXCIuL2Rpc3QvbmV4dC5kLnRzXCIsXG4gICAgICBcImltcG9ydFwiOiBcIi4vZGlzdC9uZXh0Lm1qc1wiLFxuICAgICAgXCJyZXF1aXJlXCI6IFwiLi9kaXN0L25leHQuanNcIlxuICAgIH0sXG4gICAgXCIuL25leHQvbG9hZGVyXCI6IHtcbiAgICAgIFwidHlwZXNcIjogXCIuL2Rpc3QvbmV4dC9sb2FkZXIuZC50c1wiLFxuICAgICAgXCJpbXBvcnRcIjogXCIuL2Rpc3QvbmV4dC9sb2FkZXIubWpzXCIsXG4gICAgICBcInJlcXVpcmVcIjogXCIuL2Rpc3QvbmV4dC9sb2FkZXIuanNcIlxuICAgIH0sXG4gICAgXCIuL25leHQvYWdlbnRzXCI6IHtcbiAgICAgIFwidHlwZXNcIjogXCIuL2Rpc3QvbmV4dC9hZ2VudHMuZC50c1wiLFxuICAgICAgXCJpbXBvcnRcIjogXCIuL2Rpc3QvbmV4dC9hZ2VudHMubWpzXCIsXG4gICAgICBcInJlcXVpcmVcIjogXCIuL2Rpc3QvbmV4dC9hZ2VudHMuanNcIlxuICAgIH0sXG4gICAgXCIuL2ZyYW1ld29ya1wiOiB7XG4gICAgICBcInR5cGVzXCI6IFwiLi9kaXN0L2ZyYW1ld29yay5kLnRzXCIsXG4gICAgICBcImltcG9ydFwiOiBcIi4vZGlzdC9mcmFtZXdvcmsubWpzXCIsXG4gICAgICBcInJlcXVpcmVcIjogXCIuL2Rpc3QvZnJhbWV3b3JrLmpzXCJcbiAgICB9LFxuICAgIFwiLi9mcmFtZXdvcmstcnVudGltZVwiOiB7XG4gICAgICBcInR5cGVzXCI6IFwiLi9kaXN0L2ZyYW1ld29yay1ydW50aW1lLmQudHNcIixcbiAgICAgIFwiaW1wb3J0XCI6IFwiLi9kaXN0L2ZyYW1ld29yay1ydW50aW1lLm1qc1wiLFxuICAgICAgXCJyZXF1aXJlXCI6IFwiLi9kaXN0L2ZyYW1ld29yay1ydW50aW1lLmpzXCJcbiAgICB9LFxuICAgIFwiLi9jbGllbnRcIjoge1xuICAgICAgXCJ0eXBlc1wiOiBcIi4vZGlzdC9jbGllbnQuZC50c1wiLFxuICAgICAgXCJpbXBvcnRcIjogXCIuL2Rpc3QvY2xpZW50Lm1qc1wiLFxuICAgICAgXCJyZXF1aXJlXCI6IFwiLi9kaXN0L2NsaWVudC5qc1wiXG4gICAgfSxcbiAgICBcIi4vbGlmZWN5Y2xlLXdvcmtmbG93XCI6IHtcbiAgICAgIFwidHlwZXNcIjogXCIuL2Rpc3QvbGlmZWN5Y2xlLXdvcmtmbG93LmQudHNcIixcbiAgICAgIFwiaW1wb3J0XCI6IFwiLi9kaXN0L2xpZmVjeWNsZS13b3JrZmxvdy5tanNcIixcbiAgICAgIFwicmVxdWlyZVwiOiBcIi4vZGlzdC9saWZlY3ljbGUtd29ya2Zsb3cuanNcIlxuICAgIH0sXG4gICAgXCIuL2xpZmVjeWNsZS13b3JrZmxvdy1zdGVwc1wiOiB7XG4gICAgICBcInR5cGVzXCI6IFwiLi9kaXN0L2xpZmVjeWNsZS13b3JrZmxvdy1zdGVwcy5kLnRzXCIsXG4gICAgICBcImltcG9ydFwiOiBcIi4vZGlzdC9saWZlY3ljbGUtd29ya2Zsb3ctc3RlcHMubWpzXCIsXG4gICAgICBcInJlcXVpcmVcIjogXCIuL2Rpc3QvbGlmZWN5Y2xlLXdvcmtmbG93LXN0ZXBzLmpzXCJcbiAgICB9LFxuICAgIFwiLi9yZWFjdFwiOiB7XG4gICAgICBcInR5cGVzXCI6IFwiLi9kaXN0L3JlYWN0LmQudHNcIixcbiAgICAgIFwiaW1wb3J0XCI6IFwiLi9kaXN0L3JlYWN0Lm1qc1wiLFxuICAgICAgXCJyZXF1aXJlXCI6IFwiLi9kaXN0L3JlYWN0LmpzXCJcbiAgICB9XG4gIH0sXG4gIFwic2NyaXB0c1wiOiB7XG4gICAgXCJidWlsZFwiOiBcInRzdXBcIixcbiAgICBcImRldlwiOiBcInRzdXAgLS13YXRjaFwiLFxuICAgIFwiY2xlYW5cIjogXCJybSAtcmYgLnR1cmJvICYmIHJtIC1yZiBub2RlX21vZHVsZXMgJiYgcm0gLXJmIGRpc3RcIixcbiAgICBcInJlbGVhc2VcIjogXCJwbnBtIGJ1aWxkICYmIG5wbSBwdWJsaXNoXCIsXG4gICAgXCJ0eXBlY2hlY2tcIjogXCJ0c2MgLS1ub0VtaXRcIixcbiAgICBcInRlc3RcIjogXCJ2aXRlc3QgcnVuXCIsXG4gICAgXCJ0ZXN0OndhdGNoXCI6IFwidml0ZXN0XCJcbiAgfSxcbiAgXCJkZXBlbmRlbmNpZXNcIjoge1xuICAgIFwiQGVseXNpYWpzL29wZW5hcGlcIjogXCJeMS40LjE0XCIsXG4gICAgXCJAc3djL2NvcmVcIjogXCJeMS4xMC4wXCIsXG4gICAgXCJAdGFuc3RhY2svcmVhY3QtcXVlcnlcIjogXCJeNS45MC4yMVwiLFxuICAgIFwiQHZlcmNlbC9rdjJcIjogXCJeMC4wLjE4XCIsXG4gICAgXCJAdmVyY2VsL29pZGNcIjogXCJeMy4xLjBcIixcbiAgICBcIkB2ZXJjZWwvc2FuZGJveFwiOiBcIjEuNy4xXCIsXG4gICAgXCJAd29ya2Zsb3cvc2VyZGVcIjogXCI0LjEuMC1iZXRhLjJcIixcbiAgICBcImJldHRlci1hbGxcIjogXCJeMC4wLjVcIixcbiAgICBcImVseXNpYVwiOiBcIl4xLjQuMjVcIixcbiAgICBcImVycm9yZVwiOiBcIl4wLjE0LjBcIixcbiAgICBcImZhc3QtZGVlcC1lcXVhbFwiOiBcIl4zLjEuM1wiLFxuICAgIFwidWxpZFwiOiBcIl4zLjAuMlwiLFxuICAgIFwiem9kXCI6IFwiXjQuMy42XCJcbiAgfSxcbiAgXCJkZXZEZXBlbmRlbmNpZXNcIjoge1xuICAgIFwiQGFnZW50L3RzY29uZmlnXCI6IFwid29ya3NwYWNlOipcIixcbiAgICBcIkBhaS1zZGsvcmVhY3RcIjogXCJeMy4wLjg4XCIsXG4gICAgXCJAdHlwZXMvbm9kZVwiOiBcIl4yMC4xMS4yNFwiLFxuICAgIFwiQHR5cGVzL3JlYWN0XCI6IFwiXjE5LjIuMTRcIixcbiAgICBcImRvdGVudlwiOiBcIl4xNy4yLjNcIixcbiAgICBcIm5leHRcIjogXCJeMTYuMC4wXCIsXG4gICAgXCJyZWFjdFwiOiBcIl4xOS4yLjRcIixcbiAgICBcInRzdXBcIjogXCJeOC4wLjJcIixcbiAgICBcInR5cGVzY3JpcHRcIjogXCI1LjUuNFwiLFxuICAgIFwidml0ZXN0XCI6IFwiXjMuMC4wXCIsXG4gICAgXCJ3b3JrZmxvd1wiOiBcIjQuMi4wLWJldGEuNzBcIlxuICB9LFxuICBcInBlZXJEZXBlbmRlbmNpZXNcIjoge1xuICAgIFwiYWlcIjogXCJeNi4wLjBcIlxuICB9LFxuICBcInB1Ymxpc2hDb25maWdcIjoge1xuICAgIFwiYWNjZXNzXCI6IFwicHVibGljXCJcbiAgfVxufVxuIl0sCiAgIm1hcHBpbmdzIjogIjtBQUFBO0FBQUEsRUFDRSxNQUFRO0FBQUEsRUFDUixTQUFXO0FBQUEsRUFDWCxNQUFRO0FBQUEsRUFDUixRQUFVO0FBQUEsRUFDVixPQUFTO0FBQUEsRUFDVCxhQUFlO0FBQUEsRUFDZixTQUFXO0FBQUEsRUFDWCxPQUFTO0FBQUEsSUFDUDtBQUFBLEVBQ0Y7QUFBQSxFQUNBLFNBQVc7QUFBQSxJQUNULEtBQUs7QUFBQSxNQUNILE9BQVM7QUFBQSxNQUNULFFBQVU7QUFBQSxNQUNWLFNBQVc7QUFBQSxJQUNiO0FBQUEsSUFDQSxhQUFhO0FBQUEsTUFDWCxPQUFTO0FBQUEsTUFDVCxRQUFVO0FBQUEsTUFDVixTQUFXO0FBQUEsSUFDYjtBQUFBLElBQ0EsYUFBYTtBQUFBLE1BQ1gsT0FBUztBQUFBLE1BQ1QsUUFBVTtBQUFBLE1BQ1YsU0FBVztBQUFBLElBQ2I7QUFBQSxJQUNBLFVBQVU7QUFBQSxNQUNSLE9BQVM7QUFBQSxNQUNULFFBQVU7QUFBQSxNQUNWLFNBQVc7QUFBQSxJQUNiO0FBQUEsSUFDQSxpQkFBaUI7QUFBQSxNQUNmLE9BQVM7QUFBQSxNQUNULFFBQVU7QUFBQSxNQUNWLFNBQVc7QUFBQSxJQUNiO0FBQUEsSUFDQSxpQkFBaUI7QUFBQSxNQUNmLE9BQVM7QUFBQSxNQUNULFFBQVU7QUFBQSxNQUNWLFNBQVc7QUFBQSxJQUNiO0FBQUEsSUFDQSxlQUFlO0FBQUEsTUFDYixPQUFTO0FBQUEsTUFDVCxRQUFVO0FBQUEsTUFDVixTQUFXO0FBQUEsSUFDYjtBQUFBLElBQ0EsdUJBQXVCO0FBQUEsTUFDckIsT0FBUztBQUFBLE1BQ1QsUUFBVTtBQUFBLE1BQ1YsU0FBVztBQUFBLElBQ2I7QUFBQSxJQUNBLFlBQVk7QUFBQSxNQUNWLE9BQVM7QUFBQSxNQUNULFFBQVU7QUFBQSxNQUNWLFNBQVc7QUFBQSxJQUNiO0FBQUEsSUFDQSx3QkFBd0I7QUFBQSxNQUN0QixPQUFTO0FBQUEsTUFDVCxRQUFVO0FBQUEsTUFDVixTQUFXO0FBQUEsSUFDYjtBQUFBLElBQ0EsOEJBQThCO0FBQUEsTUFDNUIsT0FBUztBQUFBLE1BQ1QsUUFBVTtBQUFBLE1BQ1YsU0FBVztBQUFBLElBQ2I7QUFBQSxJQUNBLFdBQVc7QUFBQSxNQUNULE9BQVM7QUFBQSxNQUNULFFBQVU7QUFBQSxNQUNWLFNBQVc7QUFBQSxJQUNiO0FBQUEsRUFDRjtBQUFBLEVBQ0EsU0FBVztBQUFBLElBQ1QsT0FBUztBQUFBLElBQ1QsS0FBTztBQUFBLElBQ1AsT0FBUztBQUFBLElBQ1QsU0FBVztBQUFBLElBQ1gsV0FBYTtBQUFBLElBQ2IsTUFBUTtBQUFBLElBQ1IsY0FBYztBQUFBLEVBQ2hCO0FBQUEsRUFDQSxjQUFnQjtBQUFBLElBQ2QscUJBQXFCO0FBQUEsSUFDckIsYUFBYTtBQUFBLElBQ2IseUJBQXlCO0FBQUEsSUFDekIsZUFBZTtBQUFBLElBQ2YsZ0JBQWdCO0FBQUEsSUFDaEIsbUJBQW1CO0FBQUEsSUFDbkIsbUJBQW1CO0FBQUEsSUFDbkIsY0FBYztBQUFBLElBQ2QsUUFBVTtBQUFBLElBQ1YsUUFBVTtBQUFBLElBQ1YsbUJBQW1CO0FBQUEsSUFDbkIsTUFBUTtBQUFBLElBQ1IsS0FBTztBQUFBLEVBQ1Q7QUFBQSxFQUNBLGlCQUFtQjtBQUFBLElBQ2pCLG1CQUFtQjtBQUFBLElBQ25CLGlCQUFpQjtBQUFBLElBQ2pCLGVBQWU7QUFBQSxJQUNmLGdCQUFnQjtBQUFBLElBQ2hCLFFBQVU7QUFBQSxJQUNWLE1BQVE7QUFBQSxJQUNSLE9BQVM7QUFBQSxJQUNULE1BQVE7QUFBQSxJQUNSLFlBQWM7QUFBQSxJQUNkLFFBQVU7QUFBQSxJQUNWLFVBQVk7QUFBQSxFQUNkO0FBQUEsRUFDQSxrQkFBb0I7QUFBQSxJQUNsQixJQUFNO0FBQUEsRUFDUjtBQUFBLEVBQ0EsZUFBaUI7QUFBQSxJQUNmLFFBQVU7QUFBQSxFQUNaO0FBQ0Y7IiwKICAibmFtZXMiOiBbXQp9Cg==