@applicaster/zapp-react-native-utils 15.0.0-rc.94 → 15.0.0-rc.95
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/adsUtils/__tests__/createVMAP.test.ts +419 -0
- package/appDataUtils/__tests__/urlScheme.test.ts +678 -0
- package/appUtils/HooksManager/__tests__/__snapshots__/hooksManager.test.js.snap +0 -188
- package/appUtils/HooksManager/__tests__/hooksManager.test.js +16 -2
- package/cloudEventsUtils/__tests__/index.test.ts +529 -0
- package/cloudEventsUtils/index.ts +65 -1
- package/dateUtils/__tests__/dayjs.test.ts +330 -0
- package/enumUtils/__tests__/getEnumKeyByEnumValue.test.ts +207 -0
- package/errorUtils/__tests__/GeneralError.test.ts +97 -0
- package/errorUtils/__tests__/HttpStatusCode.test.ts +344 -0
- package/errorUtils/__tests__/MissingPluginError.test.ts +113 -0
- package/errorUtils/__tests__/NetworkError.test.ts +202 -0
- package/errorUtils/__tests__/getParsedResponse.test.ts +188 -0
- package/errorUtils/__tests__/invariant.test.ts +112 -0
- package/package.json +2 -2
|
@@ -0,0 +1,529 @@
|
|
|
1
|
+
// Mock dependencies are set up before requiring the module
|
|
2
|
+
// The module is required dynamically in beforeEach to allow test-specific setup
|
|
3
|
+
// and state is reset using resetCloudEventsSource() between tests
|
|
4
|
+
jest.mock("uuid", () => ({
|
|
5
|
+
v4: jest.fn(() => "test-uuid-1234"),
|
|
6
|
+
}));
|
|
7
|
+
|
|
8
|
+
jest.mock("@applicaster/zapp-pipes-v2-client", () => ({
|
|
9
|
+
RequestBuilder: jest.fn().mockImplementation(() => ({
|
|
10
|
+
setUrl: jest.fn().mockReturnThis(),
|
|
11
|
+
setMethod: jest.fn().mockReturnThis(),
|
|
12
|
+
addBodyParams: jest.fn().mockReturnThis(),
|
|
13
|
+
call: jest.fn().mockResolvedValue({ success: true }),
|
|
14
|
+
})),
|
|
15
|
+
}));
|
|
16
|
+
|
|
17
|
+
jest.mock(
|
|
18
|
+
"@applicaster/zapp-react-native-bridge/ZappStorage/SessionStorage",
|
|
19
|
+
() => ({
|
|
20
|
+
sessionStorage: {
|
|
21
|
+
getItem: jest.fn(),
|
|
22
|
+
},
|
|
23
|
+
})
|
|
24
|
+
);
|
|
25
|
+
|
|
26
|
+
jest.mock("@applicaster/zapp-react-native-utils/appDataUtils", () => ({
|
|
27
|
+
getAppUrlScheme: jest.fn(),
|
|
28
|
+
}));
|
|
29
|
+
|
|
30
|
+
import { v4 as uuid } from "uuid";
|
|
31
|
+
import { RequestBuilder } from "@applicaster/zapp-pipes-v2-client";
|
|
32
|
+
import { sessionStorage } from "@applicaster/zapp-react-native-bridge/ZappStorage/SessionStorage";
|
|
33
|
+
import { getAppUrlScheme } from "@applicaster/zapp-react-native-utils/appDataUtils";
|
|
34
|
+
import type { CloudEvent, CreateCloudEventParams } from "../index";
|
|
35
|
+
|
|
36
|
+
describe("cloudEventsUtils", () => {
|
|
37
|
+
const mockGetAppUrlScheme = getAppUrlScheme as jest.MockedFunction<
|
|
38
|
+
typeof getAppUrlScheme
|
|
39
|
+
>;
|
|
40
|
+
|
|
41
|
+
const mockSessionStorageGetItem =
|
|
42
|
+
sessionStorage.getItem as jest.MockedFunction<
|
|
43
|
+
typeof sessionStorage.getItem
|
|
44
|
+
>;
|
|
45
|
+
|
|
46
|
+
const mockUuid = uuid as jest.MockedFunction<typeof uuid>;
|
|
47
|
+
|
|
48
|
+
let createCloudEvent: <T = any>(
|
|
49
|
+
params: CreateCloudEventParams<T>
|
|
50
|
+
) => Promise<CloudEvent<T>>;
|
|
51
|
+
|
|
52
|
+
let sendCloudEvent: <T = any>(
|
|
53
|
+
cloudEvent: CloudEvent<T>,
|
|
54
|
+
url: string
|
|
55
|
+
) => Promise<any>;
|
|
56
|
+
|
|
57
|
+
beforeEach(() => {
|
|
58
|
+
jest.clearAllMocks();
|
|
59
|
+
|
|
60
|
+
// Setup default mocks
|
|
61
|
+
mockGetAppUrlScheme.mockReturnValue("myapp");
|
|
62
|
+
|
|
63
|
+
mockSessionStorageGetItem
|
|
64
|
+
.mockResolvedValueOnce("com.example.app") // bundleId
|
|
65
|
+
.mockResolvedValueOnce("1.0.0"); // appVersion
|
|
66
|
+
|
|
67
|
+
mockUuid.mockReturnValue("test-uuid-1234");
|
|
68
|
+
|
|
69
|
+
const utils = require("../");
|
|
70
|
+
createCloudEvent = utils.createCloudEvent;
|
|
71
|
+
sendCloudEvent = utils.sendCloudEvent;
|
|
72
|
+
utils.resetCloudEventsSource();
|
|
73
|
+
|
|
74
|
+
// Note: Source caching means tests will reuse the source from the first test
|
|
75
|
+
// Tests that need specific source values or error conditions should run in isolation
|
|
76
|
+
});
|
|
77
|
+
|
|
78
|
+
describe("createCloudEvent", () => {
|
|
79
|
+
it("should create a cloud event with all required fields", async () => {
|
|
80
|
+
const { createCloudEvent } = require("../index");
|
|
81
|
+
|
|
82
|
+
const event = await createCloudEvent({
|
|
83
|
+
type: "user.action",
|
|
84
|
+
subject: "button.click",
|
|
85
|
+
data: { buttonId: "submit" },
|
|
86
|
+
});
|
|
87
|
+
|
|
88
|
+
expect(event).toHaveProperty("specversion", "1.0");
|
|
89
|
+
expect(event).toHaveProperty("type", "user.action");
|
|
90
|
+
expect(event).toHaveProperty("subject", "button.click");
|
|
91
|
+
expect(event).toHaveProperty("datacontenttype", "application/json");
|
|
92
|
+
expect(event).toHaveProperty("source");
|
|
93
|
+
expect(event).toHaveProperty("time");
|
|
94
|
+
expect(event).toHaveProperty("id", "test-uuid-1234");
|
|
95
|
+
expect(event).toHaveProperty("data", { buttonId: "submit" });
|
|
96
|
+
});
|
|
97
|
+
|
|
98
|
+
it("should generate source from appUrlScheme, bundleId, and appVersion", async () => {
|
|
99
|
+
mockGetAppUrlScheme.mockReset();
|
|
100
|
+
mockGetAppUrlScheme.mockReturnValue("testapp");
|
|
101
|
+
|
|
102
|
+
mockSessionStorageGetItem.mockReset();
|
|
103
|
+
|
|
104
|
+
mockSessionStorageGetItem
|
|
105
|
+
.mockResolvedValueOnce("com.test.bundle")
|
|
106
|
+
.mockResolvedValueOnce("2.3.4");
|
|
107
|
+
|
|
108
|
+
const { createCloudEvent } = require("../index");
|
|
109
|
+
|
|
110
|
+
const event = await createCloudEvent({
|
|
111
|
+
type: "test.event",
|
|
112
|
+
subject: "test",
|
|
113
|
+
data: {},
|
|
114
|
+
});
|
|
115
|
+
|
|
116
|
+
expect(event.source).toBe("testapp://com.test.bundle/versions/2.3.4");
|
|
117
|
+
});
|
|
118
|
+
|
|
119
|
+
it("should generate unique ID using uuid", async () => {
|
|
120
|
+
mockUuid.mockReturnValueOnce("uuid-1").mockReturnValueOnce("uuid-2");
|
|
121
|
+
|
|
122
|
+
const { createCloudEvent } = require("../index");
|
|
123
|
+
|
|
124
|
+
const event1 = await createCloudEvent({
|
|
125
|
+
type: "event1",
|
|
126
|
+
subject: "subject1",
|
|
127
|
+
data: {},
|
|
128
|
+
});
|
|
129
|
+
|
|
130
|
+
const event2 = await createCloudEvent({
|
|
131
|
+
type: "event2",
|
|
132
|
+
subject: "subject2",
|
|
133
|
+
data: {},
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
expect(event1.id).toBe("uuid-1");
|
|
137
|
+
expect(event2.id).toBe("uuid-2");
|
|
138
|
+
});
|
|
139
|
+
|
|
140
|
+
it("should generate ISO timestamp for time field", async () => {
|
|
141
|
+
const event = await createCloudEvent({
|
|
142
|
+
type: "test.event",
|
|
143
|
+
subject: "test",
|
|
144
|
+
data: {},
|
|
145
|
+
});
|
|
146
|
+
|
|
147
|
+
// Verify ISO 8601 format
|
|
148
|
+
expect(event.time).toMatch(
|
|
149
|
+
/^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}\.\d{3}Z$/
|
|
150
|
+
);
|
|
151
|
+
|
|
152
|
+
// Verify it's a valid date
|
|
153
|
+
expect(new Date(event.time).getTime()).toBeGreaterThan(0);
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
it("should include custom data in event", async () => {
|
|
157
|
+
const customData = {
|
|
158
|
+
userId: "user123",
|
|
159
|
+
action: "purchase",
|
|
160
|
+
amount: 99.99,
|
|
161
|
+
items: ["item1", "item2"],
|
|
162
|
+
};
|
|
163
|
+
|
|
164
|
+
const event = await createCloudEvent({
|
|
165
|
+
type: "commerce.purchase",
|
|
166
|
+
subject: "order.completed",
|
|
167
|
+
data: customData,
|
|
168
|
+
});
|
|
169
|
+
|
|
170
|
+
expect(event.data).toEqual(customData);
|
|
171
|
+
});
|
|
172
|
+
|
|
173
|
+
it("should throw error when appUrlScheme is missing", async () => {
|
|
174
|
+
jest.clearAllMocks();
|
|
175
|
+
mockSessionStorageGetItem.mockReset();
|
|
176
|
+
mockGetAppUrlScheme.mockReturnValue(null as any);
|
|
177
|
+
|
|
178
|
+
mockSessionStorageGetItem
|
|
179
|
+
.mockResolvedValueOnce("com.example.app")
|
|
180
|
+
.mockResolvedValueOnce("1.0.0");
|
|
181
|
+
|
|
182
|
+
mockUuid.mockReturnValue("test-uuid-1234");
|
|
183
|
+
|
|
184
|
+
const { createCloudEvent: freshCreate } = require("../index");
|
|
185
|
+
|
|
186
|
+
await expect(
|
|
187
|
+
freshCreate({
|
|
188
|
+
type: "test",
|
|
189
|
+
subject: "test",
|
|
190
|
+
data: {},
|
|
191
|
+
})
|
|
192
|
+
).rejects.toThrow("Can not create client data addSource");
|
|
193
|
+
});
|
|
194
|
+
|
|
195
|
+
it("should throw error when bundleId is missing", async () => {
|
|
196
|
+
jest.clearAllMocks();
|
|
197
|
+
mockSessionStorageGetItem.mockReset();
|
|
198
|
+
mockGetAppUrlScheme.mockReturnValue("myapp");
|
|
199
|
+
|
|
200
|
+
mockSessionStorageGetItem
|
|
201
|
+
.mockResolvedValueOnce(null) // bundleId
|
|
202
|
+
.mockResolvedValueOnce("1.0.0");
|
|
203
|
+
|
|
204
|
+
mockUuid.mockReturnValue("test-uuid-1234");
|
|
205
|
+
|
|
206
|
+
const { createCloudEvent: freshCreate } = require("../index");
|
|
207
|
+
|
|
208
|
+
await expect(
|
|
209
|
+
freshCreate({
|
|
210
|
+
type: "test",
|
|
211
|
+
subject: "test",
|
|
212
|
+
data: {},
|
|
213
|
+
})
|
|
214
|
+
).rejects.toThrow("Can not create client data addSource");
|
|
215
|
+
});
|
|
216
|
+
|
|
217
|
+
it("should throw error when appVersion is missing", async () => {
|
|
218
|
+
jest.clearAllMocks();
|
|
219
|
+
mockSessionStorageGetItem.mockReset();
|
|
220
|
+
mockGetAppUrlScheme.mockReturnValue("myapp");
|
|
221
|
+
|
|
222
|
+
mockSessionStorageGetItem
|
|
223
|
+
.mockResolvedValueOnce("com.example.app")
|
|
224
|
+
.mockResolvedValueOnce(null); // appVersion
|
|
225
|
+
|
|
226
|
+
mockUuid.mockReturnValue("test-uuid-1234");
|
|
227
|
+
|
|
228
|
+
const { createCloudEvent: freshCreate } = require("../index");
|
|
229
|
+
|
|
230
|
+
await expect(
|
|
231
|
+
freshCreate({
|
|
232
|
+
type: "test",
|
|
233
|
+
subject: "test",
|
|
234
|
+
data: {},
|
|
235
|
+
})
|
|
236
|
+
).rejects.toThrow("Can not create client data addSource");
|
|
237
|
+
});
|
|
238
|
+
|
|
239
|
+
it("should cache source after first call", async () => {
|
|
240
|
+
jest.clearAllMocks();
|
|
241
|
+
mockGetAppUrlScheme.mockReturnValue("myapp");
|
|
242
|
+
|
|
243
|
+
mockSessionStorageGetItem.mockReset();
|
|
244
|
+
|
|
245
|
+
mockSessionStorageGetItem
|
|
246
|
+
.mockResolvedValueOnce("com.example.app")
|
|
247
|
+
.mockResolvedValueOnce("1.0.0");
|
|
248
|
+
|
|
249
|
+
mockUuid.mockReturnValueOnce("uuid-1").mockReturnValueOnce("uuid-2");
|
|
250
|
+
|
|
251
|
+
const freshUtils = require("../index");
|
|
252
|
+
freshUtils.resetCloudEventsSource();
|
|
253
|
+
|
|
254
|
+
const { createCloudEvent: freshCreate } = freshUtils;
|
|
255
|
+
|
|
256
|
+
const event1 = await freshCreate({
|
|
257
|
+
type: "event1",
|
|
258
|
+
subject: "subject1",
|
|
259
|
+
data: {},
|
|
260
|
+
});
|
|
261
|
+
|
|
262
|
+
const event2 = await freshCreate({
|
|
263
|
+
type: "event2",
|
|
264
|
+
subject: "subject2",
|
|
265
|
+
data: {},
|
|
266
|
+
});
|
|
267
|
+
|
|
268
|
+
expect(event1.source).toBe(event2.source);
|
|
269
|
+
expect(mockGetAppUrlScheme).toHaveBeenCalledTimes(1);
|
|
270
|
+
expect(mockSessionStorageGetItem).toHaveBeenCalledTimes(2);
|
|
271
|
+
});
|
|
272
|
+
|
|
273
|
+
it("should handle empty data object", async () => {
|
|
274
|
+
const event = await createCloudEvent({
|
|
275
|
+
type: "test.event",
|
|
276
|
+
subject: "test.subject",
|
|
277
|
+
data: {},
|
|
278
|
+
});
|
|
279
|
+
|
|
280
|
+
expect(event.data).toEqual({});
|
|
281
|
+
});
|
|
282
|
+
|
|
283
|
+
it("should handle null data", async () => {
|
|
284
|
+
const event = await createCloudEvent({
|
|
285
|
+
type: "test.event",
|
|
286
|
+
subject: "test.subject",
|
|
287
|
+
data: null,
|
|
288
|
+
});
|
|
289
|
+
|
|
290
|
+
expect(event.data).toBeNull();
|
|
291
|
+
});
|
|
292
|
+
|
|
293
|
+
it("should handle complex nested data", async () => {
|
|
294
|
+
const complexData = {
|
|
295
|
+
level1: {
|
|
296
|
+
level2: {
|
|
297
|
+
level3: {
|
|
298
|
+
value: "deep",
|
|
299
|
+
},
|
|
300
|
+
},
|
|
301
|
+
},
|
|
302
|
+
array: [1, 2, { nested: true }],
|
|
303
|
+
};
|
|
304
|
+
|
|
305
|
+
const event = await createCloudEvent({
|
|
306
|
+
type: "test.event",
|
|
307
|
+
subject: "test.subject",
|
|
308
|
+
data: complexData,
|
|
309
|
+
});
|
|
310
|
+
|
|
311
|
+
expect(event.data).toEqual(complexData);
|
|
312
|
+
});
|
|
313
|
+
});
|
|
314
|
+
|
|
315
|
+
describe("sendCloudEvent", () => {
|
|
316
|
+
let mockRequestBuilder: any;
|
|
317
|
+
|
|
318
|
+
beforeEach(() => {
|
|
319
|
+
mockRequestBuilder = {
|
|
320
|
+
setUrl: jest.fn().mockReturnThis(),
|
|
321
|
+
setMethod: jest.fn().mockReturnThis(),
|
|
322
|
+
addBodyParams: jest.fn().mockReturnThis(),
|
|
323
|
+
call: jest.fn().mockResolvedValue({ success: true, data: "response" }),
|
|
324
|
+
};
|
|
325
|
+
|
|
326
|
+
(
|
|
327
|
+
RequestBuilder as jest.MockedClass<typeof RequestBuilder>
|
|
328
|
+
).mockImplementation(() => mockRequestBuilder);
|
|
329
|
+
});
|
|
330
|
+
|
|
331
|
+
it("should send cloud event to specified URL", async () => {
|
|
332
|
+
const cloudEvent: CloudEvent = {
|
|
333
|
+
specversion: "1.0" as const,
|
|
334
|
+
type: "test.event",
|
|
335
|
+
subject: "test",
|
|
336
|
+
source: "test://source",
|
|
337
|
+
id: "123",
|
|
338
|
+
time: new Date().toISOString(),
|
|
339
|
+
datacontenttype: "application/json" as const,
|
|
340
|
+
data: {},
|
|
341
|
+
};
|
|
342
|
+
|
|
343
|
+
const url = "https://api.example.com/events";
|
|
344
|
+
|
|
345
|
+
await sendCloudEvent(cloudEvent, url);
|
|
346
|
+
|
|
347
|
+
expect(mockRequestBuilder.setUrl).toHaveBeenCalledWith(url);
|
|
348
|
+
expect(mockRequestBuilder.setMethod).toHaveBeenCalledWith("POST");
|
|
349
|
+
expect(mockRequestBuilder.addBodyParams).toHaveBeenCalledWith(cloudEvent);
|
|
350
|
+
expect(mockRequestBuilder.call).toHaveBeenCalled();
|
|
351
|
+
});
|
|
352
|
+
|
|
353
|
+
it("should return response from API call", async () => {
|
|
354
|
+
const cloudEvent: CloudEvent = {
|
|
355
|
+
specversion: "1.0" as const,
|
|
356
|
+
type: "test.event",
|
|
357
|
+
subject: "test",
|
|
358
|
+
source: "test://source",
|
|
359
|
+
id: "123",
|
|
360
|
+
time: new Date().toISOString(),
|
|
361
|
+
datacontenttype: "application/json" as const,
|
|
362
|
+
data: {},
|
|
363
|
+
};
|
|
364
|
+
|
|
365
|
+
const expectedResponse = { success: true, eventId: "evt-123" };
|
|
366
|
+
mockRequestBuilder.call.mockResolvedValue(expectedResponse);
|
|
367
|
+
|
|
368
|
+
const result = await sendCloudEvent(
|
|
369
|
+
cloudEvent,
|
|
370
|
+
"https://api.example.com/events"
|
|
371
|
+
);
|
|
372
|
+
|
|
373
|
+
expect(result).toEqual(expectedResponse);
|
|
374
|
+
});
|
|
375
|
+
|
|
376
|
+
it("should throw error when URL is missing", async () => {
|
|
377
|
+
const cloudEvent: CloudEvent = {
|
|
378
|
+
specversion: "1.0" as const,
|
|
379
|
+
type: "test.event",
|
|
380
|
+
subject: "test",
|
|
381
|
+
source: "test://source",
|
|
382
|
+
id: "123",
|
|
383
|
+
time: new Date().toISOString(),
|
|
384
|
+
datacontenttype: "application/json" as const,
|
|
385
|
+
data: {},
|
|
386
|
+
};
|
|
387
|
+
|
|
388
|
+
await expect(sendCloudEvent(cloudEvent, "")).rejects.toThrow(
|
|
389
|
+
"Feed locator is missing"
|
|
390
|
+
);
|
|
391
|
+
});
|
|
392
|
+
|
|
393
|
+
it("should throw error when URL is null", async () => {
|
|
394
|
+
const cloudEvent: CloudEvent = {
|
|
395
|
+
specversion: "1.0" as const,
|
|
396
|
+
type: "test.event",
|
|
397
|
+
subject: "test",
|
|
398
|
+
source: "test://source",
|
|
399
|
+
id: "123",
|
|
400
|
+
time: new Date().toISOString(),
|
|
401
|
+
datacontenttype: "application/json" as const,
|
|
402
|
+
data: {},
|
|
403
|
+
};
|
|
404
|
+
|
|
405
|
+
await expect(sendCloudEvent(cloudEvent, null as any)).rejects.toThrow(
|
|
406
|
+
"Feed locator is missing"
|
|
407
|
+
);
|
|
408
|
+
});
|
|
409
|
+
|
|
410
|
+
it("should throw error when URL is undefined", async () => {
|
|
411
|
+
const cloudEvent: CloudEvent = {
|
|
412
|
+
specversion: "1.0" as const,
|
|
413
|
+
type: "test.event",
|
|
414
|
+
subject: "test",
|
|
415
|
+
source: "test://source",
|
|
416
|
+
id: "123",
|
|
417
|
+
time: new Date().toISOString(),
|
|
418
|
+
datacontenttype: "application/json" as const,
|
|
419
|
+
data: {},
|
|
420
|
+
};
|
|
421
|
+
|
|
422
|
+
await expect(
|
|
423
|
+
sendCloudEvent(cloudEvent, undefined as any)
|
|
424
|
+
).rejects.toThrow("Feed locator is missing");
|
|
425
|
+
});
|
|
426
|
+
|
|
427
|
+
it("should handle API errors", async () => {
|
|
428
|
+
const cloudEvent: CloudEvent = {
|
|
429
|
+
specversion: "1.0" as const,
|
|
430
|
+
type: "test.event",
|
|
431
|
+
subject: "test",
|
|
432
|
+
source: "test://source",
|
|
433
|
+
id: "123",
|
|
434
|
+
time: new Date().toISOString(),
|
|
435
|
+
datacontenttype: "application/json" as const,
|
|
436
|
+
data: {},
|
|
437
|
+
};
|
|
438
|
+
|
|
439
|
+
mockRequestBuilder.call.mockRejectedValue(new Error("Network error"));
|
|
440
|
+
|
|
441
|
+
await expect(
|
|
442
|
+
sendCloudEvent(cloudEvent, "https://api.example.com/events")
|
|
443
|
+
).rejects.toThrow("Network error");
|
|
444
|
+
});
|
|
445
|
+
|
|
446
|
+
it("should send event with all cloud event properties", async () => {
|
|
447
|
+
const cloudEvent: CloudEvent = {
|
|
448
|
+
specversion: "1.0" as const,
|
|
449
|
+
type: "user.created",
|
|
450
|
+
subject: "user/123",
|
|
451
|
+
source: "app://com.example/versions/1.0.0",
|
|
452
|
+
id: "uuid-123",
|
|
453
|
+
time: "2023-01-01T00:00:00.000Z",
|
|
454
|
+
datacontenttype: "application/json" as const,
|
|
455
|
+
data: {
|
|
456
|
+
userId: "123",
|
|
457
|
+
username: "testuser",
|
|
458
|
+
},
|
|
459
|
+
};
|
|
460
|
+
|
|
461
|
+
await sendCloudEvent(cloudEvent, "https://api.example.com/events");
|
|
462
|
+
|
|
463
|
+
expect(mockRequestBuilder.addBodyParams).toHaveBeenCalledWith(cloudEvent);
|
|
464
|
+
});
|
|
465
|
+
|
|
466
|
+
it("should handle different URL formats", async () => {
|
|
467
|
+
const cloudEvent: CloudEvent = {
|
|
468
|
+
specversion: "1.0" as const,
|
|
469
|
+
type: "test.event",
|
|
470
|
+
subject: "test",
|
|
471
|
+
source: "test://source",
|
|
472
|
+
id: "123",
|
|
473
|
+
time: new Date().toISOString(),
|
|
474
|
+
datacontenttype: "application/json" as const,
|
|
475
|
+
data: {},
|
|
476
|
+
};
|
|
477
|
+
|
|
478
|
+
const urls = [
|
|
479
|
+
"https://api.example.com/events",
|
|
480
|
+
"http://localhost:3000/events",
|
|
481
|
+
"https://api.example.com/v1/events?apiKey=123",
|
|
482
|
+
];
|
|
483
|
+
|
|
484
|
+
for (const url of urls) {
|
|
485
|
+
mockRequestBuilder.setUrl.mockClear();
|
|
486
|
+
await sendCloudEvent(cloudEvent, url);
|
|
487
|
+
expect(mockRequestBuilder.setUrl).toHaveBeenCalledWith(url);
|
|
488
|
+
}
|
|
489
|
+
});
|
|
490
|
+
});
|
|
491
|
+
|
|
492
|
+
describe("integration", () => {
|
|
493
|
+
it("should create and send cloud event together", async () => {
|
|
494
|
+
const mockResponse = { success: true, eventId: "evt-123" };
|
|
495
|
+
|
|
496
|
+
const mockRequestBuilder = {
|
|
497
|
+
setUrl: jest.fn().mockReturnThis(),
|
|
498
|
+
setMethod: jest.fn().mockReturnThis(),
|
|
499
|
+
addBodyParams: jest.fn().mockReturnThis(),
|
|
500
|
+
call: jest.fn().mockResolvedValue(mockResponse),
|
|
501
|
+
};
|
|
502
|
+
|
|
503
|
+
(
|
|
504
|
+
RequestBuilder as jest.MockedClass<typeof RequestBuilder>
|
|
505
|
+
).mockImplementation(() => mockRequestBuilder as any);
|
|
506
|
+
|
|
507
|
+
const event = await createCloudEvent({
|
|
508
|
+
type: "user.action",
|
|
509
|
+
subject: "button.click",
|
|
510
|
+
data: { buttonId: "submit" },
|
|
511
|
+
});
|
|
512
|
+
|
|
513
|
+
const response = await sendCloudEvent(
|
|
514
|
+
event,
|
|
515
|
+
"https://api.example.com/events"
|
|
516
|
+
);
|
|
517
|
+
|
|
518
|
+
expect(mockRequestBuilder.addBodyParams).toHaveBeenCalledWith(
|
|
519
|
+
expect.objectContaining({
|
|
520
|
+
type: "user.action",
|
|
521
|
+
subject: "button.click",
|
|
522
|
+
data: { buttonId: "submit" },
|
|
523
|
+
})
|
|
524
|
+
);
|
|
525
|
+
|
|
526
|
+
expect(response).toEqual(mockResponse);
|
|
527
|
+
});
|
|
528
|
+
});
|
|
529
|
+
});
|
|
@@ -3,6 +3,41 @@ import { RequestBuilder } from "@applicaster/zapp-pipes-v2-client";
|
|
|
3
3
|
import { sessionStorage } from "@applicaster/zapp-react-native-bridge/ZappStorage/SessionStorage";
|
|
4
4
|
import { getAppUrlScheme } from "@applicaster/zapp-react-native-utils/appDataUtils";
|
|
5
5
|
|
|
6
|
+
/**
|
|
7
|
+
* CloudEvents v1.0 compliant event object
|
|
8
|
+
* @see https://github.com/cloudevents/spec/blob/v1.0/spec.md
|
|
9
|
+
*/
|
|
10
|
+
export interface CloudEvent<T = any> {
|
|
11
|
+
/** CloudEvents specification version (always "1.0") */
|
|
12
|
+
specversion: "1.0";
|
|
13
|
+
/** Type of event (e.g., "user.action", "commerce.purchase") */
|
|
14
|
+
type: string;
|
|
15
|
+
/** Identifier for the entity that produced the event */
|
|
16
|
+
source: string;
|
|
17
|
+
/** Unique identifier for the event */
|
|
18
|
+
id: string;
|
|
19
|
+
/** Subject of the event in the context of the event producer */
|
|
20
|
+
subject: string;
|
|
21
|
+
/** Timestamp of when the event occurred (ISO 8601 format) */
|
|
22
|
+
time: string;
|
|
23
|
+
/** Content type of the data attribute (always "application/json") */
|
|
24
|
+
datacontenttype: "application/json";
|
|
25
|
+
/** Event payload */
|
|
26
|
+
data: T;
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
/**
|
|
30
|
+
* Parameters required to create a cloud event
|
|
31
|
+
*/
|
|
32
|
+
export interface CreateCloudEventParams<T = any> {
|
|
33
|
+
/** Type of event (e.g., "user.action", "commerce.purchase") */
|
|
34
|
+
type: string;
|
|
35
|
+
/** Subject of the event in the context of the event producer */
|
|
36
|
+
subject: string;
|
|
37
|
+
/** Event payload */
|
|
38
|
+
data: T;
|
|
39
|
+
}
|
|
40
|
+
|
|
6
41
|
let source: string | null = null;
|
|
7
42
|
|
|
8
43
|
const addSource = async () => {
|
|
@@ -26,7 +61,36 @@ const addSource = async () => {
|
|
|
26
61
|
return source;
|
|
27
62
|
};
|
|
28
63
|
|
|
29
|
-
|
|
64
|
+
/**
|
|
65
|
+
* Resets the cached CloudEvents source.
|
|
66
|
+
*
|
|
67
|
+
* @internal
|
|
68
|
+
* @testonly This function is intended for test use only and will only work in test environments.
|
|
69
|
+
*
|
|
70
|
+
* @remarks
|
|
71
|
+
* This function clears the cached source value, forcing the next call to `createCloudEvent`
|
|
72
|
+
* to regenerate the source from app metadata. It only operates when NODE_ENV is set to "test"
|
|
73
|
+
* to prevent accidental misuse in production code.
|
|
74
|
+
*
|
|
75
|
+
* @example
|
|
76
|
+
* ```typescript
|
|
77
|
+
* // In Jest tests
|
|
78
|
+
* beforeEach(() => {
|
|
79
|
+
* resetCloudEventsSource();
|
|
80
|
+
* });
|
|
81
|
+
* ```
|
|
82
|
+
*/
|
|
83
|
+
export const resetCloudEventsSource = (): void => {
|
|
84
|
+
if (process.env.NODE_ENV === "test") {
|
|
85
|
+
source = null;
|
|
86
|
+
}
|
|
87
|
+
};
|
|
88
|
+
|
|
89
|
+
export const createCloudEvent = async <T = any>({
|
|
90
|
+
type,
|
|
91
|
+
subject,
|
|
92
|
+
data,
|
|
93
|
+
}: CreateCloudEventParams<T>): Promise<CloudEvent<T>> => {
|
|
30
94
|
return {
|
|
31
95
|
specversion: "1.0",
|
|
32
96
|
type,
|