@mastra/client-js 0.1.0-alpha.1 → 0.1.0-alpha.11

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 (294) hide show
  1. package/.eslintrc.js +10 -0
  2. package/.prettierignore +3 -0
  3. package/.prettierrc.json +26 -0
  4. package/LICENSE +44 -201
  5. package/README.md +90 -271
  6. package/dist/index.d.mts +359 -0
  7. package/dist/index.mjs +410 -0
  8. package/package.json +30 -93
  9. package/src/client.ts +135 -0
  10. package/src/example.ts +47 -0
  11. package/src/index.test.ts +585 -0
  12. package/src/index.ts +2 -194
  13. package/src/resources/agent.ts +115 -0
  14. package/src/resources/base.ts +68 -0
  15. package/src/resources/index.ts +6 -9
  16. package/src/resources/memory-thread.ts +57 -0
  17. package/src/resources/tool.ts +31 -0
  18. package/src/resources/vector.ts +82 -0
  19. package/src/resources/workflow.ts +31 -0
  20. package/src/types.ts +142 -0
  21. package/tsconfig.json +29 -0
  22. package/vitest.config.js +8 -0
  23. package/CHANGELOG.md +0 -21
  24. package/_shims/MultipartBody.d.ts +0 -9
  25. package/_shims/MultipartBody.d.ts.map +0 -1
  26. package/_shims/MultipartBody.js +0 -16
  27. package/_shims/MultipartBody.js.map +0 -1
  28. package/_shims/MultipartBody.mjs +0 -12
  29. package/_shims/MultipartBody.mjs.map +0 -1
  30. package/_shims/README.md +0 -46
  31. package/_shims/auto/runtime-bun.d.ts +0 -5
  32. package/_shims/auto/runtime-bun.d.ts.map +0 -1
  33. package/_shims/auto/runtime-bun.js +0 -21
  34. package/_shims/auto/runtime-bun.js.map +0 -1
  35. package/_shims/auto/runtime-bun.mjs +0 -2
  36. package/_shims/auto/runtime-bun.mjs.map +0 -1
  37. package/_shims/auto/runtime-node.d.ts +0 -5
  38. package/_shims/auto/runtime-node.d.ts.map +0 -1
  39. package/_shims/auto/runtime-node.js +0 -21
  40. package/_shims/auto/runtime-node.js.map +0 -1
  41. package/_shims/auto/runtime-node.mjs +0 -2
  42. package/_shims/auto/runtime-node.mjs.map +0 -1
  43. package/_shims/auto/runtime.d.ts +0 -5
  44. package/_shims/auto/runtime.d.ts.map +0 -1
  45. package/_shims/auto/runtime.js +0 -21
  46. package/_shims/auto/runtime.js.map +0 -1
  47. package/_shims/auto/runtime.mjs +0 -2
  48. package/_shims/auto/runtime.mjs.map +0 -1
  49. package/_shims/auto/types-node.d.ts +0 -5
  50. package/_shims/auto/types-node.d.ts.map +0 -1
  51. package/_shims/auto/types-node.js +0 -21
  52. package/_shims/auto/types-node.js.map +0 -1
  53. package/_shims/auto/types-node.mjs +0 -2
  54. package/_shims/auto/types-node.mjs.map +0 -1
  55. package/_shims/auto/types.d.ts +0 -101
  56. package/_shims/auto/types.js +0 -3
  57. package/_shims/auto/types.mjs +0 -3
  58. package/_shims/bun-runtime.d.ts +0 -6
  59. package/_shims/bun-runtime.d.ts.map +0 -1
  60. package/_shims/bun-runtime.js +0 -14
  61. package/_shims/bun-runtime.js.map +0 -1
  62. package/_shims/bun-runtime.mjs +0 -10
  63. package/_shims/bun-runtime.mjs.map +0 -1
  64. package/_shims/index.d.ts +0 -81
  65. package/_shims/index.js +0 -13
  66. package/_shims/index.mjs +0 -7
  67. package/_shims/manual-types.d.ts +0 -12
  68. package/_shims/manual-types.js +0 -3
  69. package/_shims/manual-types.mjs +0 -3
  70. package/_shims/node-runtime.d.ts +0 -3
  71. package/_shims/node-runtime.d.ts.map +0 -1
  72. package/_shims/node-runtime.js +0 -89
  73. package/_shims/node-runtime.js.map +0 -1
  74. package/_shims/node-runtime.mjs +0 -56
  75. package/_shims/node-runtime.mjs.map +0 -1
  76. package/_shims/node-types.d.ts +0 -42
  77. package/_shims/node-types.js +0 -3
  78. package/_shims/node-types.mjs +0 -3
  79. package/_shims/registry.d.ts +0 -37
  80. package/_shims/registry.d.ts.map +0 -1
  81. package/_shims/registry.js +0 -41
  82. package/_shims/registry.js.map +0 -1
  83. package/_shims/registry.mjs +0 -37
  84. package/_shims/registry.mjs.map +0 -1
  85. package/_shims/web-runtime.d.ts +0 -5
  86. package/_shims/web-runtime.d.ts.map +0 -1
  87. package/_shims/web-runtime.js +0 -78
  88. package/_shims/web-runtime.js.map +0 -1
  89. package/_shims/web-runtime.mjs +0 -71
  90. package/_shims/web-runtime.mjs.map +0 -1
  91. package/_shims/web-types.d.ts +0 -83
  92. package/_shims/web-types.js +0 -3
  93. package/_shims/web-types.mjs +0 -3
  94. package/core.d.ts +0 -241
  95. package/core.d.ts.map +0 -1
  96. package/core.js +0 -908
  97. package/core.js.map +0 -1
  98. package/core.mjs +0 -876
  99. package/core.mjs.map +0 -1
  100. package/error.d.ts +0 -47
  101. package/error.d.ts.map +0 -1
  102. package/error.js +0 -113
  103. package/error.js.map +0 -1
  104. package/error.mjs +0 -97
  105. package/error.mjs.map +0 -1
  106. package/index.d.mts +0 -121
  107. package/index.d.ts +0 -121
  108. package/index.d.ts.map +0 -1
  109. package/index.js +0 -131
  110. package/index.js.map +0 -1
  111. package/index.mjs +0 -88
  112. package/index.mjs.map +0 -1
  113. package/resource.d.ts +0 -6
  114. package/resource.d.ts.map +0 -1
  115. package/resource.js +0 -11
  116. package/resource.js.map +0 -1
  117. package/resource.mjs +0 -7
  118. package/resource.mjs.map +0 -1
  119. package/resources/agents/agents.d.ts +0 -40
  120. package/resources/agents/agents.d.ts.map +0 -1
  121. package/resources/agents/agents.js +0 -74
  122. package/resources/agents/agents.js.map +0 -1
  123. package/resources/agents/agents.mjs +0 -47
  124. package/resources/agents/agents.mjs.map +0 -1
  125. package/resources/agents/index.d.ts +0 -3
  126. package/resources/agents/index.d.ts.map +0 -1
  127. package/resources/agents/index.js +0 -9
  128. package/resources/agents/index.js.map +0 -1
  129. package/resources/agents/index.mjs +0 -4
  130. package/resources/agents/index.mjs.map +0 -1
  131. package/resources/agents/tools.d.ts +0 -17
  132. package/resources/agents/tools.d.ts.map +0 -1
  133. package/resources/agents/tools.js +0 -19
  134. package/resources/agents/tools.js.map +0 -1
  135. package/resources/agents/tools.mjs +0 -15
  136. package/resources/agents/tools.mjs.map +0 -1
  137. package/resources/index.d.ts +0 -8
  138. package/resources/index.d.ts.map +0 -1
  139. package/resources/index.js +0 -19
  140. package/resources/index.js.map +0 -1
  141. package/resources/index.mjs +0 -9
  142. package/resources/index.mjs.map +0 -1
  143. package/resources/logs.d.ts +0 -13
  144. package/resources/logs.d.ts.map +0 -1
  145. package/resources/logs.js +0 -24
  146. package/resources/logs.js.map +0 -1
  147. package/resources/logs.mjs +0 -20
  148. package/resources/logs.mjs.map +0 -1
  149. package/resources/memory/index.d.ts +0 -4
  150. package/resources/memory/index.d.ts.map +0 -1
  151. package/resources/memory/index.js +0 -11
  152. package/resources/memory/index.js.map +0 -1
  153. package/resources/memory/index.mjs +0 -5
  154. package/resources/memory/index.mjs.map +0 -1
  155. package/resources/memory/memory.d.ts +0 -23
  156. package/resources/memory/memory.d.ts.map +0 -1
  157. package/resources/memory/memory.js +0 -53
  158. package/resources/memory/memory.js.map +0 -1
  159. package/resources/memory/memory.mjs +0 -26
  160. package/resources/memory/memory.mjs.map +0 -1
  161. package/resources/memory/status.d.ts +0 -9
  162. package/resources/memory/status.d.ts.map +0 -1
  163. package/resources/memory/status.js +0 -18
  164. package/resources/memory/status.js.map +0 -1
  165. package/resources/memory/status.mjs +0 -14
  166. package/resources/memory/status.mjs.map +0 -1
  167. package/resources/memory/threads/index.d.ts +0 -3
  168. package/resources/memory/threads/index.d.ts.map +0 -1
  169. package/resources/memory/threads/index.js +0 -9
  170. package/resources/memory/threads/index.js.map +0 -1
  171. package/resources/memory/threads/index.mjs +0 -4
  172. package/resources/memory/threads/index.mjs.map +0 -1
  173. package/resources/memory/threads/messages.d.ts +0 -9
  174. package/resources/memory/threads/messages.d.ts.map +0 -1
  175. package/resources/memory/threads/messages.js +0 -18
  176. package/resources/memory/threads/messages.js.map +0 -1
  177. package/resources/memory/threads/messages.mjs +0 -14
  178. package/resources/memory/threads/messages.mjs.map +0 -1
  179. package/resources/memory/threads/threads.d.ts +0 -45
  180. package/resources/memory/threads/threads.d.ts.map +0 -1
  181. package/resources/memory/threads/threads.js +0 -104
  182. package/resources/memory/threads/threads.js.map +0 -1
  183. package/resources/memory/threads/threads.mjs +0 -77
  184. package/resources/memory/threads/threads.mjs.map +0 -1
  185. package/resources/syncs.d.ts +0 -15
  186. package/resources/syncs.d.ts.map +0 -1
  187. package/resources/syncs.js +0 -19
  188. package/resources/syncs.js.map +0 -1
  189. package/resources/syncs.mjs +0 -15
  190. package/resources/syncs.mjs.map +0 -1
  191. package/resources/system.d.ts +0 -9
  192. package/resources/system.d.ts.map +0 -1
  193. package/resources/system.js +0 -15
  194. package/resources/system.js.map +0 -1
  195. package/resources/system.mjs +0 -11
  196. package/resources/system.mjs.map +0 -1
  197. package/resources/tools/index.d.ts +0 -3
  198. package/resources/tools/index.d.ts.map +0 -1
  199. package/resources/tools/index.js +0 -9
  200. package/resources/tools/index.js.map +0 -1
  201. package/resources/tools/index.mjs +0 -4
  202. package/resources/tools/index.mjs.map +0 -1
  203. package/resources/tools/result.d.ts +0 -9
  204. package/resources/tools/result.d.ts.map +0 -1
  205. package/resources/tools/result.js +0 -18
  206. package/resources/tools/result.js.map +0 -1
  207. package/resources/tools/result.mjs +0 -14
  208. package/resources/tools/result.mjs.map +0 -1
  209. package/resources/tools/tools.d.ts +0 -29
  210. package/resources/tools/tools.d.ts.map +0 -1
  211. package/resources/tools/tools.js +0 -64
  212. package/resources/tools/tools.js.map +0 -1
  213. package/resources/tools/tools.mjs +0 -37
  214. package/resources/tools/tools.mjs.map +0 -1
  215. package/resources/workflows.d.ts +0 -23
  216. package/resources/workflows.d.ts.map +0 -1
  217. package/resources/workflows.js +0 -37
  218. package/resources/workflows.js.map +0 -1
  219. package/resources/workflows.mjs +0 -33
  220. package/resources/workflows.mjs.map +0 -1
  221. package/shims/node.d.ts +0 -30
  222. package/shims/node.d.ts.map +0 -1
  223. package/shims/node.js +0 -31
  224. package/shims/node.js.map +0 -1
  225. package/shims/node.mjs +0 -5
  226. package/shims/node.mjs.map +0 -1
  227. package/shims/web.d.ts +0 -26
  228. package/shims/web.d.ts.map +0 -1
  229. package/shims/web.js +0 -31
  230. package/shims/web.js.map +0 -1
  231. package/shims/web.mjs +0 -5
  232. package/shims/web.mjs.map +0 -1
  233. package/src/_shims/MultipartBody.ts +0 -9
  234. package/src/_shims/README.md +0 -46
  235. package/src/_shims/auto/runtime-bun.ts +0 -4
  236. package/src/_shims/auto/runtime-node.ts +0 -4
  237. package/src/_shims/auto/runtime.ts +0 -4
  238. package/src/_shims/auto/types-node.ts +0 -4
  239. package/src/_shims/auto/types.d.ts +0 -101
  240. package/src/_shims/auto/types.js +0 -3
  241. package/src/_shims/auto/types.mjs +0 -3
  242. package/src/_shims/bun-runtime.ts +0 -14
  243. package/src/_shims/index.d.ts +0 -81
  244. package/src/_shims/index.js +0 -13
  245. package/src/_shims/index.mjs +0 -7
  246. package/src/_shims/manual-types.d.ts +0 -12
  247. package/src/_shims/manual-types.js +0 -3
  248. package/src/_shims/manual-types.mjs +0 -3
  249. package/src/_shims/node-runtime.ts +0 -81
  250. package/src/_shims/node-types.d.ts +0 -42
  251. package/src/_shims/node-types.js +0 -3
  252. package/src/_shims/node-types.mjs +0 -3
  253. package/src/_shims/registry.ts +0 -67
  254. package/src/_shims/web-runtime.ts +0 -103
  255. package/src/_shims/web-types.d.ts +0 -83
  256. package/src/_shims/web-types.js +0 -3
  257. package/src/_shims/web-types.mjs +0 -3
  258. package/src/core.ts +0 -1201
  259. package/src/error.ts +0 -130
  260. package/src/lib/.keep +0 -4
  261. package/src/resource.ts +0 -11
  262. package/src/resources/agents/agents.ts +0 -77
  263. package/src/resources/agents/index.ts +0 -4
  264. package/src/resources/agents/tools.ts +0 -34
  265. package/src/resources/logs.ts +0 -23
  266. package/src/resources/memory/index.ts +0 -5
  267. package/src/resources/memory/memory.ts +0 -43
  268. package/src/resources/memory/status.ts +0 -16
  269. package/src/resources/memory/threads/index.ts +0 -4
  270. package/src/resources/memory/threads/messages.ts +0 -16
  271. package/src/resources/memory/threads/threads.ts +0 -105
  272. package/src/resources/syncs.ts +0 -25
  273. package/src/resources/system.ts +0 -13
  274. package/src/resources/tools/index.ts +0 -4
  275. package/src/resources/tools/result.ts +0 -16
  276. package/src/resources/tools/tools.ts +0 -54
  277. package/src/resources/workflows.ts +0 -49
  278. package/src/shims/node.ts +0 -50
  279. package/src/shims/web.ts +0 -50
  280. package/src/tsconfig.json +0 -11
  281. package/src/uploads.ts +0 -255
  282. package/src/version.ts +0 -1
  283. package/uploads.d.ts +0 -75
  284. package/uploads.d.ts.map +0 -1
  285. package/uploads.js +0 -171
  286. package/uploads.js.map +0 -1
  287. package/uploads.mjs +0 -158
  288. package/uploads.mjs.map +0 -1
  289. package/version.d.ts +0 -2
  290. package/version.d.ts.map +0 -1
  291. package/version.js +0 -5
  292. package/version.js.map +0 -1
  293. package/version.mjs +0 -2
  294. package/version.mjs.map +0 -1
package/src/example.ts ADDED
@@ -0,0 +1,47 @@
1
+ import { MastraClient } from "./client";
2
+
3
+
4
+ (async () => {
5
+ const client = new MastraClient({
6
+ baseUrl: 'http://localhost:4111',
7
+ });
8
+
9
+ try {
10
+ const agent = client.getAgent('weatherAgent');
11
+ const response = await agent.stream({
12
+ messages: [{
13
+ role: 'user',
14
+ content: 'Hello, world!',
15
+ id: '1',
16
+ createdAt: new Date(),
17
+ threadId: '1',
18
+ type: 'text',
19
+ }]
20
+ })
21
+
22
+
23
+ const reader = response?.body?.getReader()
24
+ const decoder = new TextDecoder()
25
+ let buffer = ''
26
+
27
+ while (true) {
28
+ if (!reader) break;
29
+ const { value, done } = await reader.read()
30
+ if (done) break;
31
+
32
+ const chunk = decoder.decode(value);
33
+ buffer += chunk;
34
+
35
+ console.log(buffer);
36
+
37
+ const matches = buffer.matchAll(/0:"([^"]*)"/g);
38
+
39
+ for (const match of matches) {
40
+ const content = match[1];
41
+ process.stdout.write(`${content}\n`);
42
+ }
43
+ }
44
+ } catch (error) {
45
+ console.error(error);
46
+ }
47
+ })()
@@ -0,0 +1,585 @@
1
+ import { describe, expect, beforeEach, it, vi } from 'vitest';
2
+ import { MastraClient } from "./client";
3
+ import type { MessageType } from '@mastra/core';
4
+
5
+ // Mock fetch globally
6
+ global.fetch = vi.fn();
7
+
8
+ describe('MastraClient Resources', () => {
9
+ let client: MastraClient;
10
+ const clientOptions = {
11
+ baseUrl: 'http://localhost:4111',
12
+ headers: {
13
+ 'Authorization': 'Bearer test-key'
14
+ }
15
+ };
16
+
17
+ // Helper to mock successful API responses
18
+ const mockFetchResponse = (data: any, options: { isStream?: boolean } = {}) => {
19
+ if (options.isStream) {
20
+ const stream = new ReadableStream({
21
+ start(controller) {
22
+ controller.enqueue(new TextEncoder().encode(JSON.stringify(data)));
23
+ controller.close();
24
+ }
25
+ });
26
+ (global.fetch as any).mockResolvedValueOnce({
27
+ ok: true,
28
+ headers: {
29
+ get: (name: string) => name === 'Content-Type' ? 'text/event-stream' : null
30
+ },
31
+ body: stream
32
+ });
33
+ } else {
34
+ (global.fetch as any).mockResolvedValueOnce({
35
+ ok: true,
36
+ headers: {
37
+ get: (name: string) => name === 'Content-Type' ? 'application/json' : null
38
+ },
39
+ json: async () => data
40
+ });
41
+ }
42
+ };
43
+
44
+ beforeEach(() => {
45
+ vi.clearAllMocks();
46
+ client = new MastraClient(clientOptions);
47
+ });
48
+
49
+ describe('Vector Resource', () => {
50
+ const vectorName = 'test-vector';
51
+ let vector: ReturnType<typeof client.getVector>;
52
+
53
+ beforeEach(() => {
54
+ vector = client.getVector(vectorName);
55
+ });
56
+
57
+ it('should get vector index details', async () => {
58
+ const mockResponse = {
59
+ dimension: 128,
60
+ metric: 'cosine',
61
+ count: 1000
62
+ };
63
+ mockFetchResponse(mockResponse);
64
+
65
+ const result = await vector.details('test-index');
66
+ expect(result).toEqual(mockResponse);
67
+ expect(global.fetch).toHaveBeenCalledWith(
68
+ `${clientOptions.baseUrl}/api/vector/test-vector/indexes/test-index`,
69
+ expect.objectContaining({
70
+ headers: expect.objectContaining(clientOptions.headers)
71
+ })
72
+ );
73
+ });
74
+
75
+ it('should delete vector index', async () => {
76
+ mockFetchResponse({ success: true });
77
+ const result = await vector.delete('test-index');
78
+ expect(result).toEqual({ success: true });
79
+ expect(global.fetch).toHaveBeenCalledWith(
80
+ `${clientOptions.baseUrl}/api/vector/test-vector/indexes/test-index`,
81
+ expect.objectContaining({
82
+ method: 'DELETE',
83
+ headers: expect.objectContaining(clientOptions.headers)
84
+ })
85
+ );
86
+ });
87
+
88
+ it('should get all indexes', async () => {
89
+ const mockResponse = { indexes: ['index1', 'index2'] };
90
+ mockFetchResponse(mockResponse);
91
+ const result = await vector.getIndexes();
92
+ expect(result).toEqual(mockResponse);
93
+ expect(global.fetch).toHaveBeenCalledWith(
94
+ `${clientOptions.baseUrl}/api/vector/test-vector/indexes`,
95
+ expect.objectContaining({
96
+ headers: expect.objectContaining(clientOptions.headers)
97
+ })
98
+ );
99
+ });
100
+
101
+ it('should create vector index with all parameters', async () => {
102
+ mockFetchResponse({ success: true });
103
+ const result = await vector.createIndex({
104
+ indexName: 'test-index',
105
+ dimension: 128,
106
+ metric: 'cosine'
107
+ });
108
+ expect(result).toEqual({ success: true });
109
+ expect(global.fetch).toHaveBeenCalledWith(
110
+ `${clientOptions.baseUrl}/api/vector/test-vector/create-index`,
111
+ expect.objectContaining({
112
+ method: 'POST',
113
+ headers: expect.objectContaining(clientOptions.headers),
114
+ body: JSON.stringify({
115
+ indexName: 'test-index',
116
+ dimension: 128,
117
+ metric: 'cosine'
118
+ })
119
+ })
120
+ );
121
+ });
122
+
123
+ it('should upsert vectors with metadata and ids', async () => {
124
+ const mockResponse = ['id1', 'id2'];
125
+ mockFetchResponse(mockResponse);
126
+ const result = await vector.upsert({
127
+ indexName: 'test-index',
128
+ vectors: [[1, 2], [3, 4]],
129
+ metadata: [{ label: 'a' }, { label: 'b' }],
130
+ ids: ['id1', 'id2']
131
+ });
132
+ expect(result).toEqual(mockResponse);
133
+ expect(global.fetch).toHaveBeenCalledWith(
134
+ `${clientOptions.baseUrl}/api/vector/test-vector/upsert`,
135
+ expect.objectContaining({
136
+ method: 'POST',
137
+ headers: expect.objectContaining(clientOptions.headers),
138
+ body: JSON.stringify({
139
+ indexName: 'test-index',
140
+ vectors: [[1, 2], [3, 4]],
141
+ metadata: [{ label: 'a' }, { label: 'b' }],
142
+ ids: ['id1', 'id2']
143
+ })
144
+ })
145
+ );
146
+ });
147
+
148
+ it('should query vectors with all parameters', async () => {
149
+ const mockResponse = {
150
+ results: [{
151
+ id: 'id1',
152
+ score: 0.9,
153
+ metadata: { label: 'a' },
154
+ vector: [1, 2]
155
+ }]
156
+ };
157
+ mockFetchResponse(mockResponse);
158
+ const result = await vector.query({
159
+ indexName: 'test-index',
160
+ queryVector: [1, 2],
161
+ topK: 10,
162
+ filter: { label: 'a' },
163
+ includeVector: true
164
+ });
165
+ expect(result).toEqual(mockResponse);
166
+ expect(global.fetch).toHaveBeenCalledWith(
167
+ `${clientOptions.baseUrl}/api/vector/test-vector/query`,
168
+ expect.objectContaining({
169
+ method: 'POST',
170
+ headers: expect.objectContaining(clientOptions.headers),
171
+ body: JSON.stringify({
172
+ indexName: 'test-index',
173
+ queryVector: [1, 2],
174
+ topK: 10,
175
+ filter: { label: 'a' },
176
+ includeVector: true
177
+ })
178
+ })
179
+ );
180
+ });
181
+ });
182
+
183
+ describe('Agent Resource', () => {
184
+ const agentId = 'test-agent';
185
+ let agent: ReturnType<typeof client.getAgent>;
186
+
187
+ beforeEach(() => {
188
+ agent = client.getAgent(agentId);
189
+ });
190
+
191
+ it('should get all agents', async () => {
192
+ const mockResponse = {
193
+ 'agent1': { name: 'Agent 1', model: 'gpt-4' },
194
+ 'agent2': { name: 'Agent 2', model: 'gpt-3.5' }
195
+ };
196
+ mockFetchResponse(mockResponse);
197
+ const result = await client.getAgents();
198
+ expect(result).toEqual(mockResponse);
199
+ expect(global.fetch).toHaveBeenCalledWith(
200
+ `${clientOptions.baseUrl}/api/agents`,
201
+ expect.objectContaining({
202
+ headers: expect.objectContaining(clientOptions.headers)
203
+ })
204
+ );
205
+ });
206
+
207
+ it('should get agent details', async () => {
208
+ const mockResponse = {
209
+ name: 'Test Agent',
210
+ model: 'gpt-4',
211
+ instructions: 'Test instructions',
212
+ tools: {}
213
+ };
214
+ mockFetchResponse(mockResponse);
215
+
216
+ const result = await agent.details();
217
+ expect(result).toEqual(mockResponse);
218
+ expect(global.fetch).toHaveBeenCalledWith(
219
+ `${clientOptions.baseUrl}/api/agents/test-agent`,
220
+ expect.objectContaining({
221
+ headers: expect.objectContaining(clientOptions.headers)
222
+ })
223
+ );
224
+ });
225
+
226
+ it('should generate response', async () => {
227
+ const mockResponse = {
228
+ response: 'Generated response'
229
+ };
230
+ mockFetchResponse(mockResponse);
231
+
232
+ const result = await agent.generate({
233
+ messages: [],
234
+ threadId: 'test-thread',
235
+ resourceid: 'test-resource',
236
+ output: {}
237
+ });
238
+ expect(result).toEqual(mockResponse);
239
+ expect(global.fetch).toHaveBeenCalledWith(
240
+ `${clientOptions.baseUrl}/api/agents/test-agent/generate`,
241
+ expect.objectContaining({
242
+ method: 'POST',
243
+ headers: expect.objectContaining(clientOptions.headers),
244
+ body: JSON.stringify({
245
+ messages: [],
246
+ threadId: 'test-thread',
247
+ resourceid: 'test-resource',
248
+ output: {}
249
+ })
250
+ })
251
+ );
252
+ });
253
+
254
+ it('should stream responses', async () => {
255
+ const mockChunk = { content: 'test response' };
256
+ mockFetchResponse(mockChunk, { isStream: true });
257
+
258
+ const response = await agent.stream({
259
+ messages: [{
260
+ role: 'user',
261
+ content: 'test',
262
+ id: '1',
263
+ createdAt: new Date(),
264
+ threadId: '1',
265
+ type: 'text'
266
+ }]
267
+ });
268
+
269
+ expect(response.body).toBeInstanceOf(ReadableStream);
270
+ const reader = response?.body?.getReader();
271
+ expect(reader).toBeDefined();
272
+
273
+ if (reader) {
274
+ const { value, done } = await reader.read();
275
+ expect(done).toBe(false);
276
+ expect(new TextDecoder().decode(value)).toBe(JSON.stringify(mockChunk));
277
+ }
278
+ });
279
+
280
+ it('should get agent tool', async () => {
281
+ const mockResponse = {
282
+ id: 'tool1',
283
+ description: 'Test Tool'
284
+ };
285
+ mockFetchResponse(mockResponse);
286
+ const result = await agent.getTool('tool1');
287
+ expect(result).toEqual(mockResponse);
288
+ expect(global.fetch).toHaveBeenCalledWith(
289
+ `${clientOptions.baseUrl}/api/agents/test-agent/tools/tool1`,
290
+ expect.objectContaining({
291
+ headers: expect.objectContaining(clientOptions.headers)
292
+ })
293
+ );
294
+ });
295
+
296
+ it('should get agent evals', async () => {
297
+ const mockResponse = {
298
+ name: 'Test Agent',
299
+ evals: [{ id: 'eval1' }]
300
+ };
301
+ mockFetchResponse(mockResponse);
302
+ const result = await agent.evals();
303
+ expect(result).toEqual(mockResponse);
304
+ expect(global.fetch).toHaveBeenCalledWith(
305
+ `${clientOptions.baseUrl}/api/agents/test-agent/evals`,
306
+ expect.objectContaining({
307
+ headers: expect.objectContaining(clientOptions.headers)
308
+ })
309
+ );
310
+ });
311
+
312
+ it('should get live evals', async () => {
313
+ const mockResponse = {
314
+ name: 'Test Agent',
315
+ evals: [{ id: 'eval1', live: true }]
316
+ };
317
+ mockFetchResponse(mockResponse);
318
+ const result = await agent.liveEvals();
319
+ expect(result).toEqual(mockResponse);
320
+ expect(global.fetch).toHaveBeenCalledWith(
321
+ `${clientOptions.baseUrl}/api/agents/test-agent/evals/live`,
322
+ expect.objectContaining({
323
+ headers: expect.objectContaining(clientOptions.headers)
324
+ })
325
+ );
326
+ });
327
+ });
328
+
329
+ describe('Memory Thread Resource', () => {
330
+ const threadId = 'test-thread';
331
+ let memoryThread: ReturnType<typeof client.getMemoryThread>;
332
+
333
+ beforeEach(() => {
334
+ memoryThread = client.getMemoryThread(threadId);
335
+ });
336
+
337
+ it('should get thread details', async () => {
338
+ const mockResponse = {
339
+ id: threadId,
340
+ title: 'Test Thread',
341
+ metadata: {}
342
+ };
343
+ mockFetchResponse(mockResponse);
344
+
345
+ const result = await memoryThread.get();
346
+ expect(result).toEqual(mockResponse);
347
+ expect(global.fetch).toHaveBeenCalledWith(
348
+ `${clientOptions.baseUrl}/api/memory/threads/test-thread`,
349
+ expect.objectContaining({
350
+ headers: expect.objectContaining(clientOptions.headers)
351
+ })
352
+ );
353
+ });
354
+
355
+ it('should update thread', async () => {
356
+ const mockResponse = {
357
+ id: threadId,
358
+ title: 'Updated Thread',
359
+ metadata: { updated: true }
360
+ };
361
+ mockFetchResponse(mockResponse);
362
+
363
+ const result = await memoryThread.update({
364
+ title: 'Updated Thread',
365
+ metadata: { updated: true },
366
+ resourceid: 'test-resource'
367
+ });
368
+ expect(result).toEqual(mockResponse);
369
+ expect(global.fetch).toHaveBeenCalledWith(
370
+ `${clientOptions.baseUrl}/api/memory/threads/test-thread`,
371
+ expect.objectContaining({
372
+ method: 'PATCH',
373
+ headers: expect.objectContaining(clientOptions.headers)
374
+ })
375
+ );
376
+ });
377
+
378
+ it('should delete thread', async () => {
379
+ const mockResponse = { result: 'deleted' };
380
+ mockFetchResponse(mockResponse);
381
+ const result = await memoryThread.delete();
382
+ expect(result).toEqual(mockResponse);
383
+ expect(global.fetch).toHaveBeenCalledWith(
384
+ `${clientOptions.baseUrl}/api/memory/threads/test-thread`,
385
+ expect.objectContaining({
386
+ method: 'DELETE',
387
+ headers: expect.objectContaining(clientOptions.headers)
388
+ })
389
+ );
390
+ });
391
+
392
+ it('should get memory status', async () => {
393
+ const mockResponse = { result: true };
394
+ mockFetchResponse(mockResponse);
395
+ const result = await client.getMemoryStatus();
396
+ expect(result).toEqual(mockResponse);
397
+ expect(global.fetch).toHaveBeenCalledWith(
398
+ `${clientOptions.baseUrl}/api/memory/status`,
399
+ expect.objectContaining({
400
+ headers: expect.objectContaining(clientOptions.headers)
401
+ })
402
+ );
403
+ });
404
+
405
+ it('should save messages to memory', async () => {
406
+ const messages: MessageType[] = [{
407
+ id: '1',
408
+ type: 'text',
409
+ content: 'test',
410
+ role: 'user',
411
+ threadId: 'test-thread',
412
+ createdAt: new Date()
413
+ }];
414
+ mockFetchResponse(messages);
415
+ const result = await client.saveMessageToMemory({ messages });
416
+ expect(result).toEqual(messages);
417
+ expect(global.fetch).toHaveBeenCalledWith(
418
+ `${clientOptions.baseUrl}/api/memory/save-messages`,
419
+ expect.objectContaining({
420
+ method: 'POST',
421
+ headers: expect.objectContaining(clientOptions.headers),
422
+ body: JSON.stringify({ messages })
423
+ })
424
+ );
425
+ });
426
+ });
427
+
428
+ describe('Tool Resource', () => {
429
+ const toolId = 'test-tool';
430
+ let tool: ReturnType<typeof client.getTool>;
431
+
432
+ beforeEach(() => {
433
+ tool = client.getTool(toolId);
434
+ });
435
+
436
+ it('should get tool details', async () => {
437
+ const mockResponse = {
438
+ id: toolId,
439
+ description: 'Test Tool',
440
+ inputSchema: '{}',
441
+ outputSchema: '{}'
442
+ };
443
+ mockFetchResponse(mockResponse);
444
+
445
+ const result = await tool.details();
446
+ expect(result).toEqual(mockResponse);
447
+ expect(global.fetch).toHaveBeenCalledWith(
448
+ `${clientOptions.baseUrl}/api/tools/test-tool`,
449
+ expect.objectContaining({
450
+ headers: expect.objectContaining(clientOptions.headers)
451
+ })
452
+ );
453
+ });
454
+
455
+ it('should execute tool', async () => {
456
+ const mockResponse = {
457
+ result: 'Tool execution result'
458
+ };
459
+ mockFetchResponse(mockResponse);
460
+
461
+ const result = await tool.execute({ input: 'test' });
462
+ expect(result).toEqual(mockResponse);
463
+ expect(global.fetch).toHaveBeenCalledWith(
464
+ `${clientOptions.baseUrl}/api/tools/test-tool/execute`,
465
+ expect.objectContaining({
466
+ method: 'POST',
467
+ headers: expect.objectContaining(clientOptions.headers),
468
+ body: JSON.stringify({ input: 'test' })
469
+ })
470
+ );
471
+ });
472
+ });
473
+
474
+ describe('Workflow Resource', () => {
475
+ const workflowId = 'test-workflow';
476
+ let workflow: ReturnType<typeof client.getWorkflow>;
477
+
478
+ beforeEach(() => {
479
+ workflow = client.getWorkflow(workflowId);
480
+ });
481
+
482
+ it('should get workflow details', async () => {
483
+ const mockResponse = {
484
+ name: 'Test Workflow',
485
+ triggerSchema: '{}',
486
+ steps: {},
487
+ stepGraph: {},
488
+ stepSubscriberGraph: {}
489
+ };
490
+ mockFetchResponse(mockResponse);
491
+
492
+ const result = await workflow.details();
493
+ expect(result).toEqual(mockResponse);
494
+ expect(global.fetch).toHaveBeenCalledWith(
495
+ `${clientOptions.baseUrl}/api/workflows/test-workflow`,
496
+ expect.objectContaining({
497
+ headers: expect.objectContaining(clientOptions.headers)
498
+ })
499
+ );
500
+ });
501
+
502
+ it('should execute workflow', async () => {
503
+ const mockResponse = {
504
+ result: 'Workflow execution result'
505
+ };
506
+ mockFetchResponse(mockResponse);
507
+
508
+ const result = await workflow.execute({ trigger: 'test' });
509
+ expect(result).toEqual(mockResponse);
510
+ expect(global.fetch).toHaveBeenCalledWith(
511
+ `${clientOptions.baseUrl}/api/workflows/test-workflow/execute`,
512
+ expect.objectContaining({
513
+ method: 'POST',
514
+ headers: expect.objectContaining(clientOptions.headers),
515
+ body: JSON.stringify({ trigger: 'test' })
516
+ })
517
+ );
518
+ });
519
+ });
520
+
521
+ describe('Client Error Handling', () => {
522
+ it('should retry failed requests', async () => {
523
+ // Mock first two calls to fail, third to succeed
524
+ (global.fetch as any)
525
+ .mockRejectedValueOnce(new Error('Network error'))
526
+ .mockRejectedValueOnce(new Error('Network error'))
527
+ .mockResolvedValueOnce({
528
+ ok: true,
529
+ headers: {
530
+ get: () => 'application/json'
531
+ },
532
+ json: async () => ({ success: true })
533
+ });
534
+
535
+ const result = await client.request('/test-endpoint');
536
+ expect(result).toEqual({ success: true });
537
+ expect(global.fetch).toHaveBeenCalledTimes(3);
538
+ });
539
+
540
+ it('should throw error after max retries', async () => {
541
+ (global.fetch as any).mockRejectedValue(new Error('Network error'));
542
+
543
+ await expect(client.request('/test-endpoint'))
544
+ .rejects
545
+ .toThrow('Network error');
546
+
547
+ expect(global.fetch).toHaveBeenCalledTimes(4);
548
+ });
549
+ });
550
+
551
+ describe('Client Configuration', () => {
552
+ it('should handle custom retry configuration', async () => {
553
+ const customClient = new MastraClient({
554
+ baseUrl: 'http://localhost:4111',
555
+ retries: 2,
556
+ backoffMs: 100,
557
+ maxBackoffMs: 1000,
558
+ headers: { 'Custom-Header': 'value' }
559
+ });
560
+
561
+ (global.fetch as any)
562
+ .mockRejectedValueOnce(new Error('Network error'))
563
+ .mockRejectedValueOnce(new Error('Network error'))
564
+ .mockResolvedValueOnce({
565
+ ok: true,
566
+ headers: {
567
+ get: () => 'application/json'
568
+ },
569
+ json: async () => ({ success: true })
570
+ });
571
+
572
+ const result = await customClient.request('/test');
573
+ expect(result).toEqual({ success: true });
574
+ expect(global.fetch).toHaveBeenCalledTimes(3);
575
+ expect(global.fetch).toHaveBeenCalledWith(
576
+ 'http://localhost:4111/test',
577
+ expect.objectContaining({
578
+ headers: expect.objectContaining({
579
+ 'Custom-Header': 'value'
580
+ })
581
+ })
582
+ );
583
+ });
584
+ });
585
+ });