@agent-native/core 0.17.2 → 0.18.1

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.
Files changed (80) hide show
  1. package/dist/action.d.ts +27 -0
  2. package/dist/action.d.ts.map +1 -1
  3. package/dist/action.js +2 -0
  4. package/dist/action.js.map +1 -1
  5. package/dist/agent/production-agent.d.ts +4 -0
  6. package/dist/agent/production-agent.d.ts.map +1 -1
  7. package/dist/agent/production-agent.js.map +1 -1
  8. package/dist/cli/index.js +16 -0
  9. package/dist/cli/index.js.map +1 -1
  10. package/dist/cli/mcp.d.ts +16 -0
  11. package/dist/cli/mcp.d.ts.map +1 -0
  12. package/dist/cli/mcp.js +583 -0
  13. package/dist/cli/mcp.js.map +1 -0
  14. package/dist/db/client.d.ts +17 -14
  15. package/dist/db/client.d.ts.map +1 -1
  16. package/dist/db/client.js +31 -27
  17. package/dist/db/client.js.map +1 -1
  18. package/dist/db/create-get-db.js +2 -2
  19. package/dist/db/create-get-db.js.map +1 -1
  20. package/dist/index.d.ts +1 -1
  21. package/dist/index.d.ts.map +1 -1
  22. package/dist/index.js.map +1 -1
  23. package/dist/mcp/build-server.d.ts +152 -0
  24. package/dist/mcp/build-server.d.ts.map +1 -0
  25. package/dist/mcp/build-server.js +349 -0
  26. package/dist/mcp/build-server.js.map +1 -0
  27. package/dist/mcp/builtin-tools.d.ts +39 -0
  28. package/dist/mcp/builtin-tools.d.ts.map +1 -0
  29. package/dist/mcp/builtin-tools.js +401 -0
  30. package/dist/mcp/builtin-tools.js.map +1 -0
  31. package/dist/mcp/index.d.ts +7 -0
  32. package/dist/mcp/index.d.ts.map +1 -1
  33. package/dist/mcp/index.js +8 -0
  34. package/dist/mcp/index.js.map +1 -1
  35. package/dist/mcp/server.d.ts +3 -13
  36. package/dist/mcp/server.d.ts.map +1 -1
  37. package/dist/mcp/server.js +44 -179
  38. package/dist/mcp/server.js.map +1 -1
  39. package/dist/mcp/stdio.d.ts +44 -0
  40. package/dist/mcp/stdio.d.ts.map +1 -0
  41. package/dist/mcp/stdio.js +209 -0
  42. package/dist/mcp/stdio.js.map +1 -0
  43. package/dist/mcp/workspace-resolve.d.ts +68 -0
  44. package/dist/mcp/workspace-resolve.d.ts.map +1 -0
  45. package/dist/mcp/workspace-resolve.js +205 -0
  46. package/dist/mcp/workspace-resolve.js.map +1 -0
  47. package/dist/server/action-discovery.d.ts.map +1 -1
  48. package/dist/server/action-discovery.js +3 -0
  49. package/dist/server/action-discovery.js.map +1 -1
  50. package/dist/server/agent-chat-plugin.d.ts.map +1 -1
  51. package/dist/server/agent-chat-plugin.js +1 -0
  52. package/dist/server/agent-chat-plugin.js.map +1 -1
  53. package/dist/server/auth.d.ts +9 -0
  54. package/dist/server/auth.d.ts.map +1 -1
  55. package/dist/server/auth.js +71 -19
  56. package/dist/server/auth.js.map +1 -1
  57. package/dist/server/better-auth-instance.d.ts.map +1 -1
  58. package/dist/server/better-auth-instance.js +15 -10
  59. package/dist/server/better-auth-instance.js.map +1 -1
  60. package/dist/server/core-routes-plugin.d.ts +5 -0
  61. package/dist/server/core-routes-plugin.d.ts.map +1 -1
  62. package/dist/server/core-routes-plugin.js +9 -0
  63. package/dist/server/core-routes-plugin.js.map +1 -1
  64. package/dist/server/deep-link.d.ts +55 -0
  65. package/dist/server/deep-link.d.ts.map +1 -0
  66. package/dist/server/deep-link.js +69 -0
  67. package/dist/server/deep-link.js.map +1 -0
  68. package/dist/server/index.d.ts +2 -0
  69. package/dist/server/index.d.ts.map +1 -1
  70. package/dist/server/index.js +2 -0
  71. package/dist/server/index.js.map +1 -1
  72. package/dist/server/open-route.d.ts +12 -0
  73. package/dist/server/open-route.d.ts.map +1 -0
  74. package/dist/server/open-route.js +159 -0
  75. package/dist/server/open-route.js.map +1 -0
  76. package/dist/server/request-context.d.ts +8 -0
  77. package/dist/server/request-context.d.ts.map +1 -1
  78. package/dist/server/request-context.js.map +1 -1
  79. package/docs/content/external-agents.md +177 -0
  80. package/package.json +1 -1
@@ -0,0 +1,401 @@
1
+ /**
2
+ * Generic cross-app MCP tools — a stable verb set every external agent gets
3
+ * regardless of which template it is talking to.
4
+ *
5
+ * These are merged into the MCP action registry by
6
+ * `createMCPServerForRequest` (see `build-server.ts`). **Precedence: template
7
+ * actions win.** If a template defines an action named `list_apps` /
8
+ * `open_app` / `ask_app` / `create_workspace_app` / `list_templates`, the
9
+ * template's `ActionEntry` overwrites the builtin of the same name. This is
10
+ * the same template-over-framework precedence `autoDiscoverActions` uses.
11
+ *
12
+ * | Tool | Side effects | Returns |
13
+ * | --------------------- | ------------ | ---------------------------------------- |
14
+ * | `list_apps` | none | `{ apps: [{ id, url, running }] }` |
15
+ * | `open_app` | none | `{ url }` (+ deep-link `link`) |
16
+ * | `ask_app` | agent loop | `{ app, routedVia, response }` |
17
+ * | `create_workspace_app`| scaffolds | `{ name, url, port, deepLink }` (+ link) |
18
+ *
19
+ * `open_app` / `create_workspace_app` return an **absolute** URL on the
20
+ * *target* app's origin when it differs from this app (so a workspace link
21
+ * lands in the right app), and a relative path for the same app / standalone.
22
+ * `ask_app` routes to a *different* workspace app over A2A when possible and
23
+ * reports `routedVia: "a2a"`; otherwise it answers locally
24
+ * (`routedVia: "local"`) and never falsely claims cross-app delegation.
25
+ * | `list_templates` | none | `{ templates: [...] }` (allow-list only) |
26
+ *
27
+ * Node-only at call time (workspace resolution + scaffolding use `fs`), but
28
+ * the module has no top-level Node imports so it bundles fine alongside
29
+ * `mountMCP` — the Node bits are dynamically imported inside `run()`.
30
+ */
31
+ import { buildDeepLink } from "../server/deep-link.js";
32
+ /**
33
+ * Build an `ActionTool`. `parameters` is wrapped in the
34
+ * `{ type:"object", properties, required }` shape `createMCPServerForRequest`
35
+ * forwards verbatim as the MCP tool `inputSchema`.
36
+ */
37
+ function tool(description, parameters, required) {
38
+ if (!parameters)
39
+ return { description };
40
+ return {
41
+ description,
42
+ parameters: {
43
+ type: "object",
44
+ properties: parameters,
45
+ ...(required && required.length ? { required } : {}),
46
+ },
47
+ };
48
+ }
49
+ /**
50
+ * The canonical app id this MCP server is mounted for. `MCPConfig.appId` is
51
+ * authoritative; fall back to lowercasing `name` (which is the capitalized
52
+ * app id at every call site) for back-compat with configs that predate the
53
+ * `appId` field.
54
+ */
55
+ function currentAppId(config) {
56
+ return (config.appId || config.name || "app").toLowerCase();
57
+ }
58
+ /**
59
+ * Resolve the absolute origin of a *target* workspace app (e.g.
60
+ * `http://127.0.0.1:8101`) so cross-app deep links / A2A calls point at the
61
+ * right app instead of the current request's origin. Reuses the same
62
+ * workspace resolution `list_apps` / the stdio proxy use.
63
+ *
64
+ * Returns `null` when:
65
+ * - the target is the current app (caller should keep relative behavior),
66
+ * - there is no workspace info (standalone / single app), or
67
+ * - the target app is unknown.
68
+ */
69
+ async function resolveTargetAppOrigin(config, targetAppId) {
70
+ const target = targetAppId.trim().toLowerCase();
71
+ if (!target || target === currentAppId(config))
72
+ return null;
73
+ try {
74
+ const { resolveWorkspace } = await import("./workspace-resolve.js");
75
+ const ws = await resolveWorkspace();
76
+ if (!ws.isWorkspace)
77
+ return null;
78
+ const match = ws.apps.find((a) => a.id.toLowerCase() === target);
79
+ if (!match)
80
+ return null;
81
+ return { origin: match.url, id: match.id };
82
+ }
83
+ catch {
84
+ return null;
85
+ }
86
+ }
87
+ // ---------------------------------------------------------------------------
88
+ // list_apps
89
+ // ---------------------------------------------------------------------------
90
+ function listAppsTool() {
91
+ return {
92
+ tool: tool("List the workspace apps and their local dev URLs/ports. Use this to " +
93
+ "discover which apps exist before opening or asking one. In a single-" +
94
+ "app project this returns just that app."),
95
+ readOnly: true,
96
+ parallelSafe: true,
97
+ run: async () => {
98
+ const { resolveWorkspace } = await import("./workspace-resolve.js");
99
+ const ws = await resolveWorkspace();
100
+ return {
101
+ workspace: ws.isWorkspace,
102
+ gatewayUrl: ws.gatewayUrl,
103
+ apps: ws.apps.map((a) => ({
104
+ id: a.id,
105
+ url: a.url,
106
+ port: a.port,
107
+ running: a.running,
108
+ })),
109
+ };
110
+ },
111
+ };
112
+ }
113
+ // ---------------------------------------------------------------------------
114
+ // open_app
115
+ // ---------------------------------------------------------------------------
116
+ function openAppTool(config) {
117
+ return {
118
+ tool: tool("Build a deep link that opens an app at a specific view/record. No side " +
119
+ "effects — returns a URL the user can click to land in the running UI. " +
120
+ 'After calling, surface the returned "Open in … →" link to the user.', {
121
+ app: { type: "string", description: "App id, e.g. 'mail'" },
122
+ view: {
123
+ type: "string",
124
+ description: "Target view, e.g. 'inbox' (maps to navigate command)",
125
+ },
126
+ params: {
127
+ type: "object",
128
+ description: "Optional record-focus / filter params, e.g. { threadId: 'abc' }",
129
+ },
130
+ }, ["app", "view"]),
131
+ readOnly: true,
132
+ parallelSafe: true,
133
+ run: async (args) => {
134
+ const app = String(args.app ?? "").trim();
135
+ const view = String(args.view ?? "").trim();
136
+ if (!app || !view) {
137
+ throw new Error("open_app requires both 'app' and 'view'.");
138
+ }
139
+ let params;
140
+ const raw = args.params;
141
+ if (raw && typeof raw === "object") {
142
+ params = raw;
143
+ }
144
+ else if (typeof raw === "string" && raw.trim()) {
145
+ try {
146
+ params = JSON.parse(raw);
147
+ }
148
+ catch {
149
+ params = undefined;
150
+ }
151
+ }
152
+ const relUrl = buildDeepLink({ app, view, params });
153
+ // Cross-app target in a workspace: resolve the TARGET app's origin and
154
+ // return an absolute URL. Otherwise the MCP layer would prefix the
155
+ // relative path with the CURRENT request origin, landing the user in
156
+ // the wrong app (e.g. open_app({app:"calendar"}) served from Mail).
157
+ // Same-app / standalone keeps the relative path (current behavior).
158
+ const targetApp = await resolveTargetAppOrigin(config, app);
159
+ const url = targetApp
160
+ ? `${targetApp.origin.replace(/\/+$/, "")}${relUrl}`
161
+ : relUrl;
162
+ return { app, view, url };
163
+ },
164
+ link: ({ result }) => {
165
+ if (!result || typeof result !== "object")
166
+ return null;
167
+ const r = result;
168
+ if (!r.url)
169
+ return null;
170
+ return {
171
+ url: r.url,
172
+ label: `Open ${r.app ?? "app"}`,
173
+ view: r.view,
174
+ };
175
+ },
176
+ };
177
+ }
178
+ // ---------------------------------------------------------------------------
179
+ // ask_app
180
+ // ---------------------------------------------------------------------------
181
+ function askAppTool(config) {
182
+ return {
183
+ tool: tool("Send a natural-language message to an app's AI agent and get its " +
184
+ "response. Use for complex, multi-step tasks needing the agent's " +
185
+ "reasoning and full app context. In a single-app project the 'app' " +
186
+ "param is optional (defaults to this app). When 'app' names a " +
187
+ "different workspace app it is routed there over A2A; the result's " +
188
+ "'routedVia' field reports whether it ran cross-app or locally.", {
189
+ app: {
190
+ type: "string",
191
+ description: "App id to route to (optional in a single-app project)",
192
+ },
193
+ message: {
194
+ type: "string",
195
+ description: "The message to send to the app's agent",
196
+ },
197
+ }, ["message"]),
198
+ run: async (args) => {
199
+ const message = String(args.message ?? "").trim();
200
+ if (!message)
201
+ throw new Error("ask_app requires a 'message'.");
202
+ const requestedApp = String(args.app ?? "").trim();
203
+ const selfId = currentAppId(config);
204
+ // Cross-app: the caller named a *different* workspace app. Route the
205
+ // message to THAT app's agent over A2A (its `/_agent-native/a2a`
206
+ // endpoint runs the real agent loop with JWT identity) rather than
207
+ // silently answering from this app's agent and claiming delegation.
208
+ const targetApp = await resolveTargetAppOrigin(config, requestedApp);
209
+ if (targetApp) {
210
+ try {
211
+ const { callAgent } = await import("../a2a/client.js");
212
+ const { getRequestUserEmail } = await import("../server/request-context.js");
213
+ // The MCP handler runs inside `runWithRequestContext`, so this is
214
+ // the verified caller's email — it lets `callAgent` mint a signed
215
+ // A2A JWT so the target app honours per-user scope.
216
+ const response = await callAgent(targetApp.origin, message, {
217
+ userEmail: getRequestUserEmail(),
218
+ // Bound the wait — cross-app A2A polls async by default.
219
+ timeoutMs: 5 * 60_000,
220
+ });
221
+ return {
222
+ app: targetApp.id,
223
+ routedVia: "a2a",
224
+ response,
225
+ };
226
+ }
227
+ catch (err) {
228
+ // Be honest: routing was attempted and failed — do NOT fall back to
229
+ // this app's agent and pretend it was the target.
230
+ throw new Error(`Failed to route ask_app to "${targetApp.id}" via A2A: ` +
231
+ `${err?.message ?? err}`);
232
+ }
233
+ }
234
+ // Same app (or no workspace / unknown target): answer locally with this
235
+ // app's own ask-agent handler — the same entry point the HTTP MCP mount
236
+ // + A2A use, so there is no second agent runner.
237
+ if (!config.askAgent) {
238
+ throw new Error("This app does not expose an agent (no ask-agent handler).");
239
+ }
240
+ // If the caller named an app we couldn't route to (unknown id, or no
241
+ // workspace), say so honestly instead of claiming we reached it.
242
+ const unresolved = !!requestedApp && requestedApp.toLowerCase() !== selfId;
243
+ const response = await config.askAgent(message);
244
+ return {
245
+ app: selfId,
246
+ routedVia: "local",
247
+ ...(unresolved
248
+ ? {
249
+ note: `Requested app "${requestedApp}" is not a reachable workspace ` +
250
+ `app; answered with this app ("${selfId}") instead.`,
251
+ }
252
+ : {}),
253
+ response,
254
+ };
255
+ },
256
+ };
257
+ }
258
+ // ---------------------------------------------------------------------------
259
+ // list_templates
260
+ // ---------------------------------------------------------------------------
261
+ function listTemplatesTool() {
262
+ return {
263
+ tool: tool("List the first-party templates that can be scaffolded into a workspace " +
264
+ "(allow-listed templates only)."),
265
+ readOnly: true,
266
+ parallelSafe: true,
267
+ run: async () => {
268
+ const { visibleTemplates } = await import("../cli/templates-meta.js");
269
+ return {
270
+ templates: visibleTemplates().map((t) => ({
271
+ name: t.name,
272
+ label: t.label,
273
+ hint: t.hint,
274
+ })),
275
+ };
276
+ },
277
+ };
278
+ }
279
+ // ---------------------------------------------------------------------------
280
+ // create_workspace_app
281
+ // ---------------------------------------------------------------------------
282
+ function createWorkspaceAppTool() {
283
+ return {
284
+ tool: tool("Scaffold a new app into the current workspace from an allow-listed " +
285
+ "template, then return a deep link to open it. Idempotent: if an app " +
286
+ "with that name already exists it is reused. After calling, surface " +
287
+ 'the returned "Open … →" link to the user.', {
288
+ name: {
289
+ type: "string",
290
+ description: "New app id (directory under apps/), e.g. 'mymail'",
291
+ },
292
+ template: {
293
+ type: "string",
294
+ description: "Template to scaffold from — must be allow-listed (see list_templates)",
295
+ },
296
+ }, ["name", "template"]),
297
+ run: async (args) => {
298
+ const name = String(args.name ?? "").trim();
299
+ const template = String(args.template ?? "").trim();
300
+ if (!name || !template) {
301
+ throw new Error("create_workspace_app requires both 'name' and 'template'.");
302
+ }
303
+ // Enforce the strict public template allow-list. The authoritative,
304
+ // dependency-free source inside @agent-native/core is cli/templates-meta
305
+ // (kept in sync with packages/shared-app-config/templates.ts; CI guard).
306
+ const { visibleTemplates } = await import("../cli/templates-meta.js");
307
+ const allowed = new Set(visibleTemplates().map((t) => t.name));
308
+ if (!allowed.has(template)) {
309
+ throw new Error(`Template "${template}" is not allow-listed. Allowed: ${[...allowed]
310
+ .sort()
311
+ .join(", ")}`);
312
+ }
313
+ const { findWorkspaceRoot, resolveWorkspace } = await import("./workspace-resolve.js");
314
+ const fs = await import("node:fs");
315
+ const path = await import("node:path");
316
+ const root = findWorkspaceRoot(process.cwd());
317
+ if (!root) {
318
+ throw new Error("Not inside a workspace. create_workspace_app only works in a " +
319
+ "multi-app workspace (run from the workspace root).");
320
+ }
321
+ const appDir = path.join(root, "apps", name);
322
+ const alreadyExisted = fs.existsSync(appDir);
323
+ if (!alreadyExisted) {
324
+ // Reuse the CLI scaffolder directly (no second `agent-native`
325
+ // subprocess). `addAppToWorkspace(name, { template })` takes the
326
+ // non-interactive single-template path when name + one template are
327
+ // given. Run it from the workspace root so detectWorkspace resolves.
328
+ const prevCwd = process.cwd();
329
+ try {
330
+ process.chdir(root);
331
+ const { addAppToWorkspace } = await import("../cli/create.js");
332
+ await addAppToWorkspace(name, { template, noInstall: true });
333
+ }
334
+ finally {
335
+ try {
336
+ process.chdir(prevCwd);
337
+ }
338
+ catch {
339
+ // best-effort cwd restore
340
+ }
341
+ }
342
+ }
343
+ // The workspace gateway auto-detects new apps/* dirs (fs.watch +
344
+ // 2s sync) and lazily boots the dev server on first request, so we
345
+ // don't spawn vite ourselves — opening the deep link warms it. Resolve
346
+ // the port the gateway will use so we can report it.
347
+ const ws = await resolveWorkspace(root);
348
+ const appInfo = ws.apps.find((a) => a.id === name);
349
+ const port = appInfo?.port;
350
+ // The scaffolded app is always a *different* app from the host MCP
351
+ // server, so anchor the deep link to the new app's own origin. A
352
+ // relative path would otherwise be prefixed with the current request
353
+ // origin and land on the wrong app. Fall back to the relative path
354
+ // only if the gateway hasn't reported the new app's URL yet.
355
+ const relDeepLink = buildDeepLink({ app: name, view: "home" });
356
+ const deepLink = appInfo?.url
357
+ ? `${appInfo.url.replace(/\/+$/, "")}${relDeepLink}`
358
+ : relDeepLink;
359
+ return {
360
+ name,
361
+ template,
362
+ created: !alreadyExisted,
363
+ reused: alreadyExisted,
364
+ port,
365
+ url: appInfo?.url,
366
+ gatewayUrl: ws.gatewayUrl,
367
+ deepLink,
368
+ };
369
+ },
370
+ link: ({ result }) => {
371
+ if (!result || typeof result !== "object")
372
+ return null;
373
+ const r = result;
374
+ if (!r.deepLink)
375
+ return null;
376
+ return {
377
+ url: r.deepLink,
378
+ label: `Open ${r.name ?? "app"}`,
379
+ view: "home",
380
+ };
381
+ },
382
+ };
383
+ }
384
+ // ---------------------------------------------------------------------------
385
+ // Registry
386
+ // ---------------------------------------------------------------------------
387
+ /**
388
+ * Build the generic cross-app builtin tool registry. Called by
389
+ * `createMCPServerForRequest`; the result is merged UNDER the config's
390
+ * actions so template actions of the same name win.
391
+ */
392
+ export function getBuiltinCrossAppTools(config) {
393
+ return {
394
+ list_apps: listAppsTool(),
395
+ open_app: openAppTool(config),
396
+ ask_app: askAppTool(config),
397
+ create_workspace_app: createWorkspaceAppTool(),
398
+ list_templates: listTemplatesTool(),
399
+ };
400
+ }
401
+ //# sourceMappingURL=builtin-tools.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"builtin-tools.js","sourceRoot":"","sources":["../../src/mcp/builtin-tools.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Generic cross-app MCP tools — a stable verb set every external agent gets\n * regardless of which template it is talking to.\n *\n * These are merged into the MCP action registry by\n * `createMCPServerForRequest` (see `build-server.ts`). **Precedence: template\n * actions win.** If a template defines an action named `list_apps` /\n * `open_app` / `ask_app` / `create_workspace_app` / `list_templates`, the\n * template's `ActionEntry` overwrites the builtin of the same name. This is\n * the same template-over-framework precedence `autoDiscoverActions` uses.\n *\n * | Tool | Side effects | Returns |\n * | --------------------- | ------------ | ---------------------------------------- |\n * | `list_apps` | none | `{ apps: [{ id, url, running }] }` |\n * | `open_app` | none | `{ url }` (+ deep-link `link`) |\n * | `ask_app` | agent loop | `{ app, routedVia, response }` |\n * | `create_workspace_app`| scaffolds | `{ name, url, port, deepLink }` (+ link) |\n *\n * `open_app` / `create_workspace_app` return an **absolute** URL on the\n * *target* app's origin when it differs from this app (so a workspace link\n * lands in the right app), and a relative path for the same app / standalone.\n * `ask_app` routes to a *different* workspace app over A2A when possible and\n * reports `routedVia: \"a2a\"`; otherwise it answers locally\n * (`routedVia: \"local\"`) and never falsely claims cross-app delegation.\n * | `list_templates` | none | `{ templates: [...] }` (allow-list only) |\n *\n * Node-only at call time (workspace resolution + scaffolding use `fs`), but\n * the module has no top-level Node imports so it bundles fine alongside\n * `mountMCP` — the Node bits are dynamically imported inside `run()`.\n */\n\nimport type { ActionEntry } from \"../agent/production-agent.js\";\nimport { buildDeepLink } from \"../server/deep-link.js\";\nimport type { MCPConfig } from \"./build-server.js\";\n\nimport type { ActionTool } from \"../agent/types.js\";\n\n/** Flat map of param name → JSON-schema property. */\ntype Params = Record<\n string,\n { type: string; description?: string; enum?: string[] }\n>;\n\n/**\n * Build an `ActionTool`. `parameters` is wrapped in the\n * `{ type:\"object\", properties, required }` shape `createMCPServerForRequest`\n * forwards verbatim as the MCP tool `inputSchema`.\n */\nfunction tool(\n description: string,\n parameters?: Params,\n required?: string[],\n): ActionTool {\n if (!parameters) return { description };\n return {\n description,\n parameters: {\n type: \"object\",\n properties: parameters,\n ...(required && required.length ? { required } : {}),\n },\n };\n}\n\n/**\n * The canonical app id this MCP server is mounted for. `MCPConfig.appId` is\n * authoritative; fall back to lowercasing `name` (which is the capitalized\n * app id at every call site) for back-compat with configs that predate the\n * `appId` field.\n */\nfunction currentAppId(config: MCPConfig): string {\n return (config.appId || config.name || \"app\").toLowerCase();\n}\n\n/**\n * Resolve the absolute origin of a *target* workspace app (e.g.\n * `http://127.0.0.1:8101`) so cross-app deep links / A2A calls point at the\n * right app instead of the current request's origin. Reuses the same\n * workspace resolution `list_apps` / the stdio proxy use.\n *\n * Returns `null` when:\n * - the target is the current app (caller should keep relative behavior),\n * - there is no workspace info (standalone / single app), or\n * - the target app is unknown.\n */\nasync function resolveTargetAppOrigin(\n config: MCPConfig,\n targetAppId: string,\n): Promise<{ origin: string; id: string } | null> {\n const target = targetAppId.trim().toLowerCase();\n if (!target || target === currentAppId(config)) return null;\n try {\n const { resolveWorkspace } = await import(\"./workspace-resolve.js\");\n const ws = await resolveWorkspace();\n if (!ws.isWorkspace) return null;\n const match = ws.apps.find((a) => a.id.toLowerCase() === target);\n if (!match) return null;\n return { origin: match.url, id: match.id };\n } catch {\n return null;\n }\n}\n\n// ---------------------------------------------------------------------------\n// list_apps\n// ---------------------------------------------------------------------------\n\nfunction listAppsTool(): ActionEntry {\n return {\n tool: tool(\n \"List the workspace apps and their local dev URLs/ports. Use this to \" +\n \"discover which apps exist before opening or asking one. In a single-\" +\n \"app project this returns just that app.\",\n ),\n readOnly: true,\n parallelSafe: true,\n run: async () => {\n const { resolveWorkspace } = await import(\"./workspace-resolve.js\");\n const ws = await resolveWorkspace();\n return {\n workspace: ws.isWorkspace,\n gatewayUrl: ws.gatewayUrl,\n apps: ws.apps.map((a) => ({\n id: a.id,\n url: a.url,\n port: a.port,\n running: a.running,\n })),\n };\n },\n };\n}\n\n// ---------------------------------------------------------------------------\n// open_app\n// ---------------------------------------------------------------------------\n\nfunction openAppTool(config: MCPConfig): ActionEntry {\n return {\n tool: tool(\n \"Build a deep link that opens an app at a specific view/record. No side \" +\n \"effects — returns a URL the user can click to land in the running UI. \" +\n 'After calling, surface the returned \"Open in … →\" link to the user.',\n {\n app: { type: \"string\", description: \"App id, e.g. 'mail'\" },\n view: {\n type: \"string\",\n description: \"Target view, e.g. 'inbox' (maps to navigate command)\",\n },\n params: {\n type: \"object\",\n description:\n \"Optional record-focus / filter params, e.g. { threadId: 'abc' }\",\n },\n },\n [\"app\", \"view\"],\n ),\n readOnly: true,\n parallelSafe: true,\n run: async (args: Record<string, any>) => {\n const app = String(args.app ?? \"\").trim();\n const view = String(args.view ?? \"\").trim();\n if (!app || !view) {\n throw new Error(\"open_app requires both 'app' and 'view'.\");\n }\n let params: Record<string, string | number | boolean> | undefined;\n const raw = args.params;\n if (raw && typeof raw === \"object\") {\n params = raw as Record<string, string | number | boolean>;\n } else if (typeof raw === \"string\" && raw.trim()) {\n try {\n params = JSON.parse(raw);\n } catch {\n params = undefined;\n }\n }\n const relUrl = buildDeepLink({ app, view, params });\n\n // Cross-app target in a workspace: resolve the TARGET app's origin and\n // return an absolute URL. Otherwise the MCP layer would prefix the\n // relative path with the CURRENT request origin, landing the user in\n // the wrong app (e.g. open_app({app:\"calendar\"}) served from Mail).\n // Same-app / standalone keeps the relative path (current behavior).\n const targetApp = await resolveTargetAppOrigin(config, app);\n const url = targetApp\n ? `${targetApp.origin.replace(/\\/+$/, \"\")}${relUrl}`\n : relUrl;\n\n return { app, view, url };\n },\n link: ({ result }) => {\n if (!result || typeof result !== \"object\") return null;\n const r = result as { url?: string; app?: string; view?: string };\n if (!r.url) return null;\n return {\n url: r.url,\n label: `Open ${r.app ?? \"app\"}`,\n view: r.view,\n };\n },\n };\n}\n\n// ---------------------------------------------------------------------------\n// ask_app\n// ---------------------------------------------------------------------------\n\nfunction askAppTool(config: MCPConfig): ActionEntry {\n return {\n tool: tool(\n \"Send a natural-language message to an app's AI agent and get its \" +\n \"response. Use for complex, multi-step tasks needing the agent's \" +\n \"reasoning and full app context. In a single-app project the 'app' \" +\n \"param is optional (defaults to this app). When 'app' names a \" +\n \"different workspace app it is routed there over A2A; the result's \" +\n \"'routedVia' field reports whether it ran cross-app or locally.\",\n {\n app: {\n type: \"string\",\n description: \"App id to route to (optional in a single-app project)\",\n },\n message: {\n type: \"string\",\n description: \"The message to send to the app's agent\",\n },\n },\n [\"message\"],\n ),\n run: async (args: Record<string, any>) => {\n const message = String(args.message ?? \"\").trim();\n if (!message) throw new Error(\"ask_app requires a 'message'.\");\n const requestedApp = String(args.app ?? \"\").trim();\n const selfId = currentAppId(config);\n\n // Cross-app: the caller named a *different* workspace app. Route the\n // message to THAT app's agent over A2A (its `/_agent-native/a2a`\n // endpoint runs the real agent loop with JWT identity) rather than\n // silently answering from this app's agent and claiming delegation.\n const targetApp = await resolveTargetAppOrigin(config, requestedApp);\n if (targetApp) {\n try {\n const { callAgent } = await import(\"../a2a/client.js\");\n const { getRequestUserEmail } =\n await import(\"../server/request-context.js\");\n // The MCP handler runs inside `runWithRequestContext`, so this is\n // the verified caller's email — it lets `callAgent` mint a signed\n // A2A JWT so the target app honours per-user scope.\n const response = await callAgent(targetApp.origin, message, {\n userEmail: getRequestUserEmail(),\n // Bound the wait — cross-app A2A polls async by default.\n timeoutMs: 5 * 60_000,\n });\n return {\n app: targetApp.id,\n routedVia: \"a2a\",\n response,\n };\n } catch (err: any) {\n // Be honest: routing was attempted and failed — do NOT fall back to\n // this app's agent and pretend it was the target.\n throw new Error(\n `Failed to route ask_app to \"${targetApp.id}\" via A2A: ` +\n `${err?.message ?? err}`,\n );\n }\n }\n\n // Same app (or no workspace / unknown target): answer locally with this\n // app's own ask-agent handler — the same entry point the HTTP MCP mount\n // + A2A use, so there is no second agent runner.\n if (!config.askAgent) {\n throw new Error(\n \"This app does not expose an agent (no ask-agent handler).\",\n );\n }\n\n // If the caller named an app we couldn't route to (unknown id, or no\n // workspace), say so honestly instead of claiming we reached it.\n const unresolved =\n !!requestedApp && requestedApp.toLowerCase() !== selfId;\n const response = await config.askAgent(message);\n return {\n app: selfId,\n routedVia: \"local\",\n ...(unresolved\n ? {\n note:\n `Requested app \"${requestedApp}\" is not a reachable workspace ` +\n `app; answered with this app (\"${selfId}\") instead.`,\n }\n : {}),\n response,\n };\n },\n };\n}\n\n// ---------------------------------------------------------------------------\n// list_templates\n// ---------------------------------------------------------------------------\n\nfunction listTemplatesTool(): ActionEntry {\n return {\n tool: tool(\n \"List the first-party templates that can be scaffolded into a workspace \" +\n \"(allow-listed templates only).\",\n ),\n readOnly: true,\n parallelSafe: true,\n run: async () => {\n const { visibleTemplates } = await import(\"../cli/templates-meta.js\");\n return {\n templates: visibleTemplates().map((t) => ({\n name: t.name,\n label: t.label,\n hint: t.hint,\n })),\n };\n },\n };\n}\n\n// ---------------------------------------------------------------------------\n// create_workspace_app\n// ---------------------------------------------------------------------------\n\nfunction createWorkspaceAppTool(): ActionEntry {\n return {\n tool: tool(\n \"Scaffold a new app into the current workspace from an allow-listed \" +\n \"template, then return a deep link to open it. Idempotent: if an app \" +\n \"with that name already exists it is reused. After calling, surface \" +\n 'the returned \"Open … →\" link to the user.',\n {\n name: {\n type: \"string\",\n description: \"New app id (directory under apps/), e.g. 'mymail'\",\n },\n template: {\n type: \"string\",\n description:\n \"Template to scaffold from — must be allow-listed (see list_templates)\",\n },\n },\n [\"name\", \"template\"],\n ),\n run: async (args: Record<string, any>) => {\n const name = String(args.name ?? \"\").trim();\n const template = String(args.template ?? \"\").trim();\n if (!name || !template) {\n throw new Error(\n \"create_workspace_app requires both 'name' and 'template'.\",\n );\n }\n\n // Enforce the strict public template allow-list. The authoritative,\n // dependency-free source inside @agent-native/core is cli/templates-meta\n // (kept in sync with packages/shared-app-config/templates.ts; CI guard).\n const { visibleTemplates } = await import(\"../cli/templates-meta.js\");\n const allowed = new Set(visibleTemplates().map((t) => t.name));\n if (!allowed.has(template)) {\n throw new Error(\n `Template \"${template}\" is not allow-listed. Allowed: ${[...allowed]\n .sort()\n .join(\", \")}`,\n );\n }\n\n const { findWorkspaceRoot, resolveWorkspace } =\n await import(\"./workspace-resolve.js\");\n const fs = await import(\"node:fs\");\n const path = await import(\"node:path\");\n\n const root = findWorkspaceRoot(process.cwd());\n if (!root) {\n throw new Error(\n \"Not inside a workspace. create_workspace_app only works in a \" +\n \"multi-app workspace (run from the workspace root).\",\n );\n }\n\n const appDir = path.join(root, \"apps\", name);\n const alreadyExisted = fs.existsSync(appDir);\n\n if (!alreadyExisted) {\n // Reuse the CLI scaffolder directly (no second `agent-native`\n // subprocess). `addAppToWorkspace(name, { template })` takes the\n // non-interactive single-template path when name + one template are\n // given. Run it from the workspace root so detectWorkspace resolves.\n const prevCwd = process.cwd();\n try {\n process.chdir(root);\n const { addAppToWorkspace } = await import(\"../cli/create.js\");\n await addAppToWorkspace(name, { template, noInstall: true });\n } finally {\n try {\n process.chdir(prevCwd);\n } catch {\n // best-effort cwd restore\n }\n }\n }\n\n // The workspace gateway auto-detects new apps/* dirs (fs.watch +\n // 2s sync) and lazily boots the dev server on first request, so we\n // don't spawn vite ourselves — opening the deep link warms it. Resolve\n // the port the gateway will use so we can report it.\n const ws = await resolveWorkspace(root);\n const appInfo = ws.apps.find((a) => a.id === name);\n const port = appInfo?.port;\n // The scaffolded app is always a *different* app from the host MCP\n // server, so anchor the deep link to the new app's own origin. A\n // relative path would otherwise be prefixed with the current request\n // origin and land on the wrong app. Fall back to the relative path\n // only if the gateway hasn't reported the new app's URL yet.\n const relDeepLink = buildDeepLink({ app: name, view: \"home\" });\n const deepLink = appInfo?.url\n ? `${appInfo.url.replace(/\\/+$/, \"\")}${relDeepLink}`\n : relDeepLink;\n\n return {\n name,\n template,\n created: !alreadyExisted,\n reused: alreadyExisted,\n port,\n url: appInfo?.url,\n gatewayUrl: ws.gatewayUrl,\n deepLink,\n };\n },\n link: ({ result }) => {\n if (!result || typeof result !== \"object\") return null;\n const r = result as { deepLink?: string; name?: string };\n if (!r.deepLink) return null;\n return {\n url: r.deepLink,\n label: `Open ${r.name ?? \"app\"}`,\n view: \"home\",\n };\n },\n };\n}\n\n// ---------------------------------------------------------------------------\n// Registry\n// ---------------------------------------------------------------------------\n\n/**\n * Build the generic cross-app builtin tool registry. Called by\n * `createMCPServerForRequest`; the result is merged UNDER the config's\n * actions so template actions of the same name win.\n */\nexport function getBuiltinCrossAppTools(\n config: MCPConfig,\n): Record<string, ActionEntry> {\n return {\n list_apps: listAppsTool(),\n open_app: openAppTool(config),\n ask_app: askAppTool(config),\n create_workspace_app: createWorkspaceAppTool(),\n list_templates: listTemplatesTool(),\n };\n}\n"]}
@@ -1,3 +1,10 @@
1
1
  export { mountMCP } from "./server.js";
2
2
  export type { MCPConfig } from "./server.js";
3
+ export { createMCPServerForRequest, verifyAuth, getAccessTokens, resolveOrgIdFromDomain, buildLinkArtifacts, } from "./build-server.js";
4
+ export type { MCPCallerIdentity, MCPRequestMeta } from "./build-server.js";
5
+ export { runMCPStdio } from "./stdio.js";
6
+ export type { RunMCPStdioOptions } from "./stdio.js";
7
+ export { getBuiltinCrossAppTools } from "./builtin-tools.js";
8
+ export { resolveWorkspace, resolveLocalAppOrigin, findWorkspaceRoot, } from "./workspace-resolve.js";
9
+ export type { ResolvedApp, ResolvedWorkspace } from "./workspace-resolve.js";
3
10
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/mcp/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AACvC,YAAY,EAAE,SAAS,EAAE,MAAM,aAAa,CAAC"}
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/mcp/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AACvC,YAAY,EAAE,SAAS,EAAE,MAAM,aAAa,CAAC;AAG7C,OAAO,EACL,yBAAyB,EACzB,UAAU,EACV,eAAe,EACf,sBAAsB,EACtB,kBAAkB,GACnB,MAAM,mBAAmB,CAAC;AAC3B,YAAY,EAAE,iBAAiB,EAAE,cAAc,EAAE,MAAM,mBAAmB,CAAC;AAG3E,OAAO,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AACzC,YAAY,EAAE,kBAAkB,EAAE,MAAM,YAAY,CAAC;AAGrD,OAAO,EAAE,uBAAuB,EAAE,MAAM,oBAAoB,CAAC;AAG7D,OAAO,EACL,gBAAgB,EAChB,qBAAqB,EACrB,iBAAiB,GAClB,MAAM,wBAAwB,CAAC;AAChC,YAAY,EAAE,WAAW,EAAE,iBAAiB,EAAE,MAAM,wBAAwB,CAAC"}
package/dist/mcp/index.js CHANGED
@@ -1,2 +1,10 @@
1
1
  export { mountMCP } from "./server.js";
2
+ // Shared MCP server builder (also re-exported from ./server.js for back-compat).
3
+ export { createMCPServerForRequest, verifyAuth, getAccessTokens, resolveOrgIdFromDomain, buildLinkArtifacts, } from "./build-server.js";
4
+ // stdio transport for `agent-native mcp serve` (Node-only).
5
+ export { runMCPStdio } from "./stdio.js";
6
+ // Generic cross-app builtin tools (merged into the registry, template wins).
7
+ export { getBuiltinCrossAppTools } from "./builtin-tools.js";
8
+ // Workspace / app resolution helpers (Node-only).
9
+ export { resolveWorkspace, resolveLocalAppOrigin, findWorkspaceRoot, } from "./workspace-resolve.js";
2
10
  //# sourceMappingURL=index.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/mcp/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC","sourcesContent":["export { mountMCP } from \"./server.js\";\nexport type { MCPConfig } from \"./server.js\";\n"]}
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/mcp/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AAGvC,iFAAiF;AACjF,OAAO,EACL,yBAAyB,EACzB,UAAU,EACV,eAAe,EACf,sBAAsB,EACtB,kBAAkB,GACnB,MAAM,mBAAmB,CAAC;AAG3B,4DAA4D;AAC5D,OAAO,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAGzC,6EAA6E;AAC7E,OAAO,EAAE,uBAAuB,EAAE,MAAM,oBAAoB,CAAC;AAE7D,kDAAkD;AAClD,OAAO,EACL,gBAAgB,EAChB,qBAAqB,EACrB,iBAAiB,GAClB,MAAM,wBAAwB,CAAC","sourcesContent":["export { mountMCP } from \"./server.js\";\nexport type { MCPConfig } from \"./server.js\";\n\n// Shared MCP server builder (also re-exported from ./server.js for back-compat).\nexport {\n createMCPServerForRequest,\n verifyAuth,\n getAccessTokens,\n resolveOrgIdFromDomain,\n buildLinkArtifacts,\n} from \"./build-server.js\";\nexport type { MCPCallerIdentity, MCPRequestMeta } from \"./build-server.js\";\n\n// stdio transport for `agent-native mcp serve` (Node-only).\nexport { runMCPStdio } from \"./stdio.js\";\nexport type { RunMCPStdioOptions } from \"./stdio.js\";\n\n// Generic cross-app builtin tools (merged into the registry, template wins).\nexport { getBuiltinCrossAppTools } from \"./builtin-tools.js\";\n\n// Workspace / app resolution helpers (Node-only).\nexport {\n resolveWorkspace,\n resolveLocalAppOrigin,\n findWorkspaceRoot,\n} from \"./workspace-resolve.js\";\nexport type { ResolvedApp, ResolvedWorkspace } from \"./workspace-resolve.js\";\n"]}
@@ -1,16 +1,6 @@
1
- import type { ActionEntry } from "../agent/production-agent.js";
2
- export interface MCPConfig {
3
- /** App name shown in MCP server info */
4
- name: string;
5
- /** App description */
6
- description: string;
7
- /** Version string (default "1.0.0") */
8
- version?: string;
9
- /** Action registry — same as agent chat and A2A */
10
- actions: Record<string, ActionEntry>;
11
- /** Handler for the ask-agent meta-tool — runs the full agent loop */
12
- askAgent?: (message: string) => Promise<string>;
13
- }
1
+ import { createMCPServerForRequest, verifyAuth, getAccessTokens, resolveOrgIdFromDomain, buildLinkArtifacts, type MCPConfig, type MCPCallerIdentity, type MCPRequestMeta } from "./build-server.js";
2
+ export { createMCPServerForRequest, verifyAuth, getAccessTokens, resolveOrgIdFromDomain, buildLinkArtifacts, };
3
+ export type { MCPConfig, MCPCallerIdentity, MCPRequestMeta };
14
4
  /**
15
5
  * Mount an MCP remote server on an H3/Nitro app.
16
6
  *
@@ -1 +1 @@
1
- {"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../../src/mcp/server.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,8BAA8B,CAAC;AAIhE,MAAM,WAAW,SAAS;IACxB,wCAAwC;IACxC,IAAI,EAAE,MAAM,CAAC;IACb,sBAAsB;IACtB,WAAW,EAAE,MAAM,CAAC;IACpB,uCAAuC;IACvC,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,mDAAmD;IACnD,OAAO,EAAE,MAAM,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC;IACrC,qEAAqE;IACrE,QAAQ,CAAC,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,OAAO,CAAC,MAAM,CAAC,CAAC;CACjD;AAkOD;;;;;;;;;;GAUG;AACH,wBAAgB,QAAQ,CACtB,QAAQ,EAAE,GAAG,EACb,MAAM,EAAE,SAAS,EACjB,WAAW,SAAmB,GAC7B,IAAI,CA4EN"}
1
+ {"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../../src/mcp/server.ts"],"names":[],"mappings":"AAQA,OAAO,EACL,yBAAyB,EACzB,UAAU,EACV,eAAe,EACf,sBAAsB,EACtB,kBAAkB,EAClB,KAAK,SAAS,EACd,KAAK,iBAAiB,EACtB,KAAK,cAAc,EACpB,MAAM,mBAAmB,CAAC;AAK3B,OAAO,EACL,yBAAyB,EACzB,UAAU,EACV,eAAe,EACf,sBAAsB,EACtB,kBAAkB,GACnB,CAAC;AACF,YAAY,EAAE,SAAS,EAAE,iBAAiB,EAAE,cAAc,EAAE,CAAC;AAM7D;;;;;;;;;;GAUG;AACH,wBAAgB,QAAQ,CACtB,QAAQ,EAAE,GAAG,EACb,MAAM,EAAE,SAAS,EACjB,WAAW,SAAmB,GAC7B,IAAI,CAwHN"}