digital-tools 2.1.1 → 2.3.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.
Files changed (293) hide show
  1. package/CHANGELOG.md +17 -0
  2. package/README.md +2 -0
  3. package/dist/client.d.ts +109 -0
  4. package/dist/client.d.ts.map +1 -0
  5. package/dist/client.js +69 -0
  6. package/dist/client.js.map +1 -0
  7. package/dist/define.d.ts +2 -2
  8. package/dist/define.d.ts.map +1 -1
  9. package/dist/define.js +22 -20
  10. package/dist/define.js.map +1 -1
  11. package/dist/function-ref.d.ts +229 -0
  12. package/dist/function-ref.d.ts.map +1 -0
  13. package/dist/function-ref.js +28 -0
  14. package/dist/function-ref.js.map +1 -0
  15. package/dist/function-sugar.d.ts +57 -0
  16. package/dist/function-sugar.d.ts.map +1 -0
  17. package/dist/function-sugar.js +79 -0
  18. package/dist/function-sugar.js.map +1 -0
  19. package/dist/index.d.ts +10 -3
  20. package/dist/index.d.ts.map +1 -1
  21. package/dist/index.js +24 -4
  22. package/dist/index.js.map +1 -1
  23. package/dist/providers/analytics/mixpanel.d.ts.map +1 -1
  24. package/dist/providers/analytics/mixpanel.js +21 -18
  25. package/dist/providers/analytics/mixpanel.js.map +1 -1
  26. package/dist/providers/calendar/cal-com.d.ts.map +1 -1
  27. package/dist/providers/calendar/cal-com.js +10 -10
  28. package/dist/providers/calendar/cal-com.js.map +1 -1
  29. package/dist/providers/calendar/google-calendar.d.ts.map +1 -1
  30. package/dist/providers/calendar/google-calendar.js +4 -4
  31. package/dist/providers/calendar/google-calendar.js.map +1 -1
  32. package/dist/providers/crm/hubspot.d.ts.map +1 -1
  33. package/dist/providers/crm/hubspot.js +107 -85
  34. package/dist/providers/crm/hubspot.js.map +1 -1
  35. package/dist/providers/development/github.d.ts.map +1 -1
  36. package/dist/providers/development/github.js +40 -43
  37. package/dist/providers/development/github.js.map +1 -1
  38. package/dist/providers/ecommerce/shopify.d.ts.map +1 -1
  39. package/dist/providers/ecommerce/shopify.js +79 -62
  40. package/dist/providers/ecommerce/shopify.js.map +1 -1
  41. package/dist/providers/email/resend.d.ts.map +1 -1
  42. package/dist/providers/email/resend.js +20 -16
  43. package/dist/providers/email/resend.js.map +1 -1
  44. package/dist/providers/email/sendgrid.d.ts.map +1 -1
  45. package/dist/providers/email/sendgrid.js +12 -9
  46. package/dist/providers/email/sendgrid.js.map +1 -1
  47. package/dist/providers/finance/stripe.d.ts.map +1 -1
  48. package/dist/providers/finance/stripe.js +44 -42
  49. package/dist/providers/finance/stripe.js.map +1 -1
  50. package/dist/providers/forms/typeform.d.ts.map +1 -1
  51. package/dist/providers/forms/typeform.js +68 -58
  52. package/dist/providers/forms/typeform.js.map +1 -1
  53. package/dist/providers/knowledge/notion.d.ts.map +1 -1
  54. package/dist/providers/knowledge/notion.js +75 -41
  55. package/dist/providers/knowledge/notion.js.map +1 -1
  56. package/dist/providers/marketing/mailchimp.d.ts.map +1 -1
  57. package/dist/providers/marketing/mailchimp.js +74 -61
  58. package/dist/providers/marketing/mailchimp.js.map +1 -1
  59. package/dist/providers/media/cloudinary.d.ts.map +1 -1
  60. package/dist/providers/media/cloudinary.js +30 -28
  61. package/dist/providers/media/cloudinary.js.map +1 -1
  62. package/dist/providers/messaging/slack.d.ts.map +1 -1
  63. package/dist/providers/messaging/slack.js +75 -58
  64. package/dist/providers/messaging/slack.js.map +1 -1
  65. package/dist/providers/messaging/twilio-sms.d.ts.map +1 -1
  66. package/dist/providers/messaging/twilio-sms.js +33 -15
  67. package/dist/providers/messaging/twilio-sms.js.map +1 -1
  68. package/dist/providers/project-management/linear.d.ts.map +1 -1
  69. package/dist/providers/project-management/linear.js +31 -27
  70. package/dist/providers/project-management/linear.js.map +1 -1
  71. package/dist/providers/spreadsheet/google-sheets.d.ts.map +1 -1
  72. package/dist/providers/spreadsheet/google-sheets.js +21 -18
  73. package/dist/providers/spreadsheet/google-sheets.js.map +1 -1
  74. package/dist/providers/spreadsheet/xlsx.d.ts.map +1 -1
  75. package/dist/providers/spreadsheet/xlsx.js +4 -4
  76. package/dist/providers/spreadsheet/xlsx.js.map +1 -1
  77. package/dist/providers/storage/index.js +1 -0
  78. package/dist/providers/storage/index.js.map +1 -1
  79. package/dist/providers/storage/s3.d.ts.map +1 -1
  80. package/dist/providers/storage/s3.js +36 -27
  81. package/dist/providers/storage/s3.js.map +1 -1
  82. package/dist/providers/support/zendesk.d.ts.map +1 -1
  83. package/dist/providers/support/zendesk.js +24 -25
  84. package/dist/providers/support/zendesk.js.map +1 -1
  85. package/dist/providers/tasks/todoist.d.ts.map +1 -1
  86. package/dist/providers/tasks/todoist.js +18 -18
  87. package/dist/providers/tasks/todoist.js.map +1 -1
  88. package/dist/providers/video-conferencing/google-meet.d.ts.map +1 -1
  89. package/dist/providers/video-conferencing/google-meet.js +11 -11
  90. package/dist/providers/video-conferencing/google-meet.js.map +1 -1
  91. package/dist/providers/video-conferencing/jitsi.js +14 -14
  92. package/dist/providers/video-conferencing/jitsi.js.map +1 -1
  93. package/dist/providers/video-conferencing/teams.d.ts.map +1 -1
  94. package/dist/providers/video-conferencing/teams.js +9 -7
  95. package/dist/providers/video-conferencing/teams.js.map +1 -1
  96. package/dist/providers/video-conferencing/zoom.d.ts.map +1 -1
  97. package/dist/providers/video-conferencing/zoom.js +26 -24
  98. package/dist/providers/video-conferencing/zoom.js.map +1 -1
  99. package/dist/tools/data.d.ts.map +1 -1
  100. package/dist/tools/data.js +5 -12
  101. package/dist/tools/data.js.map +1 -1
  102. package/dist/tools/index.d.ts +1 -0
  103. package/dist/tools/index.d.ts.map +1 -1
  104. package/dist/tools/index.js +1 -0
  105. package/dist/tools/index.js.map +1 -1
  106. package/dist/tools/system.d.ts +289 -0
  107. package/dist/tools/system.d.ts.map +1 -0
  108. package/dist/tools/system.js +752 -0
  109. package/dist/tools/system.js.map +1 -0
  110. package/dist/tools/web.d.ts.map +1 -1
  111. package/dist/tools/web.js +22 -10
  112. package/dist/tools/web.js.map +1 -1
  113. package/dist/track-record.d.ts +101 -0
  114. package/dist/track-record.d.ts.map +1 -0
  115. package/dist/track-record.js +17 -0
  116. package/dist/track-record.js.map +1 -0
  117. package/dist/types.d.ts +210 -9
  118. package/dist/types.d.ts.map +1 -1
  119. package/dist/verb-registration.d.ts +122 -0
  120. package/dist/verb-registration.d.ts.map +1 -0
  121. package/dist/verb-registration.js +176 -0
  122. package/dist/verb-registration.js.map +1 -0
  123. package/dist/worker.d.ts +93 -0
  124. package/dist/worker.d.ts.map +1 -0
  125. package/dist/worker.js +315 -0
  126. package/dist/worker.js.map +1 -0
  127. package/dist/wrap.d.ts +89 -0
  128. package/dist/wrap.d.ts.map +1 -0
  129. package/dist/wrap.js +225 -0
  130. package/dist/wrap.js.map +1 -0
  131. package/package.json +21 -4
  132. package/src/client.ts +136 -0
  133. package/src/define.ts +31 -37
  134. package/src/function-ref.ts +264 -0
  135. package/src/function-sugar.ts +134 -0
  136. package/src/index.ts +132 -10
  137. package/src/providers/analytics/mixpanel.ts +19 -18
  138. package/src/providers/calendar/cal-com.ts +29 -18
  139. package/src/providers/calendar/google-calendar.ts +20 -14
  140. package/src/providers/crm/hubspot.ts +225 -99
  141. package/src/providers/development/github.ts +206 -135
  142. package/src/providers/ecommerce/shopify.ts +250 -89
  143. package/src/providers/email/resend.ts +101 -28
  144. package/src/providers/email/sendgrid.ts +12 -9
  145. package/src/providers/finance/stripe.ts +128 -49
  146. package/src/providers/forms/typeform.ts +74 -58
  147. package/src/providers/knowledge/notion.ts +340 -88
  148. package/src/providers/marketing/mailchimp.ts +86 -70
  149. package/src/providers/media/cloudinary.ts +99 -41
  150. package/src/providers/messaging/slack.ts +283 -85
  151. package/src/providers/messaging/twilio-sms.ts +35 -15
  152. package/src/providers/project-management/linear.ts +143 -55
  153. package/src/providers/spreadsheet/google-sheets.ts +222 -56
  154. package/src/providers/spreadsheet/xlsx.ts +47 -16
  155. package/src/providers/storage/s3.ts +119 -47
  156. package/src/providers/support/zendesk.ts +196 -46
  157. package/src/providers/tasks/todoist.ts +20 -26
  158. package/src/providers/video-conferencing/google-meet.ts +17 -20
  159. package/src/providers/video-conferencing/jitsi.ts +14 -14
  160. package/src/providers/video-conferencing/teams.ts +14 -13
  161. package/src/providers/video-conferencing/zoom.ts +54 -49
  162. package/src/tools/data.ts +6 -16
  163. package/src/tools/index.ts +1 -0
  164. package/src/tools/system.ts +887 -0
  165. package/src/tools/web.ts +22 -10
  166. package/src/track-record.ts +106 -0
  167. package/src/types.ts +241 -13
  168. package/src/verb-registration.ts +197 -0
  169. package/src/worker.ts +370 -0
  170. package/src/wrap.ts +260 -0
  171. package/test/client.test.ts +146 -0
  172. package/test/communication-tools-extended.test.ts +734 -0
  173. package/test/data-tools-extended.test.ts +743 -0
  174. package/test/define-extended.test.ts +819 -0
  175. package/test/define.test.ts +150 -41
  176. package/test/entities.test.ts +623 -0
  177. package/test/extended-entities.test.ts +1228 -0
  178. package/test/provider-implementations.test.ts +725 -0
  179. package/test/provider-registry-extended.test.ts +583 -0
  180. package/test/providers/google-sheets.test.ts +851 -0
  181. package/test/providers/helpers.ts +554 -0
  182. package/test/providers/hubspot.test.ts +576 -0
  183. package/test/providers/slack.test.ts +932 -0
  184. package/test/providers/stripe.test.ts +701 -0
  185. package/test/providers.test.ts +578 -0
  186. package/test/system-tools-extended.test.ts +632 -0
  187. package/test/system.test.ts +673 -0
  188. package/test/tools.test.ts +15 -11
  189. package/test/types.test.ts +402 -0
  190. package/test/verb-registration.test.ts +395 -0
  191. package/test/web-tools.test.ts +553 -0
  192. package/test/worker-extended.test.ts +699 -0
  193. package/test/worker.test.ts +576 -0
  194. package/test/wrap.test.ts +366 -0
  195. package/tsconfig.json +3 -13
  196. package/vitest.config.ts +37 -0
  197. package/wrangler.jsonc +9 -0
  198. package/.turbo/turbo-build.log +0 -5
  199. package/dist/providers/voice/vapi.d.ts +0 -27
  200. package/dist/providers/voice/vapi.d.ts.map +0 -1
  201. package/dist/providers/voice/vapi.js +0 -440
  202. package/dist/providers/voice/vapi.js.map +0 -1
  203. package/src/define.js +0 -267
  204. package/src/entities/advertising.js +0 -999
  205. package/src/entities/ai.js +0 -756
  206. package/src/entities/analytics.js +0 -1588
  207. package/src/entities/automation.js +0 -601
  208. package/src/entities/communication.js +0 -1150
  209. package/src/entities/crm.js +0 -1386
  210. package/src/entities/design.js +0 -546
  211. package/src/entities/development.js +0 -2212
  212. package/src/entities/document.js +0 -874
  213. package/src/entities/ecommerce.js +0 -1429
  214. package/src/entities/experiment.js +0 -1039
  215. package/src/entities/finance.js +0 -3478
  216. package/src/entities/forms.js +0 -1892
  217. package/src/entities/hr.js +0 -661
  218. package/src/entities/identity.js +0 -997
  219. package/src/entities/index.js +0 -282
  220. package/src/entities/infrastructure.js +0 -1153
  221. package/src/entities/knowledge.js +0 -1438
  222. package/src/entities/marketing.js +0 -1610
  223. package/src/entities/media.js +0 -1634
  224. package/src/entities/notification.js +0 -1199
  225. package/src/entities/presentation.js +0 -1274
  226. package/src/entities/productivity.js +0 -1317
  227. package/src/entities/project-management.js +0 -1136
  228. package/src/entities/recruiting.js +0 -736
  229. package/src/entities/shipping.js +0 -509
  230. package/src/entities/signature.js +0 -1102
  231. package/src/entities/site.js +0 -222
  232. package/src/entities/spreadsheet.js +0 -1341
  233. package/src/entities/storage.js +0 -1198
  234. package/src/entities/support.js +0 -1166
  235. package/src/entities/video-conferencing.js +0 -1750
  236. package/src/entities/video.js +0 -950
  237. package/src/entities.js +0 -1663
  238. package/src/index.js +0 -74
  239. package/src/providers/analytics/index.js +0 -17
  240. package/src/providers/analytics/mixpanel.js +0 -255
  241. package/src/providers/calendar/cal-com.js +0 -303
  242. package/src/providers/calendar/google-calendar.js +0 -335
  243. package/src/providers/calendar/index.js +0 -20
  244. package/src/providers/crm/hubspot.js +0 -566
  245. package/src/providers/crm/index.js +0 -17
  246. package/src/providers/development/github.js +0 -472
  247. package/src/providers/development/index.js +0 -17
  248. package/src/providers/ecommerce/index.js +0 -17
  249. package/src/providers/ecommerce/shopify.js +0 -378
  250. package/src/providers/email/index.js +0 -20
  251. package/src/providers/email/resend.js +0 -258
  252. package/src/providers/email/sendgrid.js +0 -161
  253. package/src/providers/finance/index.js +0 -17
  254. package/src/providers/finance/stripe.js +0 -549
  255. package/src/providers/forms/index.js +0 -17
  256. package/src/providers/forms/typeform.js +0 -500
  257. package/src/providers/index.js +0 -123
  258. package/src/providers/knowledge/index.js +0 -17
  259. package/src/providers/knowledge/notion.js +0 -389
  260. package/src/providers/marketing/index.js +0 -17
  261. package/src/providers/marketing/mailchimp.js +0 -443
  262. package/src/providers/media/cloudinary.js +0 -318
  263. package/src/providers/media/index.js +0 -17
  264. package/src/providers/messaging/index.js +0 -20
  265. package/src/providers/messaging/slack.js +0 -393
  266. package/src/providers/messaging/twilio-sms.js +0 -249
  267. package/src/providers/project-management/index.js +0 -17
  268. package/src/providers/project-management/linear.js +0 -575
  269. package/src/providers/registry.js +0 -86
  270. package/src/providers/spreadsheet/google-sheets.js +0 -375
  271. package/src/providers/spreadsheet/index.js +0 -20
  272. package/src/providers/spreadsheet/xlsx.js +0 -423
  273. package/src/providers/storage/index.js +0 -24
  274. package/src/providers/storage/s3.js +0 -419
  275. package/src/providers/support/index.js +0 -17
  276. package/src/providers/support/zendesk.js +0 -373
  277. package/src/providers/tasks/index.js +0 -17
  278. package/src/providers/tasks/todoist.js +0 -286
  279. package/src/providers/types.js +0 -9
  280. package/src/providers/video-conferencing/google-meet.js +0 -286
  281. package/src/providers/video-conferencing/index.js +0 -31
  282. package/src/providers/video-conferencing/jitsi.js +0 -254
  283. package/src/providers/video-conferencing/teams.js +0 -270
  284. package/src/providers/video-conferencing/zoom.js +0 -332
  285. package/src/registry.js +0 -128
  286. package/src/tools/communication.js +0 -184
  287. package/src/tools/data.js +0 -205
  288. package/src/tools/index.js +0 -11
  289. package/src/tools/web.js +0 -137
  290. package/src/types.js +0 -10
  291. package/test/define.test.js +0 -306
  292. package/test/registry.test.js +0 -357
  293. package/test/tools.test.js +0 -363
@@ -0,0 +1,743 @@
1
+ /**
2
+ * Extended Tests for Data Tools
3
+ *
4
+ * Comprehensive edge case testing for parseJson, stringifyJson,
5
+ * parseCsv, transformData, and filterData tools.
6
+ */
7
+
8
+ import { describe, it, expect, beforeEach } from 'vitest'
9
+ import {
10
+ parseJson,
11
+ stringifyJson,
12
+ parseCsv,
13
+ transformData,
14
+ filterData,
15
+ dataTools,
16
+ registry,
17
+ registerBuiltinTools,
18
+ } from '../src/index.js'
19
+
20
+ describe('Data Tools - parseJson Extended', () => {
21
+ describe('valid JSON parsing', () => {
22
+ it('parses empty object', async () => {
23
+ const result = await parseJson.handler({ text: '{}' })
24
+ expect(result.valid).toBe(true)
25
+ expect(result.data).toEqual({})
26
+ })
27
+
28
+ it('parses empty array', async () => {
29
+ const result = await parseJson.handler({ text: '[]' })
30
+ expect(result.valid).toBe(true)
31
+ expect(result.data).toEqual([])
32
+ })
33
+
34
+ it('parses null', async () => {
35
+ const result = await parseJson.handler({ text: 'null' })
36
+ expect(result.valid).toBe(true)
37
+ expect(result.data).toBeNull()
38
+ })
39
+
40
+ it('parses boolean true', async () => {
41
+ const result = await parseJson.handler({ text: 'true' })
42
+ expect(result.valid).toBe(true)
43
+ expect(result.data).toBe(true)
44
+ })
45
+
46
+ it('parses boolean false', async () => {
47
+ const result = await parseJson.handler({ text: 'false' })
48
+ expect(result.valid).toBe(true)
49
+ expect(result.data).toBe(false)
50
+ })
51
+
52
+ it('parses integer number', async () => {
53
+ const result = await parseJson.handler({ text: '42' })
54
+ expect(result.valid).toBe(true)
55
+ expect(result.data).toBe(42)
56
+ })
57
+
58
+ it('parses float number', async () => {
59
+ const result = await parseJson.handler({ text: '3.14159' })
60
+ expect(result.valid).toBe(true)
61
+ expect(result.data).toBe(3.14159)
62
+ })
63
+
64
+ it('parses negative number', async () => {
65
+ const result = await parseJson.handler({ text: '-100' })
66
+ expect(result.valid).toBe(true)
67
+ expect(result.data).toBe(-100)
68
+ })
69
+
70
+ it('parses scientific notation', async () => {
71
+ const result = await parseJson.handler({ text: '1.5e10' })
72
+ expect(result.valid).toBe(true)
73
+ expect(result.data).toBe(1.5e10)
74
+ })
75
+
76
+ it('parses string', async () => {
77
+ const result = await parseJson.handler({ text: '"hello"' })
78
+ expect(result.valid).toBe(true)
79
+ expect(result.data).toBe('hello')
80
+ })
81
+
82
+ it('parses string with unicode', async () => {
83
+ const result = await parseJson.handler({
84
+ text: '"hello \\u0048\\u0065\\u006c\\u006c\\u006f"',
85
+ })
86
+ expect(result.valid).toBe(true)
87
+ expect(result.data).toBe('hello Hello')
88
+ })
89
+
90
+ it('parses nested objects', async () => {
91
+ const result = await parseJson.handler({
92
+ text: '{"a":{"b":{"c":{"d":"deep"}}}}',
93
+ })
94
+ expect(result.valid).toBe(true)
95
+ expect(result.data.a.b.c.d).toBe('deep')
96
+ })
97
+
98
+ it('parses array with mixed types', async () => {
99
+ const result = await parseJson.handler({
100
+ text: '[1, "two", true, null, {"five": 5}]',
101
+ })
102
+ expect(result.valid).toBe(true)
103
+ expect(result.data).toHaveLength(5)
104
+ expect(result.data[0]).toBe(1)
105
+ expect(result.data[1]).toBe('two')
106
+ expect(result.data[2]).toBe(true)
107
+ expect(result.data[3]).toBeNull()
108
+ expect(result.data[4]).toEqual({ five: 5 })
109
+ })
110
+
111
+ it('parses object with special characters in keys', async () => {
112
+ const result = await parseJson.handler({
113
+ text: '{"key-with-dash": 1, "key.with.dot": 2}',
114
+ })
115
+ expect(result.valid).toBe(true)
116
+ expect(result.data['key-with-dash']).toBe(1)
117
+ expect(result.data['key.with.dot']).toBe(2)
118
+ })
119
+ })
120
+
121
+ describe('invalid JSON handling', () => {
122
+ it('handles empty string', async () => {
123
+ const result = await parseJson.handler({ text: '' })
124
+ expect(result.valid).toBe(false)
125
+ expect(result.error).toBeDefined()
126
+ })
127
+
128
+ it('handles plain text', async () => {
129
+ const result = await parseJson.handler({ text: 'not json' })
130
+ expect(result.valid).toBe(false)
131
+ expect(result.error).toBeDefined()
132
+ })
133
+
134
+ it('handles malformed object', async () => {
135
+ const result = await parseJson.handler({ text: '{key: "value"}' })
136
+ expect(result.valid).toBe(false)
137
+ })
138
+
139
+ it('handles trailing comma', async () => {
140
+ const result = await parseJson.handler({ text: '{"a": 1,}' })
141
+ expect(result.valid).toBe(false)
142
+ })
143
+
144
+ it('handles single quotes', async () => {
145
+ const result = await parseJson.handler({ text: "{'key': 'value'}" })
146
+ expect(result.valid).toBe(false)
147
+ })
148
+
149
+ it('handles unclosed brace', async () => {
150
+ const result = await parseJson.handler({ text: '{"a": 1' })
151
+ expect(result.valid).toBe(false)
152
+ })
153
+
154
+ it('handles unclosed bracket', async () => {
155
+ const result = await parseJson.handler({ text: '[1, 2, 3' })
156
+ expect(result.valid).toBe(false)
157
+ })
158
+
159
+ it('handles undefined value', async () => {
160
+ const result = await parseJson.handler({ text: 'undefined' })
161
+ expect(result.valid).toBe(false)
162
+ })
163
+
164
+ it('sets data to null on error', async () => {
165
+ const result = await parseJson.handler({ text: 'invalid' })
166
+ expect(result.data).toBeNull()
167
+ })
168
+
169
+ it('provides error message', async () => {
170
+ const result = await parseJson.handler({ text: '{invalid}' })
171
+ expect(typeof result.error).toBe('string')
172
+ expect(result.error!.length).toBeGreaterThan(0)
173
+ })
174
+ })
175
+
176
+ describe('metadata', () => {
177
+ it('is idempotent', () => {
178
+ expect(parseJson.idempotent).toBe(true)
179
+ })
180
+
181
+ it('has json tag', () => {
182
+ expect(parseJson.tags).toContain('json')
183
+ })
184
+
185
+ it('has transform tag', () => {
186
+ expect(parseJson.tags).toContain('transform')
187
+ })
188
+ })
189
+ })
190
+
191
+ describe('Data Tools - stringifyJson Extended', () => {
192
+ describe('basic stringification', () => {
193
+ it('stringifies object', async () => {
194
+ const result = await stringifyJson.handler({ data: { a: 1 } })
195
+ expect(result.text).toBe('{"a":1}')
196
+ })
197
+
198
+ it('stringifies array', async () => {
199
+ const result = await stringifyJson.handler({ data: [1, 2, 3] })
200
+ expect(result.text).toBe('[1,2,3]')
201
+ })
202
+
203
+ it('stringifies null', async () => {
204
+ const result = await stringifyJson.handler({ data: null })
205
+ expect(result.text).toBe('null')
206
+ })
207
+
208
+ it('stringifies boolean', async () => {
209
+ const result = await stringifyJson.handler({ data: true })
210
+ expect(result.text).toBe('true')
211
+ })
212
+
213
+ it('stringifies number', async () => {
214
+ const result = await stringifyJson.handler({ data: 42 })
215
+ expect(result.text).toBe('42')
216
+ })
217
+
218
+ it('stringifies string', async () => {
219
+ const result = await stringifyJson.handler({ data: 'hello' })
220
+ expect(result.text).toBe('"hello"')
221
+ })
222
+
223
+ it('stringifies nested structure', async () => {
224
+ const result = await stringifyJson.handler({
225
+ data: { a: { b: [1, 2, { c: 3 }] } },
226
+ })
227
+ expect(JSON.parse(result.text)).toEqual({ a: { b: [1, 2, { c: 3 }] } })
228
+ })
229
+ })
230
+
231
+ describe('pretty printing', () => {
232
+ it('formats with 2-space indentation', async () => {
233
+ const result = await stringifyJson.handler({
234
+ data: { a: 1 },
235
+ pretty: true,
236
+ })
237
+ expect(result.text).toBe('{\n "a": 1\n}')
238
+ })
239
+
240
+ it('formats nested objects', async () => {
241
+ const result = await stringifyJson.handler({
242
+ data: { a: { b: 1 } },
243
+ pretty: true,
244
+ })
245
+ expect(result.text).toContain(' "a"')
246
+ expect(result.text).toContain(' "b"')
247
+ })
248
+
249
+ it('formats arrays', async () => {
250
+ const result = await stringifyJson.handler({
251
+ data: [1, 2, 3],
252
+ pretty: true,
253
+ })
254
+ expect(result.text).toContain('\n')
255
+ })
256
+
257
+ it('default is not pretty', async () => {
258
+ const result = await stringifyJson.handler({
259
+ data: { a: 1, b: 2 },
260
+ })
261
+ expect(result.text).not.toContain('\n')
262
+ })
263
+ })
264
+
265
+ describe('metadata', () => {
266
+ it('is idempotent', () => {
267
+ expect(stringifyJson.idempotent).toBe(true)
268
+ })
269
+
270
+ it('has stringify tag', () => {
271
+ expect(stringifyJson.tags).toContain('stringify')
272
+ })
273
+ })
274
+ })
275
+
276
+ describe('Data Tools - parseCsv Extended', () => {
277
+ describe('basic parsing', () => {
278
+ it('parses single row', async () => {
279
+ const result = await parseCsv.handler({ text: 'a,b,c\n1,2,3' })
280
+ expect(result.rows).toHaveLength(1)
281
+ expect(result.headers).toEqual(['a', 'b', 'c'])
282
+ })
283
+
284
+ it('parses multiple rows', async () => {
285
+ const result = await parseCsv.handler({
286
+ text: 'a,b\n1,2\n3,4\n5,6',
287
+ })
288
+ expect(result.rows).toHaveLength(3)
289
+ expect(result.rowCount).toBe(3)
290
+ })
291
+
292
+ it('handles empty file', async () => {
293
+ const result = await parseCsv.handler({ text: '' })
294
+ expect(result.rows).toHaveLength(0)
295
+ expect(result.headers).toHaveLength(0)
296
+ expect(result.rowCount).toBe(0)
297
+ })
298
+
299
+ it('handles header-only file', async () => {
300
+ const result = await parseCsv.handler({ text: 'a,b,c' })
301
+ expect(result.headers).toEqual(['a', 'b', 'c'])
302
+ expect(result.rows).toHaveLength(0)
303
+ })
304
+
305
+ it('trims whitespace from values', async () => {
306
+ const result = await parseCsv.handler({
307
+ text: 'a , b , c\n 1 , 2 , 3 ',
308
+ })
309
+ expect(result.headers).toEqual(['a', 'b', 'c'])
310
+ expect(result.rows[0]).toEqual({ a: '1', b: '2', c: '3' })
311
+ })
312
+ })
313
+
314
+ describe('delimiter options', () => {
315
+ it('parses semicolon-delimited', async () => {
316
+ const result = await parseCsv.handler({
317
+ text: 'a;b;c\n1;2;3',
318
+ delimiter: ';',
319
+ })
320
+ expect(result.headers).toEqual(['a', 'b', 'c'])
321
+ expect(result.rows[0]).toEqual({ a: '1', b: '2', c: '3' })
322
+ })
323
+
324
+ it('parses tab-delimited', async () => {
325
+ const result = await parseCsv.handler({
326
+ text: 'a\tb\tc\n1\t2\t3',
327
+ delimiter: '\t',
328
+ })
329
+ expect(result.headers).toEqual(['a', 'b', 'c'])
330
+ })
331
+
332
+ it('parses pipe-delimited', async () => {
333
+ const result = await parseCsv.handler({
334
+ text: 'a|b|c\n1|2|3',
335
+ delimiter: '|',
336
+ })
337
+ expect(result.headers).toEqual(['a', 'b', 'c'])
338
+ })
339
+
340
+ it('defaults to comma', async () => {
341
+ const result = await parseCsv.handler({
342
+ text: 'a,b,c\n1,2,3',
343
+ })
344
+ expect(result.headers).toEqual(['a', 'b', 'c'])
345
+ })
346
+ })
347
+
348
+ describe('header options', () => {
349
+ it('generates column names when no headers', async () => {
350
+ const result = await parseCsv.handler({
351
+ text: '1,2,3\n4,5,6',
352
+ hasHeaders: false,
353
+ })
354
+ expect(result.headers).toEqual(['column1', 'column2', 'column3'])
355
+ expect(result.rows).toHaveLength(2)
356
+ })
357
+
358
+ it('defaults to hasHeaders: true', async () => {
359
+ const result = await parseCsv.handler({
360
+ text: 'a,b\n1,2',
361
+ })
362
+ expect(result.headers).toEqual(['a', 'b'])
363
+ expect(result.rows).toHaveLength(1)
364
+ })
365
+
366
+ it('includes first row as data when no headers', async () => {
367
+ const result = await parseCsv.handler({
368
+ text: '1,2,3',
369
+ hasHeaders: false,
370
+ })
371
+ expect(result.rows).toHaveLength(1)
372
+ expect(result.rows[0]).toEqual({ column1: '1', column2: '2', column3: '3' })
373
+ })
374
+ })
375
+
376
+ describe('edge cases', () => {
377
+ it('handles missing values', async () => {
378
+ const result = await parseCsv.handler({
379
+ text: 'a,b,c\n1,,3',
380
+ })
381
+ expect(result.rows[0]).toEqual({ a: '1', b: '', c: '3' })
382
+ })
383
+
384
+ it('handles extra values', async () => {
385
+ const result = await parseCsv.handler({
386
+ text: 'a,b\n1,2,3',
387
+ })
388
+ // Extra values are ignored based on header count
389
+ expect(result.rows[0]).toHaveProperty('a', '1')
390
+ expect(result.rows[0]).toHaveProperty('b', '2')
391
+ })
392
+
393
+ it('handles fewer values than headers', async () => {
394
+ const result = await parseCsv.handler({
395
+ text: 'a,b,c\n1,2',
396
+ })
397
+ expect(result.rows[0]).toEqual({ a: '1', b: '2', c: '' })
398
+ })
399
+
400
+ it('filters empty lines', async () => {
401
+ const result = await parseCsv.handler({
402
+ text: 'a,b\n1,2\n\n3,4\n',
403
+ })
404
+ expect(result.rows).toHaveLength(2)
405
+ })
406
+
407
+ it('handles whitespace-only lines', async () => {
408
+ const result = await parseCsv.handler({
409
+ text: 'a,b\n1,2\n \n3,4',
410
+ })
411
+ expect(result.rows).toHaveLength(2)
412
+ })
413
+ })
414
+
415
+ describe('metadata', () => {
416
+ it('is idempotent', () => {
417
+ expect(parseCsv.idempotent).toBe(true)
418
+ })
419
+
420
+ it('has csv tag', () => {
421
+ expect(parseCsv.tags).toContain('csv')
422
+ })
423
+ })
424
+ })
425
+
426
+ describe('Data Tools - transformData Extended', () => {
427
+ describe('basic transformation', () => {
428
+ it('maps single field', async () => {
429
+ const result = await transformData.handler({
430
+ data: { name: 'John' },
431
+ transform: { fullName: 'name' },
432
+ })
433
+ expect(result.result).toEqual({ fullName: 'John' })
434
+ })
435
+
436
+ it('maps multiple fields', async () => {
437
+ const result = await transformData.handler({
438
+ data: { a: 1, b: 2, c: 3 },
439
+ transform: { x: 'a', y: 'b', z: 'c' },
440
+ })
441
+ expect(result.result).toEqual({ x: 1, y: 2, z: 3 })
442
+ })
443
+
444
+ it('selects subset of fields', async () => {
445
+ const result = await transformData.handler({
446
+ data: { a: 1, b: 2, c: 3, d: 4, e: 5 },
447
+ transform: { first: 'a', last: 'e' },
448
+ })
449
+ expect(result.result).toEqual({ first: 1, last: 5 })
450
+ })
451
+ })
452
+
453
+ describe('nested path access', () => {
454
+ it('accesses single level nesting', async () => {
455
+ const result = await transformData.handler({
456
+ data: { user: { name: 'John' } },
457
+ transform: { name: 'user.name' },
458
+ })
459
+ expect(result.result.name).toBe('John')
460
+ })
461
+
462
+ it('accesses deeply nested paths', async () => {
463
+ const result = await transformData.handler({
464
+ data: { a: { b: { c: { d: { e: 'deep' } } } } },
465
+ transform: { value: 'a.b.c.d.e' },
466
+ })
467
+ expect(result.result.value).toBe('deep')
468
+ })
469
+
470
+ it('accesses multiple nested paths', async () => {
471
+ const result = await transformData.handler({
472
+ data: {
473
+ user: { profile: { name: 'John', age: 30 } },
474
+ metadata: { created: '2024-01-01' },
475
+ },
476
+ transform: {
477
+ userName: 'user.profile.name',
478
+ userAge: 'user.profile.age',
479
+ createdAt: 'metadata.created',
480
+ },
481
+ })
482
+ expect(result.result).toEqual({
483
+ userName: 'John',
484
+ userAge: 30,
485
+ createdAt: '2024-01-01',
486
+ })
487
+ })
488
+
489
+ it('accesses array by index', async () => {
490
+ const result = await transformData.handler({
491
+ data: { items: ['a', 'b', 'c'] },
492
+ transform: { first: 'items.0', second: 'items.1' },
493
+ })
494
+ expect(result.result.first).toBe('a')
495
+ expect(result.result.second).toBe('b')
496
+ })
497
+ })
498
+
499
+ describe('missing path handling', () => {
500
+ it('returns undefined for missing top-level path', async () => {
501
+ const result = await transformData.handler({
502
+ data: { a: 1 },
503
+ transform: { value: 'b' },
504
+ })
505
+ expect(result.result.value).toBeUndefined()
506
+ })
507
+
508
+ it('returns undefined for missing nested path', async () => {
509
+ const result = await transformData.handler({
510
+ data: { a: { b: 1 } },
511
+ transform: { value: 'a.c.d' },
512
+ })
513
+ expect(result.result.value).toBeUndefined()
514
+ })
515
+
516
+ it('returns undefined for path on non-object', async () => {
517
+ const result = await transformData.handler({
518
+ data: { a: 'string' },
519
+ transform: { value: 'a.b' },
520
+ })
521
+ expect(result.result.value).toBeUndefined()
522
+ })
523
+
524
+ it('returns undefined for path on null', async () => {
525
+ const result = await transformData.handler({
526
+ data: { a: null },
527
+ transform: { value: 'a.b' },
528
+ })
529
+ expect(result.result.value).toBeUndefined()
530
+ })
531
+ })
532
+
533
+ describe('edge cases', () => {
534
+ it('handles empty transform', async () => {
535
+ const result = await transformData.handler({
536
+ data: { a: 1, b: 2 },
537
+ transform: {},
538
+ })
539
+ expect(result.result).toEqual({})
540
+ })
541
+
542
+ it('handles empty data', async () => {
543
+ const result = await transformData.handler({
544
+ data: {},
545
+ transform: { value: 'a' },
546
+ })
547
+ expect(result.result.value).toBeUndefined()
548
+ })
549
+
550
+ it('preserves value types', async () => {
551
+ const result = await transformData.handler({
552
+ data: {
553
+ str: 'hello',
554
+ num: 42,
555
+ bool: true,
556
+ arr: [1, 2],
557
+ obj: { nested: true },
558
+ nil: null,
559
+ },
560
+ transform: {
561
+ s: 'str',
562
+ n: 'num',
563
+ b: 'bool',
564
+ a: 'arr',
565
+ o: 'obj',
566
+ x: 'nil',
567
+ },
568
+ })
569
+ expect(result.result.s).toBe('hello')
570
+ expect(result.result.n).toBe(42)
571
+ expect(result.result.b).toBe(true)
572
+ expect(result.result.a).toEqual([1, 2])
573
+ expect(result.result.o).toEqual({ nested: true })
574
+ expect(result.result.x).toBeNull()
575
+ })
576
+ })
577
+ })
578
+
579
+ describe('Data Tools - filterData Extended', () => {
580
+ describe('basic filtering', () => {
581
+ it('filters by single criterion', async () => {
582
+ const result = await filterData.handler({
583
+ data: [{ status: 'active' }, { status: 'inactive' }, { status: 'active' }],
584
+ filter: { status: 'active' },
585
+ })
586
+ expect(result.count).toBe(2)
587
+ })
588
+
589
+ it('filters by multiple criteria', async () => {
590
+ const result = await filterData.handler({
591
+ data: [
592
+ { status: 'active', role: 'admin' },
593
+ { status: 'active', role: 'user' },
594
+ { status: 'inactive', role: 'admin' },
595
+ ],
596
+ filter: { status: 'active', role: 'admin' },
597
+ })
598
+ expect(result.count).toBe(1)
599
+ })
600
+
601
+ it('returns empty for no matches', async () => {
602
+ const result = await filterData.handler({
603
+ data: [{ a: 1 }, { a: 2 }],
604
+ filter: { a: 3 },
605
+ })
606
+ expect(result.results).toHaveLength(0)
607
+ expect(result.count).toBe(0)
608
+ })
609
+
610
+ it('returns all for empty filter', async () => {
611
+ const result = await filterData.handler({
612
+ data: [{ a: 1 }, { a: 2 }, { a: 3 }],
613
+ filter: {},
614
+ })
615
+ expect(result.count).toBe(3)
616
+ })
617
+ })
618
+
619
+ describe('type matching', () => {
620
+ it('matches string values', async () => {
621
+ const result = await filterData.handler({
622
+ data: [{ name: 'john' }, { name: 'jane' }],
623
+ filter: { name: 'john' },
624
+ })
625
+ expect(result.count).toBe(1)
626
+ })
627
+
628
+ it('matches number values', async () => {
629
+ const result = await filterData.handler({
630
+ data: [{ age: 25 }, { age: 30 }, { age: 25 }],
631
+ filter: { age: 25 },
632
+ })
633
+ expect(result.count).toBe(2)
634
+ })
635
+
636
+ it('matches boolean values', async () => {
637
+ const result = await filterData.handler({
638
+ data: [{ active: true }, { active: false }, { active: true }],
639
+ filter: { active: true },
640
+ })
641
+ expect(result.count).toBe(2)
642
+ })
643
+
644
+ it('matches null values', async () => {
645
+ const result = await filterData.handler({
646
+ data: [{ value: null }, { value: 1 }, { value: null }],
647
+ filter: { value: null },
648
+ })
649
+ expect(result.count).toBe(2)
650
+ })
651
+
652
+ it('uses strict equality', async () => {
653
+ const result = await filterData.handler({
654
+ data: [{ value: '1' }, { value: 1 }],
655
+ filter: { value: 1 },
656
+ })
657
+ expect(result.count).toBe(1)
658
+ })
659
+ })
660
+
661
+ describe('edge cases', () => {
662
+ it('handles empty data array', async () => {
663
+ const result = await filterData.handler({
664
+ data: [],
665
+ filter: { a: 1 },
666
+ })
667
+ expect(result.results).toHaveLength(0)
668
+ })
669
+
670
+ it('handles non-object items in array', async () => {
671
+ const result = await filterData.handler({
672
+ data: [1, 'string', null, { a: 1 }] as unknown[],
673
+ filter: { a: 1 },
674
+ })
675
+ expect(result.count).toBe(1)
676
+ })
677
+
678
+ it('skips items missing the filter key', async () => {
679
+ const result = await filterData.handler({
680
+ data: [{ a: 1 }, { b: 2 }, { a: 1 }],
681
+ filter: { a: 1 },
682
+ })
683
+ expect(result.count).toBe(2)
684
+ })
685
+
686
+ it('preserves original item structure', async () => {
687
+ const result = await filterData.handler({
688
+ data: [{ a: 1, b: 2, c: 3 }],
689
+ filter: { a: 1 },
690
+ })
691
+ expect(result.results[0]).toEqual({ a: 1, b: 2, c: 3 })
692
+ })
693
+ })
694
+ })
695
+
696
+ describe('Data Tools Array', () => {
697
+ it('has 5 tools', () => {
698
+ expect(dataTools).toHaveLength(5)
699
+ })
700
+
701
+ it('all tools are in data category', () => {
702
+ expect(dataTools.every((t) => t.category === 'data')).toBe(true)
703
+ })
704
+
705
+ it('all tools have transform subcategory', () => {
706
+ expect(dataTools.every((t) => t.subcategory === 'transform')).toBe(true)
707
+ })
708
+
709
+ it('all tools are idempotent', () => {
710
+ expect(dataTools.every((t) => t.idempotent === true)).toBe(true)
711
+ })
712
+ })
713
+
714
+ describe('Data Tools Registry Integration', () => {
715
+ beforeEach(() => {
716
+ registry.clear()
717
+ })
718
+
719
+ it('registers all data tools', () => {
720
+ registerBuiltinTools()
721
+
722
+ expect(registry.has('data.json.parse')).toBe(true)
723
+ expect(registry.has('data.json.stringify')).toBe(true)
724
+ expect(registry.has('data.csv.parse')).toBe(true)
725
+ expect(registry.has('data.transform')).toBe(true)
726
+ expect(registry.has('data.filter')).toBe(true)
727
+ })
728
+
729
+ it('can query by data category', () => {
730
+ registerBuiltinTools()
731
+
732
+ const tools = registry.byCategory('data')
733
+ expect(tools.length).toBeGreaterThanOrEqual(5)
734
+ })
735
+
736
+ it('can search by json tag', () => {
737
+ registerBuiltinTools()
738
+
739
+ const tools = registry.query({ tags: ['json'] })
740
+ expect(tools.some((t) => t.id === 'data.json.parse')).toBe(true)
741
+ expect(tools.some((t) => t.id === 'data.json.stringify')).toBe(true)
742
+ })
743
+ })