webfunny-event-node-sdk-public 1.0.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 (47) hide show
  1. package/README.md +302 -0
  2. package/dist/constants/index.d.ts +69 -0
  3. package/dist/constants/index.d.ts.map +1 -0
  4. package/dist/constants/index.js +76 -0
  5. package/dist/core/EventBuilder.d.ts +98 -0
  6. package/dist/core/EventBuilder.d.ts.map +1 -0
  7. package/dist/core/EventBuilder.js +222 -0
  8. package/dist/core/WebfunnyEvent.d.ts +89 -0
  9. package/dist/core/WebfunnyEvent.d.ts.map +1 -0
  10. package/dist/core/WebfunnyEvent.js +261 -0
  11. package/dist/core/WebfunnyEventClient.d.ts +123 -0
  12. package/dist/core/WebfunnyEventClient.d.ts.map +1 -0
  13. package/dist/core/WebfunnyEventClient.js +493 -0
  14. package/dist/core/WebfunnyEventException.d.ts +21 -0
  15. package/dist/core/WebfunnyEventException.d.ts.map +1 -0
  16. package/dist/core/WebfunnyEventException.js +50 -0
  17. package/dist/core/WebfunnyEventSDK.d.ts +118 -0
  18. package/dist/core/WebfunnyEventSDK.d.ts.map +1 -0
  19. package/dist/core/WebfunnyEventSDK.js +304 -0
  20. package/dist/core/index.d.ts +7 -0
  21. package/dist/core/index.d.ts.map +1 -0
  22. package/dist/core/index.js +14 -0
  23. package/dist/index.d.ts +54 -0
  24. package/dist/index.d.ts.map +1 -0
  25. package/dist/index.js +71 -0
  26. package/dist/types/index.d.ts +124 -0
  27. package/dist/types/index.d.ts.map +1 -0
  28. package/dist/types/index.js +7 -0
  29. package/dist/utils/Base64Util.d.ts +52 -0
  30. package/dist/utils/Base64Util.d.ts.map +1 -0
  31. package/dist/utils/Base64Util.js +136 -0
  32. package/dist/utils/EventQueue.d.ts +66 -0
  33. package/dist/utils/EventQueue.d.ts.map +1 -0
  34. package/dist/utils/EventQueue.js +181 -0
  35. package/dist/utils/HttpClient.d.ts +56 -0
  36. package/dist/utils/HttpClient.d.ts.map +1 -0
  37. package/dist/utils/HttpClient.js +146 -0
  38. package/dist/utils/IdGenerator.d.ts +32 -0
  39. package/dist/utils/IdGenerator.d.ts.map +1 -0
  40. package/dist/utils/IdGenerator.js +68 -0
  41. package/dist/utils/SignatureUtil.d.ts +48 -0
  42. package/dist/utils/SignatureUtil.d.ts.map +1 -0
  43. package/dist/utils/SignatureUtil.js +115 -0
  44. package/dist/utils/index.d.ts +6 -0
  45. package/dist/utils/index.d.ts.map +1 -0
  46. package/dist/utils/index.js +11 -0
  47. package/package.json +47 -0
@@ -0,0 +1,124 @@
1
+ /**
2
+ * Webfunny Event SDK 类型定义
3
+ * 对齐Java SDK的类型系统
4
+ */
5
+ /**
6
+ * SDK配置接口 - 对齐Java版WebfunnyEventConfig
7
+ */
8
+ export interface SDKConfig {
9
+ /** 服务端点配置 */
10
+ serverUrl: string;
11
+ /** 项目ID */
12
+ projectId: string;
13
+ /** 签名配置 */
14
+ signatureEnabled?: boolean;
15
+ /** 事件密钥 */
16
+ eventSecret?: string;
17
+ /** 批量上报大小 */
18
+ batchSize?: number;
19
+ /** 上报间隔(ms) */
20
+ flushIntervalMs?: number;
21
+ /** 最大队列大小 */
22
+ maxQueueSize?: number;
23
+ /** 最大重试次数 */
24
+ maxRetries?: number;
25
+ /** 重试间隔(ms) */
26
+ retryIntervalMs?: number;
27
+ /** 连接超时(ms) */
28
+ connectTimeoutMs?: number;
29
+ /** 读取超时(ms) */
30
+ socketTimeoutMs?: number;
31
+ /** 调试模式 */
32
+ debug?: boolean;
33
+ }
34
+ /**
35
+ * 事件数据接口 - 对齐Java版WebfunnyEvent
36
+ */
37
+ export interface EventData {
38
+ /** 项目ID */
39
+ projectId: string;
40
+ /** 点位ID */
41
+ pointId: string;
42
+ /** 用户ID */
43
+ weUserId?: string;
44
+ /** 客户Key */
45
+ weCustomerKey?: string;
46
+ /** IP地址 */
47
+ weIp?: string;
48
+ /** 国家 */
49
+ weCountry?: string;
50
+ /** 省份 */
51
+ weProvince?: string;
52
+ /** 城市 */
53
+ weCity?: string;
54
+ /** 发生时间戳 */
55
+ weHappenTime?: number;
56
+ /** 发生日期 */
57
+ weHappenDate?: string;
58
+ /** 发生小时 */
59
+ weHappenHour?: string;
60
+ /** 发生分钟 */
61
+ weHappenMinute?: string;
62
+ /** 页面路径 */
63
+ wePath?: string;
64
+ /** 首次访问日期 */
65
+ weFirstStepDay?: string;
66
+ /** 关联点位ID */
67
+ weRelationPointId?: string;
68
+ /** 关联字段 */
69
+ weRelationField?: string;
70
+ [key: string]: any;
71
+ }
72
+ /**
73
+ * 上报响应接口 - 对齐Java版UploadResponse
74
+ */
75
+ export interface UploadResponse {
76
+ /** 响应码 */
77
+ code: number;
78
+ /** 响应消息 */
79
+ message?: string;
80
+ /** 响应数据 */
81
+ data?: any;
82
+ /** 签名 */
83
+ signature?: string;
84
+ }
85
+ /**
86
+ * 上报选项接口
87
+ */
88
+ export interface ReportOptions {
89
+ /** 是否立即上报 */
90
+ immediate?: boolean;
91
+ /** 优先级 */
92
+ priority?: number;
93
+ }
94
+ /**
95
+ * 队列项接口
96
+ */
97
+ export interface QueueItem {
98
+ /** 事件数据 */
99
+ event: EventData;
100
+ /** 创建时间 */
101
+ createTime: number;
102
+ }
103
+ /**
104
+ * 回调函数类型
105
+ */
106
+ export type EventCallback = (event: EventData) => void;
107
+ export type ReportCallback = (response: UploadResponse) => void;
108
+ export type ErrorCallback = (error: Error) => void;
109
+ /**
110
+ * 生命周期钩子接口
111
+ */
112
+ export interface LifecycleHooks {
113
+ /** 初始化前 */
114
+ beforeInit?: () => void;
115
+ /** 初始化后 */
116
+ afterInit?: () => void;
117
+ /** 上报前 */
118
+ beforeReport?: (event: EventData) => EventData | boolean;
119
+ /** 上报后 */
120
+ afterReport?: (response: UploadResponse) => void;
121
+ /** 发生错误 */
122
+ onError?: (error: Error) => void;
123
+ }
124
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../src/types/index.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH;;GAEG;AACH,MAAM,WAAW,SAAS;IACxB,aAAa;IACb,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW;IACX,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW;IACX,gBAAgB,CAAC,EAAE,OAAO,CAAC;IAC3B,WAAW;IACX,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,aAAa;IACb,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,eAAe;IACf,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,aAAa;IACb,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,aAAa;IACb,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,eAAe;IACf,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,eAAe;IACf,gBAAgB,CAAC,EAAE,MAAM,CAAC;IAC1B,eAAe;IACf,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,WAAW;IACX,KAAK,CAAC,EAAE,OAAO,CAAC;CACjB;AAED;;GAEG;AACH,MAAM,WAAW,SAAS;IAExB,WAAW;IACX,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW;IACX,OAAO,EAAE,MAAM,CAAC;IAGhB,WAAW;IACX,QAAQ,CAAC,EAAE,MAAM,CAAC;IAClB,YAAY;IACZ,aAAa,CAAC,EAAE,MAAM,CAAC;IACvB,WAAW;IACX,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,SAAS;IACT,SAAS,CAAC,EAAE,MAAM,CAAC;IACnB,SAAS;IACT,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,SAAS;IACT,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,YAAY;IACZ,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,WAAW;IACX,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,WAAW;IACX,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,WAAW;IACX,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,WAAW;IACX,MAAM,CAAC,EAAE,MAAM,CAAC;IAChB,aAAa;IACb,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,aAAa;IACb,iBAAiB,CAAC,EAAE,MAAM,CAAC;IAC3B,WAAW;IACX,eAAe,CAAC,EAAE,MAAM,CAAC;IAGzB,CAAC,GAAG,EAAE,MAAM,GAAG,GAAG,CAAC;CACpB;AAED;;GAEG;AACH,MAAM,WAAW,cAAc;IAC7B,UAAU;IACV,IAAI,EAAE,MAAM,CAAC;IACb,WAAW;IACX,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,WAAW;IACX,IAAI,CAAC,EAAE,GAAG,CAAC;IACX,SAAS;IACT,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AAED;;GAEG;AACH,MAAM,WAAW,aAAa;IAC5B,aAAa;IACb,SAAS,CAAC,EAAE,OAAO,CAAC;IACpB,UAAU;IACV,QAAQ,CAAC,EAAE,MAAM,CAAC;CACnB;AAED;;GAEG;AACH,MAAM,WAAW,SAAS;IACxB,WAAW;IACX,KAAK,EAAE,SAAS,CAAC;IACjB,WAAW;IACX,UAAU,EAAE,MAAM,CAAC;CACpB;AAED;;GAEG;AACH,MAAM,MAAM,aAAa,GAAG,CAAC,KAAK,EAAE,SAAS,KAAK,IAAI,CAAC;AACvD,MAAM,MAAM,cAAc,GAAG,CAAC,QAAQ,EAAE,cAAc,KAAK,IAAI,CAAC;AAChE,MAAM,MAAM,aAAa,GAAG,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,CAAC;AAEnD;;GAEG;AACH,MAAM,WAAW,cAAc;IAC7B,WAAW;IACX,UAAU,CAAC,EAAE,MAAM,IAAI,CAAC;IACxB,WAAW;IACX,SAAS,CAAC,EAAE,MAAM,IAAI,CAAC;IACvB,UAAU;IACV,YAAY,CAAC,EAAE,CAAC,KAAK,EAAE,SAAS,KAAK,SAAS,GAAG,OAAO,CAAC;IACzD,UAAU;IACV,WAAW,CAAC,EAAE,CAAC,QAAQ,EAAE,cAAc,KAAK,IAAI,CAAC;IACjD,WAAW;IACX,OAAO,CAAC,EAAE,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,CAAC;CAClC"}
@@ -0,0 +1,7 @@
1
+ "use strict";
2
+ /**
3
+ * Webfunny Event SDK 类型定义
4
+ * 对齐Java SDK的类型系统
5
+ */
6
+ Object.defineProperty(exports, "__esModule", { value: true });
7
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdHlwZXMvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBOzs7R0FHRyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogV2ViZnVubnkgRXZlbnQgU0RLIOexu+Wei+WumuS5iVxuICog5a+56b2QSmF2YSBTREvnmoTnsbvlnovns7vnu59cbiAqL1xuXG4vKipcbiAqIFNES+mFjee9ruaOpeWPoyAtIOWvuem9kEphdmHniYhXZWJmdW5ueUV2ZW50Q29uZmlnXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU0RLQ29uZmlnIHtcbiAgLyoqIOacjeWKoeerr+eCuemFjee9riAqL1xuICBzZXJ2ZXJVcmw6IHN0cmluZztcbiAgLyoqIOmhueebrklEICovXG4gIHByb2plY3RJZDogc3RyaW5nO1xuICAvKiog562+5ZCN6YWN572uICovXG4gIHNpZ25hdHVyZUVuYWJsZWQ/OiBib29sZWFuO1xuICAvKiog5LqL5Lu25a+G6ZKlICovXG4gIGV2ZW50U2VjcmV0Pzogc3RyaW5nO1xuICAvKiog5om56YeP5LiK5oql5aSn5bCPICovXG4gIGJhdGNoU2l6ZT86IG51bWJlcjtcbiAgLyoqIOS4iuaKpemXtOmalChtcykgKi9cbiAgZmx1c2hJbnRlcnZhbE1zPzogbnVtYmVyO1xuICAvKiog5pyA5aSn6Zif5YiX5aSn5bCPICovXG4gIG1heFF1ZXVlU2l6ZT86IG51bWJlcjtcbiAgLyoqIOacgOWkp+mHjeivleasoeaVsCAqL1xuICBtYXhSZXRyaWVzPzogbnVtYmVyO1xuICAvKiog6YeN6K+V6Ze06ZqUKG1zKSAqL1xuICByZXRyeUludGVydmFsTXM/OiBudW1iZXI7XG4gIC8qKiDov57mjqXotoXml7YobXMpICovXG4gIGNvbm5lY3RUaW1lb3V0TXM/OiBudW1iZXI7XG4gIC8qKiDor7vlj5botoXml7YobXMpICovXG4gIHNvY2tldFRpbWVvdXRNcz86IG51bWJlcjtcbiAgLyoqIOiwg+ivleaooeW8jyAqL1xuICBkZWJ1Zz86IGJvb2xlYW47XG59XG5cbi8qKlxuICog5LqL5Lu25pWw5o2u5o6l5Y+jIC0g5a+56b2QSmF2YeeJiFdlYmZ1bm55RXZlbnRcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBFdmVudERhdGEge1xuICAvLyDlv4XloavlrZfmrrVcbiAgLyoqIOmhueebrklEICovXG4gIHByb2plY3RJZDogc3RyaW5nO1xuICAvKiog54K55L2NSUQgKi9cbiAgcG9pbnRJZDogc3RyaW5nO1xuICBcbiAgLy8g6YCa55So5a2X5q61ICh3ZeW8gOWktOeahOezu+e7n+Wtl+autSlcbiAgLyoqIOeUqOaIt0lEICovXG4gIHdlVXNlcklkPzogc3RyaW5nO1xuICAvKiog5a6i5oi3S2V5ICovXG4gIHdlQ3VzdG9tZXJLZXk/OiBzdHJpbmc7XG4gIC8qKiBJUOWcsOWdgCAqL1xuICB3ZUlwPzogc3RyaW5nO1xuICAvKiog5Zu95a62ICovXG4gIHdlQ291bnRyeT86IHN0cmluZztcbiAgLyoqIOecgeS7vSAqL1xuICB3ZVByb3ZpbmNlPzogc3RyaW5nO1xuICAvKiog5Z+O5biCICovXG4gIHdlQ2l0eT86IHN0cmluZztcbiAgLyoqIOWPkeeUn+aXtumXtOaIsyAqL1xuICB3ZUhhcHBlblRpbWU/OiBudW1iZXI7XG4gIC8qKiDlj5HnlJ/ml6XmnJ8gKi9cbiAgd2VIYXBwZW5EYXRlPzogc3RyaW5nO1xuICAvKiog5Y+R55Sf5bCP5pe2ICovXG4gIHdlSGFwcGVuSG91cj86IHN0cmluZztcbiAgLyoqIOWPkeeUn+WIhumSnyAqL1xuICB3ZUhhcHBlbk1pbnV0ZT86IHN0cmluZztcbiAgLyoqIOmhtemdoui3r+W+hCAqL1xuICB3ZVBhdGg/OiBzdHJpbmc7XG4gIC8qKiDpppbmrKHorr/pl67ml6XmnJ8gKi9cbiAgd2VGaXJzdFN0ZXBEYXk/OiBzdHJpbmc7XG4gIC8qKiDlhbPogZTngrnkvY1JRCAqL1xuICB3ZVJlbGF0aW9uUG9pbnRJZD86IHN0cmluZztcbiAgLyoqIOWFs+iBlOWtl+autSAqL1xuICB3ZVJlbGF0aW9uRmllbGQ/OiBzdHJpbmc7XG4gIFxuICAvLyDoh6rlrprkuYnlj4LmlbDlrZfmrrUgKOmdnndl5byA5aS055qE5Lia5Yqh5a2X5q61KVxuICBba2V5OiBzdHJpbmddOiBhbnk7XG59XG5cbi8qKlxuICog5LiK5oql5ZON5bqU5o6l5Y+jIC0g5a+56b2QSmF2YeeJiFVwbG9hZFJlc3BvbnNlXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgVXBsb2FkUmVzcG9uc2Uge1xuICAvKiog5ZON5bqU56CBICovXG4gIGNvZGU6IG51bWJlcjtcbiAgLyoqIOWTjeW6lOa2iOaBryAqL1xuICBtZXNzYWdlPzogc3RyaW5nO1xuICAvKiog5ZON5bqU5pWw5o2uICovXG4gIGRhdGE/OiBhbnk7XG4gIC8qKiDnrb7lkI0gKi9cbiAgc2lnbmF0dXJlPzogc3RyaW5nO1xufVxuXG4vKipcbiAqIOS4iuaKpemAiemhueaOpeWPo1xuICovXG5leHBvcnQgaW50ZXJmYWNlIFJlcG9ydE9wdGlvbnMge1xuICAvKiog5piv5ZCm56uL5Y2z5LiK5oqlICovXG4gIGltbWVkaWF0ZT86IGJvb2xlYW47XG4gIC8qKiDkvJjlhYjnuqcgKi9cbiAgcHJpb3JpdHk/OiBudW1iZXI7XG59XG5cbi8qKlxuICog6Zif5YiX6aG55o6l5Y+jXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgUXVldWVJdGVtIHtcbiAgLyoqIOS6i+S7tuaVsOaNriAqL1xuICBldmVudDogRXZlbnREYXRhO1xuICAvKiog5Yib5bu65pe26Ze0ICovXG4gIGNyZWF0ZVRpbWU6IG51bWJlcjtcbn1cblxuLyoqXG4gKiDlm57osIPlh73mlbDnsbvlnotcbiAqL1xuZXhwb3J0IHR5cGUgRXZlbnRDYWxsYmFjayA9IChldmVudDogRXZlbnREYXRhKSA9PiB2b2lkO1xuZXhwb3J0IHR5cGUgUmVwb3J0Q2FsbGJhY2sgPSAocmVzcG9uc2U6IFVwbG9hZFJlc3BvbnNlKSA9PiB2b2lkO1xuZXhwb3J0IHR5cGUgRXJyb3JDYWxsYmFjayA9IChlcnJvcjogRXJyb3IpID0+IHZvaWQ7XG5cbi8qKlxuICog55Sf5ZG95ZGo5pyf6ZKp5a2Q5o6l5Y+jXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgTGlmZWN5Y2xlSG9va3Mge1xuICAvKiog5Yid5aeL5YyW5YmNICovXG4gIGJlZm9yZUluaXQ/OiAoKSA9PiB2b2lkO1xuICAvKiog5Yid5aeL5YyW5ZCOICovXG4gIGFmdGVySW5pdD86ICgpID0+IHZvaWQ7XG4gIC8qKiDkuIrmiqXliY0gKi9cbiAgYmVmb3JlUmVwb3J0PzogKGV2ZW50OiBFdmVudERhdGEpID0+IEV2ZW50RGF0YSB8IGJvb2xlYW47XG4gIC8qKiDkuIrmiqXlkI4gKi9cbiAgYWZ0ZXJSZXBvcnQ/OiAocmVzcG9uc2U6IFVwbG9hZFJlc3BvbnNlKSA9PiB2b2lkO1xuICAvKiog5Y+R55Sf6ZSZ6K+vICovXG4gIG9uRXJyb3I/OiAoZXJyb3I6IEVycm9yKSA9PiB2b2lkO1xufVxuIl19
@@ -0,0 +1,52 @@
1
+ /**
2
+ * Base64编解码工具类
3
+ * 对齐Java版Base64Util
4
+ */
5
+ /**
6
+ * Base64工具类
7
+ */
8
+ export declare class Base64Util {
9
+ /**
10
+ * Base64编码 (UTF-8)
11
+ * 对齐Java版encode方法
12
+ * @param input 原始字符串
13
+ * @returns Base64编码后的字符串
14
+ */
15
+ static encode(input: string | null): string | null;
16
+ /**
17
+ * Base64解码 (UTF-8)
18
+ * 对齐Java版decode方法
19
+ * @param input Base64编码的字符串
20
+ * @returns 解码后的字符串
21
+ */
22
+ static decode(input: string | null): string | null;
23
+ /**
24
+ * URL安全的Base64编码
25
+ * @param input 原始字符串
26
+ * @returns URL安全的Base64编码
27
+ */
28
+ static encodeUrlSafe(input: string | null): string | null;
29
+ /**
30
+ * URL安全的Base64解码
31
+ * @param input URL安全的Base64编码
32
+ * @returns 解码后的字符串
33
+ */
34
+ static decodeUrlSafe(input: string | null): string | null;
35
+ /**
36
+ * 处理关联字段的特殊解码
37
+ * 对齐Java版decodeRelationField方法
38
+ * 格式: value1~wf~value2~wf~value3
39
+ * 每个value都是Base64编码的
40
+ * @param input 关联字段值
41
+ * @returns 解码后的值
42
+ */
43
+ static decodeRelationField(input: string | null): string | null;
44
+ /**
45
+ * 编码关联字段
46
+ * 对齐Java版encodeRelationField方法
47
+ * @param values 要编码的值数组
48
+ * @returns 编码后的关联字段值
49
+ */
50
+ static encodeRelationField(...values: (string | null)[]): string;
51
+ }
52
+ //# sourceMappingURL=Base64Util.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Base64Util.d.ts","sourceRoot":"","sources":["../../src/utils/Base64Util.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH;;GAEG;AACH,qBAAa,UAAU;IACrB;;;;;OAKG;WACW,MAAM,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI,GAAG,MAAM,GAAG,IAAI;IAWzD;;;;;OAKG;WACW,MAAM,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI,GAAG,MAAM,GAAG,IAAI;IAWzD;;;;OAIG;WACW,aAAa,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI,GAAG,MAAM,GAAG,IAAI;IAehE;;;;OAIG;WACW,aAAa,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI,GAAG,MAAM,GAAG,IAAI;IAkBhE;;;;;;;OAOG;WACW,mBAAmB,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI,GAAG,MAAM,GAAG,IAAI;IAqBtE;;;;;OAKG;WACW,mBAAmB,CAAC,GAAG,MAAM,EAAE,CAAC,MAAM,GAAG,IAAI,CAAC,EAAE,GAAG,MAAM;CAexE"}
@@ -0,0 +1,136 @@
1
+ "use strict";
2
+ /**
3
+ * Base64编解码工具类
4
+ * 对齐Java版Base64Util
5
+ */
6
+ Object.defineProperty(exports, "__esModule", { value: true });
7
+ exports.Base64Util = void 0;
8
+ /**
9
+ * Base64工具类
10
+ */
11
+ class Base64Util {
12
+ /**
13
+ * Base64编码 (UTF-8)
14
+ * 对齐Java版encode方法
15
+ * @param input 原始字符串
16
+ * @returns Base64编码后的字符串
17
+ */
18
+ static encode(input) {
19
+ if (input === null || input === undefined) {
20
+ return null;
21
+ }
22
+ try {
23
+ return Buffer.from(input, 'utf8').toString('base64');
24
+ }
25
+ catch (e) {
26
+ return null;
27
+ }
28
+ }
29
+ /**
30
+ * Base64解码 (UTF-8)
31
+ * 对齐Java版decode方法
32
+ * @param input Base64编码的字符串
33
+ * @returns 解码后的字符串
34
+ */
35
+ static decode(input) {
36
+ if (input === null || input === undefined) {
37
+ return null;
38
+ }
39
+ try {
40
+ return Buffer.from(input, 'base64').toString('utf8');
41
+ }
42
+ catch (e) {
43
+ return null;
44
+ }
45
+ }
46
+ /**
47
+ * URL安全的Base64编码
48
+ * @param input 原始字符串
49
+ * @returns URL安全的Base64编码
50
+ */
51
+ static encodeUrlSafe(input) {
52
+ if (input === null || input === undefined) {
53
+ return null;
54
+ }
55
+ try {
56
+ return Buffer.from(input, 'utf8')
57
+ .toString('base64')
58
+ .replace(/\+/g, '-')
59
+ .replace(/\//g, '_')
60
+ .replace(/=/g, '');
61
+ }
62
+ catch (e) {
63
+ return null;
64
+ }
65
+ }
66
+ /**
67
+ * URL安全的Base64解码
68
+ * @param input URL安全的Base64编码
69
+ * @returns 解码后的字符串
70
+ */
71
+ static decodeUrlSafe(input) {
72
+ if (input === null || input === undefined) {
73
+ return null;
74
+ }
75
+ try {
76
+ // 还原URL安全字符
77
+ let base64 = input.replace(/-/g, '+').replace(/_/g, '/');
78
+ // 补齐padding
79
+ const padding = 4 - (base64.length % 4);
80
+ if (padding !== 4) {
81
+ base64 += '='.repeat(padding);
82
+ }
83
+ return Buffer.from(base64, 'base64').toString('utf8');
84
+ }
85
+ catch (e) {
86
+ return null;
87
+ }
88
+ }
89
+ /**
90
+ * 处理关联字段的特殊解码
91
+ * 对齐Java版decodeRelationField方法
92
+ * 格式: value1~wf~value2~wf~value3
93
+ * 每个value都是Base64编码的
94
+ * @param input 关联字段值
95
+ * @returns 解码后的值
96
+ */
97
+ static decodeRelationField(input) {
98
+ if (input === null || input === undefined || input === '') {
99
+ return input;
100
+ }
101
+ const parts = input.split('~wf~');
102
+ const result = [];
103
+ for (let i = 0; i < parts.length; i++) {
104
+ try {
105
+ const decoded = this.decode(parts[i]);
106
+ result.push(decoded !== null ? decoded : parts[i]);
107
+ }
108
+ catch (e) {
109
+ // 解码失败,使用原始值
110
+ result.push(parts[i]);
111
+ }
112
+ }
113
+ return result.join('~wf~');
114
+ }
115
+ /**
116
+ * 编码关联字段
117
+ * 对齐Java版encodeRelationField方法
118
+ * @param values 要编码的值数组
119
+ * @returns 编码后的关联字段值
120
+ */
121
+ static encodeRelationField(...values) {
122
+ if (values === null || values.length === 0) {
123
+ return '';
124
+ }
125
+ const encodedValues = [];
126
+ for (const value of values) {
127
+ if (value !== null && value !== undefined) {
128
+ const encoded = this.encode(value);
129
+ encodedValues.push(encoded !== null ? encoded : value);
130
+ }
131
+ }
132
+ return encodedValues.join('~wf~');
133
+ }
134
+ }
135
+ exports.Base64Util = Base64Util;
136
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Base64Util.js","sourceRoot":"","sources":["../../src/utils/Base64Util.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Base64编解码工具类\n * 对齐Java版Base64Util\n */\n\n/**\n * Base64工具类\n */\nexport class Base64Util {\n  /**\n   * Base64编码 (UTF-8)\n   * 对齐Java版encode方法\n   * @param input 原始字符串\n   * @returns Base64编码后的字符串\n   */\n  public static encode(input: string | null): string | null {\n    if (input === null || input === undefined) {\n      return null;\n    }\n    try {\n      return Buffer.from(input, 'utf8').toString('base64');\n    } catch (e) {\n      return null;\n    }\n  }\n\n  /**\n   * Base64解码 (UTF-8)\n   * 对齐Java版decode方法\n   * @param input Base64编码的字符串\n   * @returns 解码后的字符串\n   */\n  public static decode(input: string | null): string | null {\n    if (input === null || input === undefined) {\n      return null;\n    }\n    try {\n      return Buffer.from(input, 'base64').toString('utf8');\n    } catch (e) {\n      return null;\n    }\n  }\n\n  /**\n   * URL安全的Base64编码\n   * @param input 原始字符串\n   * @returns URL安全的Base64编码\n   */\n  public static encodeUrlSafe(input: string | null): string | null {\n    if (input === null || input === undefined) {\n      return null;\n    }\n    try {\n      return Buffer.from(input, 'utf8')\n        .toString('base64')\n        .replace(/\\+/g, '-')\n        .replace(/\\//g, '_')\n        .replace(/=/g, '');\n    } catch (e) {\n      return null;\n    }\n  }\n\n  /**\n   * URL安全的Base64解码\n   * @param input URL安全的Base64编码\n   * @returns 解码后的字符串\n   */\n  public static decodeUrlSafe(input: string | null): string | null {\n    if (input === null || input === undefined) {\n      return null;\n    }\n    try {\n      // 还原URL安全字符\n      let base64 = input.replace(/-/g, '+').replace(/_/g, '/');\n      // 补齐padding\n      const padding = 4 - (base64.length % 4);\n      if (padding !== 4) {\n        base64 += '='.repeat(padding);\n      }\n      return Buffer.from(base64, 'base64').toString('utf8');\n    } catch (e) {\n      return null;\n    }\n  }\n\n  /**\n   * 处理关联字段的特殊解码\n   * 对齐Java版decodeRelationField方法\n   * 格式: value1~wf~value2~wf~value3\n   * 每个value都是Base64编码的\n   * @param input 关联字段值\n   * @returns 解码后的值\n   */\n  public static decodeRelationField(input: string | null): string | null {\n    if (input === null || input === undefined || input === '') {\n      return input;\n    }\n\n    const parts = input.split('~wf~');\n    const result: string[] = [];\n\n    for (let i = 0; i < parts.length; i++) {\n      try {\n        const decoded = this.decode(parts[i]);\n        result.push(decoded !== null ? decoded : parts[i]);\n      } catch (e) {\n        // 解码失败，使用原始值\n        result.push(parts[i]);\n      }\n    }\n\n    return result.join('~wf~');\n  }\n\n  /**\n   * 编码关联字段\n   * 对齐Java版encodeRelationField方法\n   * @param values 要编码的值数组\n   * @returns 编码后的关联字段值\n   */\n  public static encodeRelationField(...values: (string | null)[]): string {\n    if (values === null || values.length === 0) {\n      return '';\n    }\n\n    const encodedValues: string[] = [];\n    for (const value of values) {\n      if (value !== null && value !== undefined) {\n        const encoded = this.encode(value);\n        encodedValues.push(encoded !== null ? encoded : value);\n      }\n    }\n\n    return encodedValues.join('~wf~');\n  }\n}\n"]}
@@ -0,0 +1,66 @@
1
+ /**
2
+ * 事件队列管理器
3
+ * 类似Java SDK的队列设计
4
+ */
5
+ import { EventData, ReportOptions, ReportResponse } from '../types';
6
+ /**
7
+ * 事件队列管理器
8
+ */
9
+ export declare class EventQueue {
10
+ private queue;
11
+ private maxSize;
12
+ private batchSize;
13
+ private flushInterval;
14
+ private timer;
15
+ private isProcessing;
16
+ private httpClient;
17
+ /**
18
+ * 构造函数
19
+ */
20
+ constructor(serverUrl: string, maxSize?: number, batchSize?: number, flushInterval?: number);
21
+ /**
22
+ * 添加事件到队列
23
+ */
24
+ enqueue(event: EventData, options?: ReportOptions): boolean;
25
+ /**
26
+ * 获取队列大小
27
+ */
28
+ size(): number;
29
+ /**
30
+ * 清空队列
31
+ */
32
+ clear(): void;
33
+ /**
34
+ * 刷新队列(上报事件)
35
+ */
36
+ flush(): Promise<ReportResponse>;
37
+ /**
38
+ * 立即上报单条事件
39
+ */
40
+ reportImmediately(event: EventData): Promise<ReportResponse>;
41
+ /**
42
+ * 处理重试
43
+ */
44
+ private handleRetry;
45
+ /**
46
+ * 启动定时器
47
+ */
48
+ private startTimer;
49
+ /**
50
+ * 停止定时器
51
+ */
52
+ stopTimer(): void;
53
+ /**
54
+ * 销毁队列
55
+ */
56
+ destroy(): void;
57
+ /**
58
+ * 更新配置
59
+ */
60
+ updateConfig(config: {
61
+ maxSize?: number;
62
+ batchSize?: number;
63
+ flushInterval?: number;
64
+ }): void;
65
+ }
66
+ //# sourceMappingURL=EventQueue.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"EventQueue.d.ts","sourceRoot":"","sources":["../../src/utils/EventQueue.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAEH,OAAO,EAAa,SAAS,EAAE,aAAa,EAAE,cAAc,EAAE,MAAM,UAAU,CAAC;AAI/E;;GAEG;AACH,qBAAa,UAAU;IACrB,OAAO,CAAC,KAAK,CAAmB;IAChC,OAAO,CAAC,OAAO,CAAS;IACxB,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,aAAa,CAAS;IAC9B,OAAO,CAAC,KAAK,CAA+C;IAC5D,OAAO,CAAC,YAAY,CAAkB;IACtC,OAAO,CAAC,UAAU,CAAa;IAE/B;;OAEG;gBAED,SAAS,EAAE,MAAM,EACjB,OAAO,GAAE,MAA6C,EACtD,SAAS,GAAE,MAAyC,EACpD,aAAa,GAAE,MAA6C;IAS9D;;OAEG;IACI,OAAO,CAAC,KAAK,EAAE,SAAS,EAAE,OAAO,GAAE,aAAkB,GAAG,OAAO;IAwCtE;;OAEG;IACI,IAAI,IAAI,MAAM;IAIrB;;OAEG;IACI,KAAK,IAAI,IAAI;IAIpB;;OAEG;IACU,KAAK,IAAI,OAAO,CAAC,cAAc,CAAC;IAoC7C;;OAEG;IACU,iBAAiB,CAAC,KAAK,EAAE,SAAS,GAAG,OAAO,CAAC,cAAc,CAAC;IAIzE;;OAEG;IACH,OAAO,CAAC,WAAW;IAYnB;;OAEG;IACH,OAAO,CAAC,UAAU;IAYlB;;OAEG;IACI,SAAS,IAAI,IAAI;IAOxB;;OAEG;IACI,OAAO,IAAI,IAAI;IAKtB;;OAEG;IACI,YAAY,CAAC,MAAM,EAAE;QAC1B,OAAO,CAAC,EAAE,MAAM,CAAC;QACjB,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,aAAa,CAAC,EAAE,MAAM,CAAC;KACxB,GAAG,IAAI;CAYT"}
@@ -0,0 +1,181 @@
1
+ "use strict";
2
+ /**
3
+ * 事件队列管理器
4
+ * 类似Java SDK的队列设计
5
+ */
6
+ Object.defineProperty(exports, "__esModule", { value: true });
7
+ exports.EventQueue = void 0;
8
+ const constants_1 = require("../constants");
9
+ const HttpClient_1 = require("./HttpClient");
10
+ /**
11
+ * 事件队列管理器
12
+ */
13
+ class EventQueue {
14
+ /**
15
+ * 构造函数
16
+ */
17
+ constructor(serverUrl, maxSize = constants_1.DefaultConfig.DEFAULT_MAX_QUEUE_SIZE, batchSize = constants_1.DefaultConfig.DEFAULT_BATCH_SIZE, flushInterval = constants_1.DefaultConfig.DEFAULT_FLUSH_INTERVAL) {
18
+ this.queue = [];
19
+ this.timer = null;
20
+ this.isProcessing = false;
21
+ this.maxSize = maxSize;
22
+ this.batchSize = batchSize;
23
+ this.flushInterval = flushInterval;
24
+ this.httpClient = HttpClient_1.HttpClient.getInstance(serverUrl);
25
+ this.startTimer();
26
+ }
27
+ /**
28
+ * 添加事件到队列
29
+ */
30
+ enqueue(event, options = {}) {
31
+ // 检查队列是否已满
32
+ if (this.queue.length >= this.maxSize) {
33
+ console.warn(constants_1.ErrorMessage.QUEUE_IS_FULL);
34
+ return false;
35
+ }
36
+ const item = {
37
+ event,
38
+ options: {
39
+ immediate: options.immediate || false,
40
+ priority: options.priority || 0,
41
+ retryTimes: options.retryTimes || constants_1.DefaultConfig.DEFAULT_RETRY_TIMES,
42
+ timeout: options.timeout || constants_1.DefaultConfig.DEFAULT_TIMEOUT,
43
+ },
44
+ retryCount: 0,
45
+ createTime: Date.now(),
46
+ };
47
+ // 根据优先级插入队列
48
+ if (item.options.priority && item.options.priority > 0) {
49
+ // 高优先级插入队首
50
+ const insertIndex = this.queue.findIndex(i => (i.options.priority || 0) < item.options.priority);
51
+ if (insertIndex === -1) {
52
+ this.queue.push(item);
53
+ }
54
+ else {
55
+ this.queue.splice(insertIndex, 0, item);
56
+ }
57
+ }
58
+ else {
59
+ this.queue.push(item);
60
+ }
61
+ // 如果是立即上报,触发刷新
62
+ if (item.options.immediate) {
63
+ this.flush();
64
+ }
65
+ return true;
66
+ }
67
+ /**
68
+ * 获取队列大小
69
+ */
70
+ size() {
71
+ return this.queue.length;
72
+ }
73
+ /**
74
+ * 清空队列
75
+ */
76
+ clear() {
77
+ this.queue = [];
78
+ }
79
+ /**
80
+ * 刷新队列(上报事件)
81
+ */
82
+ async flush() {
83
+ if (this.isProcessing || this.queue.length === 0) {
84
+ return {
85
+ success: true,
86
+ code: constants_1.ErrorCode.SUCCESS,
87
+ message: '队列为空或正在处理中',
88
+ };
89
+ }
90
+ this.isProcessing = true;
91
+ try {
92
+ // 取出批次数据
93
+ const batch = this.queue.splice(0, this.batchSize);
94
+ const events = batch.map(item => item.event);
95
+ // 批量上报
96
+ const response = await this.httpClient.batchReport(events);
97
+ // 处理失败的重试
98
+ if (!response.success) {
99
+ this.handleRetry(batch);
100
+ }
101
+ return response;
102
+ }
103
+ catch (error) {
104
+ return {
105
+ success: false,
106
+ code: constants_1.ErrorCode.REPORT_FAILED,
107
+ message: error instanceof Error ? error.message : '上报失败',
108
+ };
109
+ }
110
+ finally {
111
+ this.isProcessing = false;
112
+ }
113
+ }
114
+ /**
115
+ * 立即上报单条事件
116
+ */
117
+ async reportImmediately(event) {
118
+ return await this.httpClient.singleReport(event);
119
+ }
120
+ /**
121
+ * 处理重试
122
+ */
123
+ handleRetry(failedItems) {
124
+ for (const item of failedItems) {
125
+ if (item.retryCount < (item.options.retryTimes || constants_1.DefaultConfig.DEFAULT_RETRY_TIMES)) {
126
+ item.retryCount++;
127
+ // 重新加入队列尾部
128
+ this.queue.push(item);
129
+ }
130
+ else {
131
+ console.warn(`事件 ${item.event.eventId} 重试次数已用完,丢弃`);
132
+ }
133
+ }
134
+ }
135
+ /**
136
+ * 启动定时器
137
+ */
138
+ startTimer() {
139
+ if (this.timer) {
140
+ clearInterval(this.timer);
141
+ }
142
+ this.timer = setInterval(() => {
143
+ if (this.queue.length > 0) {
144
+ this.flush();
145
+ }
146
+ }, this.flushInterval);
147
+ }
148
+ /**
149
+ * 停止定时器
150
+ */
151
+ stopTimer() {
152
+ if (this.timer) {
153
+ clearInterval(this.timer);
154
+ this.timer = null;
155
+ }
156
+ }
157
+ /**
158
+ * 销毁队列
159
+ */
160
+ destroy() {
161
+ this.stopTimer();
162
+ this.flush();
163
+ }
164
+ /**
165
+ * 更新配置
166
+ */
167
+ updateConfig(config) {
168
+ if (config.maxSize !== undefined) {
169
+ this.maxSize = config.maxSize;
170
+ }
171
+ if (config.batchSize !== undefined) {
172
+ this.batchSize = config.batchSize;
173
+ }
174
+ if (config.flushInterval !== undefined) {
175
+ this.flushInterval = config.flushInterval;
176
+ this.startTimer();
177
+ }
178
+ }
179
+ }
180
+ exports.EventQueue = EventQueue;
181
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"EventQueue.js","sourceRoot":"","sources":["../../src/utils/EventQueue.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * 事件队列管理器\n * 类似Java SDK的队列设计\n */\n\nimport { QueueItem, EventData, ReportOptions, ReportResponse } from '../types';\nimport { DefaultConfig, ErrorCode, ErrorMessage } from '../constants';\nimport { HttpClient } from './HttpClient';\n\n/**\n * 事件队列管理器\n */\nexport class EventQueue {\n  private queue: QueueItem[] = [];\n  private maxSize: number;\n  private batchSize: number;\n  private flushInterval: number;\n  private timer: ReturnType<typeof setInterval> | null = null;\n  private isProcessing: boolean = false;\n  private httpClient: HttpClient;\n\n  /**\n   * 构造函数\n   */\n  constructor(\n    serverUrl: string,\n    maxSize: number = DefaultConfig.DEFAULT_MAX_QUEUE_SIZE,\n    batchSize: number = DefaultConfig.DEFAULT_BATCH_SIZE,\n    flushInterval: number = DefaultConfig.DEFAULT_FLUSH_INTERVAL\n  ) {\n    this.maxSize = maxSize;\n    this.batchSize = batchSize;\n    this.flushInterval = flushInterval;\n    this.httpClient = HttpClient.getInstance(serverUrl);\n    this.startTimer();\n  }\n\n  /**\n   * 添加事件到队列\n   */\n  public enqueue(event: EventData, options: ReportOptions = {}): boolean {\n    // 检查队列是否已满\n    if (this.queue.length >= this.maxSize) {\n      console.warn(ErrorMessage.QUEUE_IS_FULL);\n      return false;\n    }\n\n    const item: QueueItem = {\n      event,\n      options: {\n        immediate: options.immediate || false,\n        priority: options.priority || 0,\n        retryTimes: options.retryTimes || DefaultConfig.DEFAULT_RETRY_TIMES,\n        timeout: options.timeout || DefaultConfig.DEFAULT_TIMEOUT,\n      },\n      retryCount: 0,\n      createTime: Date.now(),\n    };\n\n    // 根据优先级插入队列\n    if (item.options.priority && item.options.priority > 0) {\n      // 高优先级插入队首\n      const insertIndex = this.queue.findIndex(i => (i.options.priority || 0) < item.options.priority!);\n      if (insertIndex === -1) {\n        this.queue.push(item);\n      } else {\n        this.queue.splice(insertIndex, 0, item);\n      }\n    } else {\n      this.queue.push(item);\n    }\n\n    // 如果是立即上报，触发刷新\n    if (item.options.immediate) {\n      this.flush();\n    }\n\n    return true;\n  }\n\n  /**\n   * 获取队列大小\n   */\n  public size(): number {\n    return this.queue.length;\n  }\n\n  /**\n   * 清空队列\n   */\n  public clear(): void {\n    this.queue = [];\n  }\n\n  /**\n   * 刷新队列（上报事件）\n   */\n  public async flush(): Promise<ReportResponse> {\n    if (this.isProcessing || this.queue.length === 0) {\n      return {\n        success: true,\n        code: ErrorCode.SUCCESS,\n        message: '队列为空或正在处理中',\n      };\n    }\n\n    this.isProcessing = true;\n\n    try {\n      // 取出批次数据\n      const batch = this.queue.splice(0, this.batchSize);\n      const events = batch.map(item => item.event);\n\n      // 批量上报\n      const response = await this.httpClient.batchReport(events);\n\n      // 处理失败的重试\n      if (!response.success) {\n        this.handleRetry(batch);\n      }\n\n      return response;\n    } catch (error) {\n      return {\n        success: false,\n        code: ErrorCode.REPORT_FAILED,\n        message: error instanceof Error ? error.message : '上报失败',\n      };\n    } finally {\n      this.isProcessing = false;\n    }\n  }\n\n  /**\n   * 立即上报单条事件\n   */\n  public async reportImmediately(event: EventData): Promise<ReportResponse> {\n    return await this.httpClient.singleReport(event);\n  }\n\n  /**\n   * 处理重试\n   */\n  private handleRetry(failedItems: QueueItem[]): void {\n    for (const item of failedItems) {\n      if (item.retryCount < (item.options.retryTimes || DefaultConfig.DEFAULT_RETRY_TIMES)) {\n        item.retryCount++;\n        // 重新加入队列尾部\n        this.queue.push(item);\n      } else {\n        console.warn(`事件 ${item.event.eventId} 重试次数已用完，丢弃`);\n      }\n    }\n  }\n\n  /**\n   * 启动定时器\n   */\n  private startTimer(): void {\n    if (this.timer) {\n      clearInterval(this.timer);\n    }\n\n    this.timer = setInterval(() => {\n      if (this.queue.length > 0) {\n        this.flush();\n      }\n    }, this.flushInterval);\n  }\n\n  /**\n   * 停止定时器\n   */\n  public stopTimer(): void {\n    if (this.timer) {\n      clearInterval(this.timer);\n      this.timer = null;\n    }\n  }\n\n  /**\n   * 销毁队列\n   */\n  public destroy(): void {\n    this.stopTimer();\n    this.flush();\n  }\n\n  /**\n   * 更新配置\n   */\n  public updateConfig(config: {\n    maxSize?: number;\n    batchSize?: number;\n    flushInterval?: number;\n  }): void {\n    if (config.maxSize !== undefined) {\n      this.maxSize = config.maxSize;\n    }\n    if (config.batchSize !== undefined) {\n      this.batchSize = config.batchSize;\n    }\n    if (config.flushInterval !== undefined) {\n      this.flushInterval = config.flushInterval;\n      this.startTimer();\n    }\n  }\n}\n"]}