digital-tools 2.1.3 → 2.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.turbo/turbo-build.log +1 -1
- package/CHANGELOG.md +19 -0
- package/README.md +2 -0
- package/dist/client.d.ts +109 -0
- package/dist/client.d.ts.map +1 -0
- package/dist/client.js +69 -0
- package/dist/client.js.map +1 -0
- package/dist/define.d.ts +2 -2
- package/dist/define.d.ts.map +1 -1
- package/dist/define.js +21 -11
- package/dist/define.js.map +1 -1
- package/dist/function-ref.d.ts +229 -0
- package/dist/function-ref.d.ts.map +1 -0
- package/dist/function-ref.js +28 -0
- package/dist/function-ref.js.map +1 -0
- package/dist/function-sugar.d.ts +57 -0
- package/dist/function-sugar.d.ts.map +1 -0
- package/dist/function-sugar.js +79 -0
- package/dist/function-sugar.js.map +1 -0
- package/dist/index.d.ts +10 -3
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +24 -4
- package/dist/index.js.map +1 -1
- package/dist/providers/analytics/mixpanel.d.ts.map +1 -1
- package/dist/providers/analytics/mixpanel.js +21 -18
- package/dist/providers/analytics/mixpanel.js.map +1 -1
- package/dist/providers/calendar/cal-com.d.ts.map +1 -1
- package/dist/providers/calendar/cal-com.js +10 -10
- package/dist/providers/calendar/cal-com.js.map +1 -1
- package/dist/providers/calendar/google-calendar.d.ts.map +1 -1
- package/dist/providers/calendar/google-calendar.js +4 -4
- package/dist/providers/calendar/google-calendar.js.map +1 -1
- package/dist/providers/crm/hubspot.d.ts.map +1 -1
- package/dist/providers/crm/hubspot.js +107 -85
- package/dist/providers/crm/hubspot.js.map +1 -1
- package/dist/providers/development/github.d.ts.map +1 -1
- package/dist/providers/development/github.js +40 -43
- package/dist/providers/development/github.js.map +1 -1
- package/dist/providers/ecommerce/shopify.d.ts.map +1 -1
- package/dist/providers/ecommerce/shopify.js +79 -62
- package/dist/providers/ecommerce/shopify.js.map +1 -1
- package/dist/providers/email/resend.d.ts.map +1 -1
- package/dist/providers/email/resend.js +20 -16
- package/dist/providers/email/resend.js.map +1 -1
- package/dist/providers/email/sendgrid.d.ts.map +1 -1
- package/dist/providers/email/sendgrid.js +12 -9
- package/dist/providers/email/sendgrid.js.map +1 -1
- package/dist/providers/finance/stripe.d.ts.map +1 -1
- package/dist/providers/finance/stripe.js +44 -42
- package/dist/providers/finance/stripe.js.map +1 -1
- package/dist/providers/forms/typeform.d.ts.map +1 -1
- package/dist/providers/forms/typeform.js +68 -58
- package/dist/providers/forms/typeform.js.map +1 -1
- package/dist/providers/knowledge/notion.d.ts.map +1 -1
- package/dist/providers/knowledge/notion.js +75 -41
- package/dist/providers/knowledge/notion.js.map +1 -1
- package/dist/providers/marketing/mailchimp.d.ts.map +1 -1
- package/dist/providers/marketing/mailchimp.js +74 -61
- package/dist/providers/marketing/mailchimp.js.map +1 -1
- package/dist/providers/media/cloudinary.d.ts.map +1 -1
- package/dist/providers/media/cloudinary.js +30 -28
- package/dist/providers/media/cloudinary.js.map +1 -1
- package/dist/providers/messaging/slack.d.ts.map +1 -1
- package/dist/providers/messaging/slack.js +75 -58
- package/dist/providers/messaging/slack.js.map +1 -1
- package/dist/providers/messaging/twilio-sms.d.ts.map +1 -1
- package/dist/providers/messaging/twilio-sms.js +33 -15
- package/dist/providers/messaging/twilio-sms.js.map +1 -1
- package/dist/providers/project-management/linear.d.ts.map +1 -1
- package/dist/providers/project-management/linear.js +31 -27
- package/dist/providers/project-management/linear.js.map +1 -1
- package/dist/providers/spreadsheet/google-sheets.d.ts.map +1 -1
- package/dist/providers/spreadsheet/google-sheets.js +21 -18
- package/dist/providers/spreadsheet/google-sheets.js.map +1 -1
- package/dist/providers/spreadsheet/xlsx.d.ts.map +1 -1
- package/dist/providers/spreadsheet/xlsx.js +4 -4
- package/dist/providers/spreadsheet/xlsx.js.map +1 -1
- package/dist/providers/storage/index.js +1 -0
- package/dist/providers/storage/index.js.map +1 -1
- package/dist/providers/storage/s3.d.ts.map +1 -1
- package/dist/providers/storage/s3.js +36 -27
- package/dist/providers/storage/s3.js.map +1 -1
- package/dist/providers/support/zendesk.d.ts.map +1 -1
- package/dist/providers/support/zendesk.js +24 -25
- package/dist/providers/support/zendesk.js.map +1 -1
- package/dist/providers/tasks/todoist.d.ts.map +1 -1
- package/dist/providers/tasks/todoist.js +18 -18
- package/dist/providers/tasks/todoist.js.map +1 -1
- package/dist/providers/video-conferencing/google-meet.d.ts.map +1 -1
- package/dist/providers/video-conferencing/google-meet.js +11 -11
- package/dist/providers/video-conferencing/google-meet.js.map +1 -1
- package/dist/providers/video-conferencing/jitsi.js +14 -14
- package/dist/providers/video-conferencing/jitsi.js.map +1 -1
- package/dist/providers/video-conferencing/teams.d.ts.map +1 -1
- package/dist/providers/video-conferencing/teams.js +9 -7
- package/dist/providers/video-conferencing/teams.js.map +1 -1
- package/dist/providers/video-conferencing/zoom.d.ts.map +1 -1
- package/dist/providers/video-conferencing/zoom.js +26 -24
- package/dist/providers/video-conferencing/zoom.js.map +1 -1
- package/dist/tools/data.d.ts.map +1 -1
- package/dist/tools/data.js +5 -12
- package/dist/tools/data.js.map +1 -1
- package/dist/tools/index.d.ts +1 -0
- package/dist/tools/index.d.ts.map +1 -1
- package/dist/tools/index.js +1 -0
- package/dist/tools/index.js.map +1 -1
- package/dist/tools/system.d.ts +289 -0
- package/dist/tools/system.d.ts.map +1 -0
- package/dist/tools/system.js +752 -0
- package/dist/tools/system.js.map +1 -0
- package/dist/tools/web.d.ts.map +1 -1
- package/dist/tools/web.js +22 -10
- package/dist/tools/web.js.map +1 -1
- package/dist/track-record.d.ts +101 -0
- package/dist/track-record.d.ts.map +1 -0
- package/dist/track-record.js +17 -0
- package/dist/track-record.js.map +1 -0
- package/dist/types.d.ts +210 -9
- package/dist/types.d.ts.map +1 -1
- package/dist/verb-registration.d.ts +122 -0
- package/dist/verb-registration.d.ts.map +1 -0
- package/dist/verb-registration.js +176 -0
- package/dist/verb-registration.js.map +1 -0
- package/dist/worker.d.ts +93 -0
- package/dist/worker.d.ts.map +1 -0
- package/dist/worker.js +315 -0
- package/dist/worker.js.map +1 -0
- package/dist/wrap.d.ts +89 -0
- package/dist/wrap.d.ts.map +1 -0
- package/dist/wrap.js +225 -0
- package/dist/wrap.js.map +1 -0
- package/package.json +31 -14
- package/src/client.ts +136 -0
- package/src/define.ts +30 -24
- package/src/function-ref.ts +264 -0
- package/src/function-sugar.ts +134 -0
- package/src/index.ts +132 -10
- package/src/providers/analytics/mixpanel.ts +19 -18
- package/src/providers/calendar/cal-com.ts +29 -18
- package/src/providers/calendar/google-calendar.ts +20 -14
- package/src/providers/crm/hubspot.ts +225 -99
- package/src/providers/development/github.ts +206 -135
- package/src/providers/ecommerce/shopify.ts +250 -89
- package/src/providers/email/resend.ts +101 -28
- package/src/providers/email/sendgrid.ts +12 -9
- package/src/providers/finance/stripe.ts +128 -49
- package/src/providers/forms/typeform.ts +74 -58
- package/src/providers/knowledge/notion.ts +340 -88
- package/src/providers/marketing/mailchimp.ts +86 -70
- package/src/providers/media/cloudinary.ts +99 -41
- package/src/providers/messaging/slack.ts +283 -85
- package/src/providers/messaging/twilio-sms.ts +35 -15
- package/src/providers/project-management/linear.ts +143 -55
- package/src/providers/spreadsheet/google-sheets.ts +222 -56
- package/src/providers/spreadsheet/xlsx.ts +47 -16
- package/src/providers/storage/s3.ts +119 -47
- package/src/providers/support/zendesk.ts +196 -46
- package/src/providers/tasks/todoist.ts +20 -26
- package/src/providers/video-conferencing/google-meet.ts +17 -20
- package/src/providers/video-conferencing/jitsi.ts +14 -14
- package/src/providers/video-conferencing/teams.ts +14 -13
- package/src/providers/video-conferencing/zoom.ts +54 -49
- package/src/tools/data.ts +6 -16
- package/src/tools/index.ts +1 -0
- package/src/tools/system.ts +887 -0
- package/src/tools/web.ts +22 -10
- package/src/track-record.ts +106 -0
- package/src/types.ts +241 -13
- package/src/verb-registration.ts +197 -0
- package/src/worker.ts +370 -0
- package/src/wrap.ts +260 -0
- package/test/client.test.ts +146 -0
- package/test/communication-tools-extended.test.ts +734 -0
- package/test/data-tools-extended.test.ts +743 -0
- package/test/define-extended.test.ts +819 -0
- package/test/define.test.ts +150 -41
- package/test/entities.test.ts +623 -0
- package/test/extended-entities.test.ts +1228 -0
- package/test/provider-implementations.test.ts +725 -0
- package/test/provider-registry-extended.test.ts +583 -0
- package/test/providers/google-sheets.test.ts +851 -0
- package/test/providers/helpers.ts +554 -0
- package/test/providers/hubspot.test.ts +576 -0
- package/test/providers/slack.test.ts +932 -0
- package/test/providers/stripe.test.ts +701 -0
- package/test/providers.test.ts +578 -0
- package/test/system-tools-extended.test.ts +632 -0
- package/test/system.test.ts +673 -0
- package/test/tools.test.ts +15 -11
- package/test/types.test.ts +402 -0
- package/test/verb-registration.test.ts +395 -0
- package/test/web-tools.test.ts +553 -0
- package/test/worker-extended.test.ts +699 -0
- package/test/worker.test.ts +576 -0
- package/test/wrap.test.ts +366 -0
- package/tsconfig.json +3 -13
- package/vitest.config.ts +37 -0
- package/wrangler.jsonc +9 -0
- package/LICENSE +0 -21
- package/dist/providers/voice/vapi.d.ts +0 -27
- package/dist/providers/voice/vapi.d.ts.map +0 -1
- package/dist/providers/voice/vapi.js +0 -440
- package/dist/providers/voice/vapi.js.map +0 -1
- package/src/define.js +0 -259
- package/src/entities/advertising.js +0 -999
- package/src/entities/ai.js +0 -756
- package/src/entities/analytics.js +0 -1588
- package/src/entities/automation.js +0 -601
- package/src/entities/communication.js +0 -1150
- package/src/entities/crm.js +0 -1386
- package/src/entities/design.js +0 -546
- package/src/entities/development.js +0 -2212
- package/src/entities/document.js +0 -874
- package/src/entities/ecommerce.js +0 -1429
- package/src/entities/experiment.js +0 -1039
- package/src/entities/finance.js +0 -3478
- package/src/entities/forms.js +0 -1892
- package/src/entities/hr.js +0 -661
- package/src/entities/identity.js +0 -997
- package/src/entities/index.js +0 -282
- package/src/entities/infrastructure.js +0 -1153
- package/src/entities/knowledge.js +0 -1438
- package/src/entities/marketing.js +0 -1610
- package/src/entities/media.js +0 -1634
- package/src/entities/notification.js +0 -1199
- package/src/entities/presentation.js +0 -1274
- package/src/entities/productivity.js +0 -1317
- package/src/entities/project-management.js +0 -1136
- package/src/entities/recruiting.js +0 -736
- package/src/entities/shipping.js +0 -509
- package/src/entities/signature.js +0 -1102
- package/src/entities/site.js +0 -222
- package/src/entities/spreadsheet.js +0 -1341
- package/src/entities/storage.js +0 -1198
- package/src/entities/support.js +0 -1166
- package/src/entities/video-conferencing.js +0 -1750
- package/src/entities/video.js +0 -950
- package/src/entities.js +0 -1663
- package/src/index.js +0 -74
- package/src/providers/analytics/index.js +0 -17
- package/src/providers/analytics/mixpanel.js +0 -255
- package/src/providers/calendar/cal-com.js +0 -303
- package/src/providers/calendar/google-calendar.js +0 -335
- package/src/providers/calendar/index.js +0 -20
- package/src/providers/crm/hubspot.js +0 -566
- package/src/providers/crm/index.js +0 -17
- package/src/providers/development/github.js +0 -472
- package/src/providers/development/index.js +0 -17
- package/src/providers/ecommerce/index.js +0 -17
- package/src/providers/ecommerce/shopify.js +0 -378
- package/src/providers/email/index.js +0 -20
- package/src/providers/email/resend.js +0 -258
- package/src/providers/email/sendgrid.js +0 -161
- package/src/providers/finance/index.js +0 -17
- package/src/providers/finance/stripe.js +0 -549
- package/src/providers/forms/index.js +0 -17
- package/src/providers/forms/typeform.js +0 -500
- package/src/providers/index.js +0 -123
- package/src/providers/knowledge/index.js +0 -17
- package/src/providers/knowledge/notion.js +0 -389
- package/src/providers/marketing/index.js +0 -17
- package/src/providers/marketing/mailchimp.js +0 -443
- package/src/providers/media/cloudinary.js +0 -318
- package/src/providers/media/index.js +0 -17
- package/src/providers/messaging/index.js +0 -20
- package/src/providers/messaging/slack.js +0 -393
- package/src/providers/messaging/twilio-sms.js +0 -249
- package/src/providers/project-management/index.js +0 -17
- package/src/providers/project-management/linear.js +0 -575
- package/src/providers/registry.js +0 -86
- package/src/providers/spreadsheet/google-sheets.js +0 -375
- package/src/providers/spreadsheet/index.js +0 -20
- package/src/providers/spreadsheet/xlsx.js +0 -423
- package/src/providers/storage/index.js +0 -24
- package/src/providers/storage/s3.js +0 -419
- package/src/providers/support/index.js +0 -17
- package/src/providers/support/zendesk.js +0 -373
- package/src/providers/tasks/index.js +0 -17
- package/src/providers/tasks/todoist.js +0 -286
- package/src/providers/types.js +0 -9
- package/src/providers/video-conferencing/google-meet.js +0 -286
- package/src/providers/video-conferencing/index.js +0 -31
- package/src/providers/video-conferencing/jitsi.js +0 -254
- package/src/providers/video-conferencing/teams.js +0 -270
- package/src/providers/video-conferencing/zoom.js +0 -332
- package/src/registry.js +0 -128
- package/src/tools/communication.js +0 -184
- package/src/tools/data.js +0 -205
- package/src/tools/index.js +0 -11
- package/src/tools/web.js +0 -137
- package/src/types.js +0 -10
- package/test/define.test.js +0 -306
- package/test/registry.test.js +0 -357
- 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
|
+
})
|