@agent-native/dispatch 0.8.5 → 0.8.7

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 (97) hide show
  1. package/dist/actions/ask_app.d.ts +3 -0
  2. package/dist/actions/ask_app.d.ts.map +1 -0
  3. package/dist/actions/ask_app.js +12 -0
  4. package/dist/actions/ask_app.js.map +1 -0
  5. package/dist/actions/create_embed_session.d.ts +3 -0
  6. package/dist/actions/create_embed_session.d.ts.map +1 -0
  7. package/dist/actions/create_embed_session.js +28 -0
  8. package/dist/actions/create_embed_session.js.map +1 -0
  9. package/dist/actions/index.d.ts.map +1 -1
  10. package/dist/actions/index.js +12 -0
  11. package/dist/actions/index.js.map +1 -1
  12. package/dist/actions/list-mcp-app-access.d.ts +3 -0
  13. package/dist/actions/list-mcp-app-access.d.ts.map +1 -0
  14. package/dist/actions/list-mcp-app-access.js +25 -0
  15. package/dist/actions/list-mcp-app-access.js.map +1 -0
  16. package/dist/actions/list_apps.d.ts +3 -0
  17. package/dist/actions/list_apps.d.ts.map +1 -0
  18. package/dist/actions/list_apps.js +26 -0
  19. package/dist/actions/list_apps.js.map +1 -0
  20. package/dist/actions/open_app.d.ts +3 -0
  21. package/dist/actions/open_app.d.ts.map +1 -0
  22. package/dist/actions/open_app.js +62 -0
  23. package/dist/actions/open_app.js.map +1 -0
  24. package/dist/actions/set-mcp-app-access.d.ts +3 -0
  25. package/dist/actions/set-mcp-app-access.d.ts.map +1 -0
  26. package/dist/actions/set-mcp-app-access.js +46 -0
  27. package/dist/actions/set-mcp-app-access.js.map +1 -0
  28. package/dist/actions/start-workspace-app-creation.js +1 -1
  29. package/dist/actions/start-workspace-app-creation.js.map +1 -1
  30. package/dist/actions/view-screen.d.ts.map +1 -1
  31. package/dist/actions/view-screen.js +8 -0
  32. package/dist/actions/view-screen.js.map +1 -1
  33. package/dist/components/create-app-popover.d.ts.map +1 -1
  34. package/dist/components/create-app-popover.js +1 -0
  35. package/dist/components/create-app-popover.js.map +1 -1
  36. package/dist/components/workspace-app-card.d.ts.map +1 -1
  37. package/dist/components/workspace-app-card.js +2 -1
  38. package/dist/components/workspace-app-card.js.map +1 -1
  39. package/dist/routes/pages/agents.d.ts.map +1 -1
  40. package/dist/routes/pages/agents.js +74 -3
  41. package/dist/routes/pages/agents.js.map +1 -1
  42. package/dist/routes/pages/apps.d.ts.map +1 -1
  43. package/dist/routes/pages/apps.js +23 -8
  44. package/dist/routes/pages/apps.js.map +1 -1
  45. package/dist/routes/pages/metrics.d.ts.map +1 -1
  46. package/dist/routes/pages/metrics.js +3 -1
  47. package/dist/routes/pages/metrics.js.map +1 -1
  48. package/dist/routes/pages/overview.d.ts.map +1 -1
  49. package/dist/routes/pages/overview.js +1 -3
  50. package/dist/routes/pages/overview.js.map +1 -1
  51. package/dist/server/lib/app-creation-store.d.ts.map +1 -1
  52. package/dist/server/lib/app-creation-store.js +104 -10
  53. package/dist/server/lib/app-creation-store.js.map +1 -1
  54. package/dist/server/lib/mcp-access-store.d.ts +16 -0
  55. package/dist/server/lib/mcp-access-store.d.ts.map +1 -0
  56. package/dist/server/lib/mcp-access-store.js +64 -0
  57. package/dist/server/lib/mcp-access-store.js.map +1 -0
  58. package/dist/server/lib/mcp-gateway.d.ts +47 -0
  59. package/dist/server/lib/mcp-gateway.d.ts.map +1 -0
  60. package/dist/server/lib/mcp-gateway.js +393 -0
  61. package/dist/server/lib/mcp-gateway.js.map +1 -0
  62. package/dist/server/lib/usage-metrics-store.d.ts +1 -0
  63. package/dist/server/lib/usage-metrics-store.d.ts.map +1 -1
  64. package/dist/server/lib/usage-metrics-store.js +1 -0
  65. package/dist/server/lib/usage-metrics-store.js.map +1 -1
  66. package/dist/server/plugins/agent-chat.d.ts.map +1 -1
  67. package/dist/server/plugins/agent-chat.js +1 -0
  68. package/dist/server/plugins/agent-chat.js.map +1 -1
  69. package/dist/server/plugins/integrations.d.ts.map +1 -1
  70. package/dist/server/plugins/integrations.js +2 -1
  71. package/dist/server/plugins/integrations.js.map +1 -1
  72. package/package.json +1 -1
  73. package/src/actions/ask_app.ts +13 -0
  74. package/src/actions/create_embed_session.ts +29 -0
  75. package/src/actions/index.spec.ts +6 -0
  76. package/src/actions/index.ts +12 -0
  77. package/src/actions/list-mcp-app-access.ts +26 -0
  78. package/src/actions/list_apps.ts +27 -0
  79. package/src/actions/open_app.ts +68 -0
  80. package/src/actions/set-mcp-app-access.ts +59 -0
  81. package/src/actions/start-workspace-app-creation.ts +1 -1
  82. package/src/actions/view-screen.ts +8 -0
  83. package/src/components/create-app-popover.tsx +1 -0
  84. package/src/components/workspace-app-card.tsx +3 -2
  85. package/src/routes/pages/agents.tsx +187 -5
  86. package/src/routes/pages/apps.tsx +209 -67
  87. package/src/routes/pages/metrics.tsx +4 -1
  88. package/src/routes/pages/overview.tsx +16 -10
  89. package/src/server/lib/app-creation-store.spec.ts +240 -0
  90. package/src/server/lib/app-creation-store.ts +130 -11
  91. package/src/server/lib/mcp-access-store.spec.ts +58 -0
  92. package/src/server/lib/mcp-access-store.ts +104 -0
  93. package/src/server/lib/mcp-gateway.spec.ts +295 -0
  94. package/src/server/lib/mcp-gateway.ts +516 -0
  95. package/src/server/lib/usage-metrics-store.ts +2 -0
  96. package/src/server/plugins/agent-chat.ts +1 -0
  97. package/src/server/plugins/integrations.ts +2 -1
@@ -0,0 +1,295 @@
1
+ import { afterEach, beforeEach, describe, expect, it, vi } from "vitest";
2
+
3
+ const mocks = vi.hoisted(() => ({
4
+ discoverAgents: vi.fn(),
5
+ getUserSetting: vi.fn(),
6
+ getOrgSetting: vi.fn(),
7
+ createEmbedSessionTicket: vi.fn(),
8
+ buildEmbedStartPath: vi.fn((ticket: string) => {
9
+ return `/_agent-native/embed/start?ticket=${encodeURIComponent(ticket)}`;
10
+ }),
11
+ managerStart: vi.fn(),
12
+ managerStop: vi.fn(),
13
+ managerCallTool: vi.fn(),
14
+ managerConstructor: vi.fn(),
15
+ signA2AToken: vi.fn(),
16
+ getOrgA2ASecret: vi.fn(),
17
+ getOrgDomain: vi.fn(),
18
+ }));
19
+
20
+ vi.mock("@agent-native/core/server/agent-discovery", () => ({
21
+ discoverAgents: mocks.discoverAgents,
22
+ }));
23
+
24
+ vi.mock("@agent-native/core/settings", () => ({
25
+ getUserSetting: mocks.getUserSetting,
26
+ getOrgSetting: mocks.getOrgSetting,
27
+ putUserSetting: vi.fn(),
28
+ putOrgSetting: vi.fn(),
29
+ }));
30
+
31
+ vi.mock("@agent-native/core/server", async (importOriginal) => {
32
+ const actual =
33
+ await importOriginal<typeof import("@agent-native/core/server")>();
34
+ return {
35
+ ...actual,
36
+ createEmbedSessionTicket: mocks.createEmbedSessionTicket,
37
+ buildEmbedStartPath: mocks.buildEmbedStartPath,
38
+ };
39
+ });
40
+
41
+ vi.mock("@agent-native/core/a2a", () => ({
42
+ callAgent: vi.fn(),
43
+ signA2AToken: mocks.signA2AToken,
44
+ }));
45
+
46
+ vi.mock("@agent-native/core/org", () => ({
47
+ getOrgA2ASecret: mocks.getOrgA2ASecret,
48
+ getOrgDomain: mocks.getOrgDomain,
49
+ }));
50
+
51
+ vi.mock("@agent-native/core/mcp-client", () => ({
52
+ buildMcpToolName: (serverId: string, toolName: string) =>
53
+ `mcp__${serverId}__${toolName}`,
54
+ McpClientManager: class MockMcpClientManager {
55
+ constructor(config: unknown) {
56
+ mocks.managerConstructor(config);
57
+ }
58
+
59
+ start() {
60
+ return mocks.managerStart();
61
+ }
62
+
63
+ stop() {
64
+ return mocks.managerStop();
65
+ }
66
+
67
+ callTool(name: string, args: unknown) {
68
+ return mocks.managerCallTool(name, args);
69
+ }
70
+ },
71
+ }));
72
+
73
+ import {
74
+ createGrantedDispatchMcpEmbedSession,
75
+ listGrantedDispatchMcpApps,
76
+ openGrantedDispatchMcpApp,
77
+ } from "./mcp-gateway.js";
78
+ import { runWithRequestContext } from "@agent-native/core/server";
79
+
80
+ const analyticsAgent = {
81
+ id: "analytics",
82
+ name: "Analytics",
83
+ description: "Dashboards and metrics",
84
+ url: "http://localhost:8086",
85
+ color: "#6366F1",
86
+ };
87
+
88
+ beforeEach(() => {
89
+ mocks.discoverAgents.mockResolvedValue([analyticsAgent]);
90
+ mocks.getUserSetting.mockResolvedValue(null);
91
+ mocks.getOrgSetting.mockResolvedValue(null);
92
+ mocks.createEmbedSessionTicket.mockResolvedValue({
93
+ ticket: "ticket-123",
94
+ ticketHash: "hash-123",
95
+ expiresAt: 12345,
96
+ });
97
+ mocks.managerStart.mockResolvedValue(undefined);
98
+ mocks.managerStop.mockResolvedValue(undefined);
99
+ mocks.managerCallTool.mockResolvedValue({
100
+ structuredContent: {
101
+ startUrl: "http://localhost:8086/_agent-native/embed/start?ticket=remote",
102
+ },
103
+ });
104
+ mocks.signA2AToken.mockResolvedValue("signed-token");
105
+ mocks.getOrgA2ASecret.mockResolvedValue(null);
106
+ mocks.getOrgDomain.mockResolvedValue(null);
107
+ });
108
+
109
+ afterEach(() => {
110
+ vi.unstubAllEnvs();
111
+ vi.clearAllMocks();
112
+ });
113
+
114
+ describe("Dispatch MCP gateway app discovery", () => {
115
+ it("includes Dispatch itself so agents can target extension routes", async () => {
116
+ const apps = await runWithRequestContext(
117
+ {
118
+ userEmail: "owner@example.test",
119
+ requestOrigin: "http://localhost:8092",
120
+ },
121
+ () => listGrantedDispatchMcpApps(),
122
+ );
123
+
124
+ expect(apps.map((app) => app.id)).toEqual(["dispatch", "analytics"]);
125
+ expect(apps[0]).toMatchObject({
126
+ id: "dispatch",
127
+ name: "Agent-Native Dispatch",
128
+ url: "http://localhost:8092",
129
+ granted: true,
130
+ });
131
+ });
132
+
133
+ it("honors selected app grants for the Dispatch self target", async () => {
134
+ mocks.getUserSetting.mockResolvedValue({
135
+ mode: "selected-apps",
136
+ selectedAppIds: ["dispatch"],
137
+ });
138
+
139
+ const apps = await runWithRequestContext(
140
+ {
141
+ userEmail: "owner@example.test",
142
+ requestOrigin: "http://localhost:8092",
143
+ },
144
+ () => listGrantedDispatchMcpApps(),
145
+ );
146
+
147
+ expect(apps.map((app) => app.id)).toEqual(["dispatch"]);
148
+ });
149
+ });
150
+
151
+ describe("openGrantedDispatchMcpApp", () => {
152
+ it("opens Dispatch extension routes through the Dispatch app id", async () => {
153
+ const result = await runWithRequestContext(
154
+ {
155
+ userEmail: "owner@example.test",
156
+ requestOrigin: "http://localhost:8092",
157
+ },
158
+ () =>
159
+ openGrantedDispatchMcpApp({
160
+ app: "dispatch",
161
+ path: "/extensions/ext-1/github-stars-over-time",
162
+ embed: true,
163
+ chrome: "minimal",
164
+ }),
165
+ );
166
+
167
+ expect(result).toEqual({
168
+ app: "dispatch",
169
+ path: "/extensions/ext-1/github-stars-over-time",
170
+ url: "http://localhost:8092/extensions/ext-1/github-stars-over-time",
171
+ embed: true,
172
+ chrome: "minimal",
173
+ });
174
+ });
175
+
176
+ it("rejects Dispatch-owned extension routes on sibling apps", async () => {
177
+ await expect(
178
+ runWithRequestContext(
179
+ {
180
+ userEmail: "owner@example.test",
181
+ requestOrigin: "http://localhost:8092",
182
+ },
183
+ () =>
184
+ openGrantedDispatchMcpApp({
185
+ app: "analytics",
186
+ path: "/extensions/ext-1/github-stars-over-time",
187
+ }),
188
+ ),
189
+ ).rejects.toThrow(/belongs to Dispatch/);
190
+ });
191
+
192
+ it("rejects traversal that normalizes into Dispatch-owned routes on sibling apps", async () => {
193
+ await expect(
194
+ runWithRequestContext(
195
+ {
196
+ userEmail: "owner@example.test",
197
+ requestOrigin: "http://localhost:8092",
198
+ },
199
+ () =>
200
+ openGrantedDispatchMcpApp({
201
+ app: "analytics",
202
+ path: "/../dispatch/extensions/ext-1",
203
+ }),
204
+ ),
205
+ ).rejects.toThrow(/safe app-relative route/);
206
+ });
207
+ });
208
+
209
+ describe("createGrantedDispatchMcpEmbedSession", () => {
210
+ it("mints Dispatch self embeds locally instead of recursively calling Dispatch MCP", async () => {
211
+ const result = await runWithRequestContext(
212
+ {
213
+ userEmail: "owner@example.test",
214
+ requestOrigin: "http://localhost:8092",
215
+ },
216
+ () =>
217
+ createGrantedDispatchMcpEmbedSession({
218
+ app: "dispatch",
219
+ path: "/extensions/ext-1/github-stars-over-time",
220
+ chrome: "minimal",
221
+ }),
222
+ );
223
+
224
+ expect(mocks.createEmbedSessionTicket).toHaveBeenCalledWith({
225
+ ownerEmail: "owner@example.test",
226
+ orgId: undefined,
227
+ targetPath: "/extensions/ext-1/github-stars-over-time",
228
+ scope: "minimal",
229
+ });
230
+ expect(mocks.managerConstructor).not.toHaveBeenCalled();
231
+ expect(result).toEqual({
232
+ app: "dispatch",
233
+ startUrl:
234
+ "http://localhost:8092/_agent-native/embed/start?ticket=ticket-123",
235
+ targetPath: "/extensions/ext-1/github-stars-over-time",
236
+ expiresAt: 12345,
237
+ });
238
+ });
239
+
240
+ it("rejects traversal into Dispatch-owned embed routes on sibling apps", async () => {
241
+ await expect(
242
+ runWithRequestContext(
243
+ {
244
+ userEmail: "owner@example.test",
245
+ requestOrigin: "http://localhost:8092",
246
+ },
247
+ () =>
248
+ createGrantedDispatchMcpEmbedSession({
249
+ app: "analytics",
250
+ path: "/../dispatch/extensions/ext-1",
251
+ }),
252
+ ),
253
+ ).rejects.toThrow(/safe app-relative route/);
254
+ });
255
+
256
+ it("routes same-origin mounted app embed URLs to the mounted app", async () => {
257
+ mocks.discoverAgents.mockResolvedValue([
258
+ {
259
+ ...analyticsAgent,
260
+ url: "http://localhost:8092/analytics",
261
+ },
262
+ ]);
263
+
264
+ const result = await runWithRequestContext(
265
+ {
266
+ userEmail: "owner@example.test",
267
+ requestOrigin: "http://localhost:8092",
268
+ },
269
+ () =>
270
+ createGrantedDispatchMcpEmbedSession({
271
+ url: "http://localhost:8092/analytics/dashboards?range=30d",
272
+ }),
273
+ );
274
+
275
+ expect(mocks.createEmbedSessionTicket).not.toHaveBeenCalled();
276
+ expect(mocks.managerConstructor).toHaveBeenCalledWith({
277
+ servers: {
278
+ target: expect.objectContaining({
279
+ url: "http://localhost:8092/analytics/_agent-native/mcp",
280
+ }),
281
+ },
282
+ });
283
+ expect(mocks.managerCallTool).toHaveBeenCalledWith(
284
+ "mcp__target__create_embed_session",
285
+ {
286
+ url: "http://localhost:8092/analytics/dashboards?range=30d",
287
+ chrome: "full",
288
+ },
289
+ );
290
+ expect(result).toEqual({
291
+ app: "analytics",
292
+ startUrl: "http://localhost:8086/_agent-native/embed/start?ticket=remote",
293
+ });
294
+ });
295
+ });