@formo/analytics 1.13.4-alpha.1 → 1.13.4-alpha.10

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 (108) hide show
  1. package/CONTRIBUTING.md +1 -1
  2. package/dist/cjs/src/FormoAnalytics.d.ts +6 -8
  3. package/dist/cjs/src/FormoAnalytics.d.ts.map +1 -1
  4. package/dist/cjs/src/FormoAnalytics.js +118 -113
  5. package/dist/cjs/src/FormoAnalytics.js.map +1 -1
  6. package/dist/cjs/src/FormoAnalyticsProvider.d.ts +2 -3
  7. package/dist/cjs/src/FormoAnalyticsProvider.d.ts.map +1 -1
  8. package/dist/cjs/src/FormoAnalyticsProvider.js +2 -33
  9. package/dist/cjs/src/FormoAnalyticsProvider.js.map +1 -1
  10. package/dist/cjs/src/constants/base.d.ts +3 -0
  11. package/dist/cjs/src/constants/base.d.ts.map +1 -0
  12. package/dist/cjs/src/constants/base.js +6 -0
  13. package/dist/cjs/src/constants/base.js.map +1 -0
  14. package/dist/cjs/src/constants/index.d.ts +3 -2
  15. package/dist/cjs/src/constants/index.d.ts.map +1 -1
  16. package/dist/cjs/src/constants/index.js +1 -0
  17. package/dist/cjs/src/constants/index.js.map +1 -1
  18. package/dist/cjs/src/lib/index.d.ts +3 -0
  19. package/dist/cjs/src/lib/index.d.ts.map +1 -0
  20. package/dist/cjs/src/lib/index.js +24 -0
  21. package/dist/cjs/src/lib/index.js.map +1 -0
  22. package/dist/cjs/src/lib/session-storage.d.ts +11 -0
  23. package/dist/cjs/src/lib/session-storage.d.ts.map +1 -0
  24. package/dist/cjs/src/lib/session-storage.js +52 -0
  25. package/dist/cjs/src/lib/session-storage.js.map +1 -0
  26. package/dist/cjs/src/lib/utils.d.ts +1 -0
  27. package/dist/cjs/src/lib/utils.d.ts.map +1 -1
  28. package/dist/cjs/src/lib/utils.js +5 -0
  29. package/dist/cjs/src/lib/utils.js.map +1 -1
  30. package/dist/cjs/src/types/base.d.ts +2 -1
  31. package/dist/cjs/src/types/base.d.ts.map +1 -1
  32. package/dist/cjs/test/lib.spec.js +3 -3
  33. package/dist/cjs/test/lib.spec.js.map +1 -1
  34. package/dist/cjs/tsconfig.tsbuildinfo +1 -1
  35. package/dist/esm/src/FormoAnalytics.d.ts +168 -0
  36. package/dist/esm/src/FormoAnalytics.d.ts.map +1 -0
  37. package/dist/esm/src/FormoAnalytics.js +947 -0
  38. package/dist/esm/src/FormoAnalytics.js.map +1 -0
  39. package/dist/esm/src/FormoAnalyticsProvider.d.ts +6 -0
  40. package/dist/esm/src/FormoAnalyticsProvider.d.ts.map +1 -0
  41. package/dist/esm/src/FormoAnalyticsProvider.js +123 -0
  42. package/dist/esm/src/FormoAnalyticsProvider.js.map +1 -0
  43. package/dist/esm/src/constants/base.d.ts +3 -0
  44. package/dist/esm/src/constants/base.d.ts.map +1 -0
  45. package/dist/esm/src/constants/base.js +3 -0
  46. package/dist/esm/src/constants/base.js.map +1 -0
  47. package/dist/esm/src/constants/config.d.ts +602 -0
  48. package/dist/esm/src/constants/config.d.ts.map +1 -0
  49. package/dist/esm/src/constants/config.js +654 -0
  50. package/dist/esm/src/constants/config.js.map +1 -0
  51. package/dist/esm/src/constants/events.d.ts +10 -0
  52. package/dist/esm/src/constants/events.d.ts.map +1 -0
  53. package/dist/esm/src/constants/events.js +11 -0
  54. package/dist/esm/src/constants/events.js.map +1 -0
  55. package/dist/esm/src/constants/index.d.ts +4 -0
  56. package/dist/esm/src/constants/index.d.ts.map +1 -0
  57. package/dist/esm/src/constants/index.js +4 -0
  58. package/dist/esm/src/constants/index.js.map +1 -0
  59. package/dist/esm/src/index.d.ts +4 -0
  60. package/dist/esm/src/index.d.ts.map +1 -0
  61. package/dist/esm/src/index.js +4 -0
  62. package/dist/esm/src/index.js.map +1 -0
  63. package/dist/esm/src/lib/index.d.ts +3 -0
  64. package/dist/esm/src/lib/index.d.ts.map +1 -0
  65. package/dist/esm/src/lib/index.js +3 -0
  66. package/dist/esm/src/lib/index.js.map +1 -0
  67. package/dist/esm/src/lib/session-storage.d.ts +11 -0
  68. package/dist/esm/src/lib/session-storage.d.ts.map +1 -0
  69. package/dist/esm/src/lib/session-storage.js +49 -0
  70. package/dist/esm/src/lib/session-storage.js.map +1 -0
  71. package/dist/esm/src/lib/utils.d.ts +3 -0
  72. package/dist/esm/src/lib/utils.d.ts.map +1 -0
  73. package/dist/esm/src/lib/utils.js +31 -0
  74. package/dist/esm/src/lib/utils.js.map +1 -0
  75. package/dist/esm/src/types/base.d.ts +18 -0
  76. package/dist/esm/src/types/base.d.ts.map +1 -0
  77. package/dist/esm/src/types/base.js +2 -0
  78. package/dist/esm/src/types/base.js.map +1 -0
  79. package/dist/esm/src/types/events.d.ts +11 -0
  80. package/dist/esm/src/types/events.d.ts.map +1 -0
  81. package/dist/esm/src/types/events.js +13 -0
  82. package/dist/esm/src/types/events.js.map +1 -0
  83. package/dist/esm/src/types/index.d.ts +4 -0
  84. package/dist/esm/src/types/index.d.ts.map +1 -0
  85. package/dist/esm/src/types/index.js +4 -0
  86. package/dist/esm/src/types/index.js.map +1 -0
  87. package/dist/esm/src/types/provider.d.ts +15 -0
  88. package/dist/esm/src/types/provider.d.ts.map +1 -0
  89. package/dist/esm/src/types/provider.js +2 -0
  90. package/dist/esm/src/types/provider.js.map +1 -0
  91. package/dist/esm/test/lib.spec.d.ts +2 -0
  92. package/dist/esm/test/lib.spec.d.ts.map +1 -0
  93. package/dist/esm/test/lib.spec.js +23 -0
  94. package/dist/esm/test/lib.spec.js.map +1 -0
  95. package/dist/esm/tsconfig.tsbuildinfo +1 -0
  96. package/dist/index.umd.min.js +3 -0
  97. package/dist/index.umd.min.js.LICENSE.txt +19 -0
  98. package/dist/index.umd.min.js.map +1 -0
  99. package/package.json +4 -3
  100. package/src/FormoAnalytics.ts +60 -17
  101. package/src/FormoAnalyticsProvider.tsx +2 -8
  102. package/src/constants/base.ts +2 -0
  103. package/src/constants/index.ts +3 -2
  104. package/src/lib/index.ts +2 -0
  105. package/src/lib/session-storage.ts +53 -0
  106. package/src/lib/utils.ts +5 -0
  107. package/src/types/base.ts +5 -3
  108. package/test/lib.spec.ts +1 -1
@@ -0,0 +1,947 @@
1
+ var __assign = (this && this.__assign) || function () {
2
+ __assign = Object.assign || function(t) {
3
+ for (var s, i = 1, n = arguments.length; i < n; i++) {
4
+ s = arguments[i];
5
+ for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))
6
+ t[p] = s[p];
7
+ }
8
+ return t;
9
+ };
10
+ return __assign.apply(this, arguments);
11
+ };
12
+ var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
13
+ function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
14
+ return new (P || (P = Promise))(function (resolve, reject) {
15
+ function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
16
+ function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
17
+ function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
18
+ step((generator = generator.apply(thisArg, _arguments || [])).next());
19
+ });
20
+ };
21
+ var __generator = (this && this.__generator) || function (thisArg, body) {
22
+ var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g = Object.create((typeof Iterator === "function" ? Iterator : Object).prototype);
23
+ return g.next = verb(0), g["throw"] = verb(1), g["return"] = verb(2), typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
24
+ function verb(n) { return function (v) { return step([n, v]); }; }
25
+ function step(op) {
26
+ if (f) throw new TypeError("Generator is already executing.");
27
+ while (g && (g = 0, op[0] && (_ = 0)), _) try {
28
+ if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
29
+ if (y = 0, t) op = [op[0] & 2, t.value];
30
+ switch (op[0]) {
31
+ case 0: case 1: t = op; break;
32
+ case 4: _.label++; return { value: op[1], done: false };
33
+ case 5: _.label++; y = op[1]; op = [0]; continue;
34
+ case 7: op = _.ops.pop(); _.trys.pop(); continue;
35
+ default:
36
+ if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
37
+ if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
38
+ if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
39
+ if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
40
+ if (t[2]) _.ops.pop();
41
+ _.trys.pop(); continue;
42
+ }
43
+ op = body.call(thisArg, _);
44
+ } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
45
+ if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
46
+ }
47
+ };
48
+ var __spreadArray = (this && this.__spreadArray) || function (to, from, pack) {
49
+ if (pack || arguments.length === 2) for (var i = 0, l = from.length, ar; i < l; i++) {
50
+ if (ar || !(i in from)) {
51
+ if (!ar) ar = Array.prototype.slice.call(from, 0, i);
52
+ ar[i] = from[i];
53
+ }
54
+ }
55
+ return to.concat(ar || Array.prototype.slice.call(from));
56
+ };
57
+ import axios from "axios";
58
+ import { createStore } from "mipd";
59
+ import { COUNTRY_LIST, CURRENT_URL_KEY, EVENTS_API_URL, Event, } from "./constants";
60
+ import { SignatureStatus, TransactionStatus, } from "./types";
61
+ import { session, isLocalhost, toSnakeCase } from "./lib";
62
+ import { SESSION_IDENTIFIED_KEY } from "./constants";
63
+ var FormoAnalytics = /** @class */ (function () {
64
+ function FormoAnalytics(apiKey, options) {
65
+ if (options === void 0) { options = {}; }
66
+ this.apiKey = apiKey;
67
+ this.options = options;
68
+ this._providerListeners = {};
69
+ this.config = {
70
+ apiKey: apiKey,
71
+ trackLocalhost: options.trackLocalhost,
72
+ };
73
+ this.session = new FormoAnalyticsSession();
74
+ // TODO: replace with eip6963
75
+ var provider = options.provider || (window === null || window === void 0 ? void 0 : window.ethereum);
76
+ if (provider) {
77
+ this.trackProvider(provider);
78
+ }
79
+ this.trackFirstPageHit();
80
+ this.trackPageHits();
81
+ }
82
+ FormoAnalytics.init = function (apiKey, options) {
83
+ return __awaiter(this, void 0, void 0, function () {
84
+ var analytics, providers;
85
+ return __generator(this, function (_a) {
86
+ switch (_a.label) {
87
+ case 0:
88
+ analytics = new FormoAnalytics(apiKey, options);
89
+ return [4 /*yield*/, analytics.getProviders()];
90
+ case 1:
91
+ providers = _a.sent();
92
+ return [4 /*yield*/, analytics.identifyAll(providers)];
93
+ case 2:
94
+ _a.sent();
95
+ return [2 /*return*/, analytics];
96
+ }
97
+ });
98
+ });
99
+ };
100
+ /*
101
+ Public SDK functions
102
+ */
103
+ /**
104
+ * Emits a page visit event with the current URL information, fire on page change.
105
+ * @returns {Promise<void>}
106
+ */
107
+ FormoAnalytics.prototype.page = function () {
108
+ return __awaiter(this, void 0, void 0, function () {
109
+ return __generator(this, function (_a) {
110
+ switch (_a.label) {
111
+ case 0: return [4 /*yield*/, this.trackPageHit()];
112
+ case 1:
113
+ _a.sent();
114
+ return [2 /*return*/];
115
+ }
116
+ });
117
+ });
118
+ };
119
+ /**
120
+ * Emits a wallet connect event.
121
+ * @param {ChainID} params.chainId
122
+ * @param {Address} params.address
123
+ * @throws {Error} If chainId or address is empty
124
+ * @returns {Promise<void>}
125
+ */
126
+ FormoAnalytics.prototype.connect = function (_a) {
127
+ return __awaiter(this, arguments, void 0, function (_b) {
128
+ var chainId = _b.chainId, address = _b.address;
129
+ return __generator(this, function (_c) {
130
+ switch (_c.label) {
131
+ case 0:
132
+ if (!chainId) {
133
+ throw new Error("FormoAnalytics::connect: chain ID cannot be empty");
134
+ }
135
+ if (!address) {
136
+ throw new Error("FormoAnalytics::connect: address cannot be empty");
137
+ }
138
+ this.currentChainId = chainId;
139
+ this.currentConnectedAddress = address;
140
+ return [4 /*yield*/, this.trackEvent(Event.CONNECT, {
141
+ chainId: chainId,
142
+ address: address,
143
+ })];
144
+ case 1:
145
+ _c.sent();
146
+ return [2 /*return*/];
147
+ }
148
+ });
149
+ });
150
+ };
151
+ /**
152
+ * Emits a wallet disconnect event.
153
+ * @param {ChainID} params.chainId
154
+ * @param {Address} params.address
155
+ * @returns {Promise<void>}
156
+ */
157
+ FormoAnalytics.prototype.disconnect = function (params) {
158
+ return __awaiter(this, void 0, void 0, function () {
159
+ var address, chainId;
160
+ return __generator(this, function (_a) {
161
+ switch (_a.label) {
162
+ case 0:
163
+ address = (params === null || params === void 0 ? void 0 : params.address) || this.currentConnectedAddress;
164
+ chainId = (params === null || params === void 0 ? void 0 : params.chainId) || this.currentChainId;
165
+ return [4 /*yield*/, this.handleDisconnect(chainId, address)];
166
+ case 1:
167
+ _a.sent();
168
+ return [2 /*return*/];
169
+ }
170
+ });
171
+ });
172
+ };
173
+ /**
174
+ * Emits a chain network change event.
175
+ * @param {ChainID} params.chainId
176
+ * @param {Address} params.address
177
+ * @throws {Error} If chainId is empty, zero, or not a valid number
178
+ * @throws {Error} If no address is provided and no previous address is recorded
179
+ * @returns {Promise<void>}
180
+ */
181
+ FormoAnalytics.prototype.chain = function (_a) {
182
+ return __awaiter(this, arguments, void 0, function (_b) {
183
+ var chainId = _b.chainId, address = _b.address;
184
+ return __generator(this, function (_c) {
185
+ switch (_c.label) {
186
+ case 0:
187
+ if (!chainId || Number(chainId) === 0) {
188
+ throw new Error("FormoAnalytics::chain: chainId cannot be empty or 0");
189
+ }
190
+ if (isNaN(Number(chainId))) {
191
+ throw new Error("FormoAnalytics::chain: chainId must be a valid decimal number");
192
+ }
193
+ if (!address && !this.currentConnectedAddress) {
194
+ throw new Error("FormoAnalytics::chain: address was empty and no previous address has been recorded");
195
+ }
196
+ this.currentChainId = chainId;
197
+ return [4 /*yield*/, this.trackEvent(Event.CHAIN_CHANGED, {
198
+ chainId: chainId,
199
+ address: address || this.currentConnectedAddress,
200
+ })];
201
+ case 1:
202
+ _c.sent();
203
+ return [2 /*return*/];
204
+ }
205
+ });
206
+ });
207
+ };
208
+ /**
209
+ * Emits a signature event.
210
+ * @param {SignatureStatus} params.status - requested, confirmed, rejected
211
+ * @param {ChainID} params.chainId
212
+ * @param {Address} params.address
213
+ * @param {string} params.message
214
+ * @param {string} params.signatureHash - only provided if status is confirmed
215
+ * @returns {Promise<void>}
216
+ */
217
+ FormoAnalytics.prototype.signature = function (_a) {
218
+ return __awaiter(this, arguments, void 0, function (_b) {
219
+ var status = _b.status, chainId = _b.chainId, address = _b.address, message = _b.message, signatureHash = _b.signatureHash;
220
+ return __generator(this, function (_c) {
221
+ switch (_c.label) {
222
+ case 0: return [4 /*yield*/, this.trackEvent(Event.SIGNATURE, __assign({ status: status, chainId: chainId, address: address, message: message }, (signatureHash && { signatureHash: signatureHash })))];
223
+ case 1:
224
+ _c.sent();
225
+ return [2 /*return*/];
226
+ }
227
+ });
228
+ });
229
+ };
230
+ /**
231
+ * Emits a transaction event.
232
+ * @param {TransactionStatus} params.status - started, broadcasted, rejected
233
+ * @param {ChainID} params.chainId
234
+ * @param {Address} params.address
235
+ * @param {string} params.data
236
+ * @param {string} params.to
237
+ * @param {string} params.value
238
+ * @param {string} params.transactionHash - only provided if status is broadcasted
239
+ * @returns {Promise<void>}
240
+ */
241
+ FormoAnalytics.prototype.transaction = function (_a) {
242
+ return __awaiter(this, arguments, void 0, function (_b) {
243
+ var status = _b.status, chainId = _b.chainId, address = _b.address, data = _b.data, to = _b.to, value = _b.value, transactionHash = _b.transactionHash;
244
+ return __generator(this, function (_c) {
245
+ switch (_c.label) {
246
+ case 0: return [4 /*yield*/, this.trackEvent(Event.TRANSACTION, __assign({ status: status, chainId: chainId, address: address, data: data, to: to, value: value }, (transactionHash && { transactionHash: transactionHash })))];
247
+ case 1:
248
+ _c.sent();
249
+ return [2 /*return*/];
250
+ }
251
+ });
252
+ });
253
+ };
254
+ /**
255
+ * Emits an identify event with current wallet address.
256
+ * @param {Address} params.address
257
+ * @returns {Promise<void>}
258
+ */
259
+ FormoAnalytics.prototype.identify = function (_a) {
260
+ return __awaiter(this, arguments, void 0, function (_b) {
261
+ var address = _b.address, providerName = _b.providerName, rdns = _b.rdns;
262
+ return __generator(this, function (_c) {
263
+ switch (_c.label) {
264
+ case 0:
265
+ if (this.session.isIdentified())
266
+ return [2 /*return*/, console.warn("FormoAnalytics::identify: Wallet already identified in this session")];
267
+ this.session.identify();
268
+ return [4 /*yield*/, this.trackEvent(Event.IDENTIFY, {
269
+ address: address,
270
+ providerName: providerName,
271
+ rdns: rdns,
272
+ })];
273
+ case 1:
274
+ _c.sent();
275
+ return [2 /*return*/];
276
+ }
277
+ });
278
+ });
279
+ };
280
+ /**
281
+ * Emits a custom event with custom data.
282
+ * @param {string} action
283
+ * @param {Record<string, any>} payload
284
+ * @returns {Promise<void>}
285
+ */
286
+ FormoAnalytics.prototype.track = function (action, payload) {
287
+ return __awaiter(this, void 0, void 0, function () {
288
+ return __generator(this, function (_a) {
289
+ switch (_a.label) {
290
+ case 0: return [4 /*yield*/, this.trackEvent(action, payload)];
291
+ case 1:
292
+ _a.sent();
293
+ return [2 /*return*/];
294
+ }
295
+ });
296
+ });
297
+ };
298
+ /*
299
+ SDK tracking and event listener functions
300
+ */
301
+ FormoAnalytics.prototype.trackProvider = function (provider) {
302
+ if (provider === this._provider) {
303
+ console.warn("FormoAnalytics::trackProvider: Provider already tracked.");
304
+ return;
305
+ }
306
+ this.currentChainId = undefined;
307
+ this.currentConnectedAddress = undefined;
308
+ if (this._provider) {
309
+ var actions = Object.keys(this._providerListeners);
310
+ for (var _i = 0, actions_1 = actions; _i < actions_1.length; _i++) {
311
+ var action = actions_1[_i];
312
+ this._provider.removeListener(action, this._providerListeners[action]);
313
+ delete this._providerListeners[action];
314
+ }
315
+ }
316
+ this._provider = provider;
317
+ // Register listeners for web3 provider events
318
+ this.registerAddressChangedListener();
319
+ this.registerChainChangedListener();
320
+ this.registerSignatureListener();
321
+ this.registerTransactionListener();
322
+ };
323
+ FormoAnalytics.prototype.registerAddressChangedListener = function () {
324
+ var _this = this;
325
+ var _a, _b;
326
+ var listener = function () {
327
+ var args = [];
328
+ for (var _i = 0; _i < arguments.length; _i++) {
329
+ args[_i] = arguments[_i];
330
+ }
331
+ return _this.onAddressChanged(args[0]);
332
+ };
333
+ (_a = this._provider) === null || _a === void 0 ? void 0 : _a.on("accountsChanged", listener);
334
+ this._providerListeners["accountsChanged"] = listener;
335
+ var onAddressDisconnected = this.onAddressDisconnected.bind(this);
336
+ (_b = this._provider) === null || _b === void 0 ? void 0 : _b.on("disconnect", onAddressDisconnected);
337
+ this._providerListeners["disconnect"] = onAddressDisconnected;
338
+ };
339
+ FormoAnalytics.prototype.registerChainChangedListener = function () {
340
+ var _this = this;
341
+ var _a;
342
+ var listener = function () {
343
+ var args = [];
344
+ for (var _i = 0; _i < arguments.length; _i++) {
345
+ args[_i] = arguments[_i];
346
+ }
347
+ return _this.onChainChanged(args[0]);
348
+ };
349
+ (_a = this.provider) === null || _a === void 0 ? void 0 : _a.on("chainChanged", listener);
350
+ this._providerListeners["chainChanged"] = listener;
351
+ };
352
+ FormoAnalytics.prototype.registerSignatureListener = function () {
353
+ var _this = this;
354
+ var _a;
355
+ if (!this.provider) {
356
+ console.error("_trackSigning: provider not found");
357
+ return;
358
+ }
359
+ if (((_a = Object.getOwnPropertyDescriptor(this.provider, "request")) === null || _a === void 0 ? void 0 : _a.writable) ===
360
+ false) {
361
+ console.warn("_trackSigning: provider.request is not writable");
362
+ return;
363
+ }
364
+ var request = this.provider.request.bind(this.provider);
365
+ this.provider.request = function (_a) { return __awaiter(_this, [_a], void 0, function (_b) {
366
+ var response, error_1, rpcError;
367
+ var method = _b.method, params = _b.params;
368
+ return __generator(this, function (_c) {
369
+ switch (_c.label) {
370
+ case 0:
371
+ if (!(Array.isArray(params) &&
372
+ ["eth_signTypedData_v4", "personal_sign"].includes(method))) return [3 /*break*/, 4];
373
+ // Emit signature request event
374
+ this.signature(__assign({ status: SignatureStatus.REQUESTED }, this.buildSignatureEventPayload(method, params)));
375
+ _c.label = 1;
376
+ case 1:
377
+ _c.trys.push([1, 3, , 4]);
378
+ return [4 /*yield*/, request({ method: method, params: params })];
379
+ case 2:
380
+ response = (_c.sent());
381
+ if (response) {
382
+ // Emit signature confirmed event
383
+ this.signature(__assign({ status: SignatureStatus.CONFIRMED }, this.buildSignatureEventPayload(method, params, response)));
384
+ }
385
+ return [2 /*return*/, response];
386
+ case 3:
387
+ error_1 = _c.sent();
388
+ rpcError = error_1;
389
+ if (rpcError && (rpcError === null || rpcError === void 0 ? void 0 : rpcError.code) === 4001) {
390
+ // Emit signature rejected event
391
+ this.signature(__assign({ status: SignatureStatus.REJECTED }, this.buildSignatureEventPayload(method, params)));
392
+ }
393
+ throw error_1;
394
+ case 4: return [2 /*return*/, request({ method: method, params: params })];
395
+ }
396
+ });
397
+ }); };
398
+ return;
399
+ };
400
+ FormoAnalytics.prototype.registerTransactionListener = function () {
401
+ var _this = this;
402
+ var _a;
403
+ if (!this.provider) {
404
+ console.error("_trackTransactions: provider not found");
405
+ return;
406
+ }
407
+ if (((_a = Object.getOwnPropertyDescriptor(this.provider, "request")) === null || _a === void 0 ? void 0 : _a.writable) ===
408
+ false) {
409
+ console.warn("_trackTransactions: provider.request is not writable");
410
+ return;
411
+ }
412
+ var request = this.provider.request.bind(this.provider);
413
+ this.provider.request = function (_a) { return __awaiter(_this, [_a], void 0, function (_b) {
414
+ var payload, transactionHash, error_2, rpcError;
415
+ var method = _b.method, params = _b.params;
416
+ return __generator(this, function (_c) {
417
+ switch (_c.label) {
418
+ case 0:
419
+ if (!(Array.isArray(params) &&
420
+ method === "eth_sendTransaction" &&
421
+ params[0])) return [3 /*break*/, 5];
422
+ return [4 /*yield*/, this.buildTransactionEventPayload(params)];
423
+ case 1:
424
+ payload = _c.sent();
425
+ this.transaction(__assign({ status: TransactionStatus.STARTED }, payload));
426
+ _c.label = 2;
427
+ case 2:
428
+ _c.trys.push([2, 4, , 5]);
429
+ return [4 /*yield*/, request({ method: method, params: params })];
430
+ case 3:
431
+ transactionHash = (_c.sent());
432
+ // Track transaction broadcast
433
+ this.transaction(__assign(__assign({ status: TransactionStatus.BROADCASTED }, payload), { transactionHash: transactionHash }));
434
+ return [2 /*return*/];
435
+ case 4:
436
+ error_2 = _c.sent();
437
+ console.log("transaction listener catch");
438
+ console.log(error_2);
439
+ rpcError = error_2;
440
+ if (rpcError && (rpcError === null || rpcError === void 0 ? void 0 : rpcError.code) === 4001) {
441
+ // Emit transaction rejected event
442
+ this.transaction(__assign({ status: TransactionStatus.REJECTED }, payload));
443
+ }
444
+ throw error_2;
445
+ case 5: return [2 /*return*/, request({ method: method, params: params })];
446
+ }
447
+ });
448
+ }); };
449
+ return;
450
+ };
451
+ FormoAnalytics.prototype.onAddressChanged = function (addresses) {
452
+ return __awaiter(this, void 0, void 0, function () {
453
+ return __generator(this, function (_a) {
454
+ if (addresses.length > 0) {
455
+ this.onAddressConnected(addresses[0]);
456
+ }
457
+ else {
458
+ this.onAddressDisconnected();
459
+ }
460
+ return [2 /*return*/];
461
+ });
462
+ });
463
+ };
464
+ FormoAnalytics.prototype.onAddressConnected = function (address) {
465
+ return __awaiter(this, void 0, void 0, function () {
466
+ var _a;
467
+ return __generator(this, function (_b) {
468
+ switch (_b.label) {
469
+ case 0:
470
+ if (address === this.currentConnectedAddress) {
471
+ // We have already reported this address
472
+ return [2 /*return*/];
473
+ }
474
+ else {
475
+ this.currentConnectedAddress = address;
476
+ }
477
+ _a = this;
478
+ return [4 /*yield*/, this.getCurrentChainId()];
479
+ case 1:
480
+ _a.currentChainId = _b.sent();
481
+ this.connect({ chainId: this.currentChainId, address: address });
482
+ return [2 /*return*/];
483
+ }
484
+ });
485
+ });
486
+ };
487
+ FormoAnalytics.prototype.handleDisconnect = function (chainId, address) {
488
+ return __awaiter(this, void 0, void 0, function () {
489
+ var payload;
490
+ return __generator(this, function (_a) {
491
+ switch (_a.label) {
492
+ case 0:
493
+ payload = {
494
+ chain_id: chainId || this.currentChainId,
495
+ address: address || this.currentConnectedAddress,
496
+ };
497
+ this.currentChainId = undefined;
498
+ this.currentConnectedAddress = undefined;
499
+ return [4 /*yield*/, this.trackEvent(Event.DISCONNECT, payload)];
500
+ case 1:
501
+ _a.sent();
502
+ return [2 /*return*/];
503
+ }
504
+ });
505
+ });
506
+ };
507
+ FormoAnalytics.prototype.onAddressDisconnected = function () {
508
+ return __awaiter(this, void 0, void 0, function () {
509
+ return __generator(this, function (_a) {
510
+ switch (_a.label) {
511
+ case 0: return [4 /*yield*/, this.handleDisconnect(this.currentChainId, this.currentConnectedAddress)];
512
+ case 1:
513
+ _a.sent();
514
+ return [2 /*return*/];
515
+ }
516
+ });
517
+ });
518
+ };
519
+ FormoAnalytics.prototype.onChainChanged = function (chainIdHex) {
520
+ return __awaiter(this, void 0, void 0, function () {
521
+ var address;
522
+ return __generator(this, function (_a) {
523
+ switch (_a.label) {
524
+ case 0:
525
+ this.currentChainId = parseInt(chainIdHex);
526
+ if (!!this.currentConnectedAddress) return [3 /*break*/, 2];
527
+ if (!this.provider) {
528
+ console.log("FormoAnalytics::onChainChanged: provider not found. CHAIN_CHANGED not reported");
529
+ return [2 /*return*/, Promise.resolve()];
530
+ }
531
+ return [4 /*yield*/, this.getAddress()];
532
+ case 1:
533
+ address = _a.sent();
534
+ if (!address) {
535
+ console.log("FormoAnalytics::onChainChanged: Unable to fetch or store connected address");
536
+ return [2 /*return*/, Promise.resolve()];
537
+ }
538
+ this.currentConnectedAddress = address;
539
+ _a.label = 2;
540
+ case 2:
541
+ // Proceed only if the address exists
542
+ if (this.currentConnectedAddress) {
543
+ return [2 /*return*/, this.chain({
544
+ chainId: this.currentChainId,
545
+ address: this.currentConnectedAddress,
546
+ })];
547
+ }
548
+ else {
549
+ console.log("FormoAnalytics::onChainChanged: currentConnectedAddress is null despite fetch attempt");
550
+ }
551
+ return [2 /*return*/];
552
+ }
553
+ });
554
+ });
555
+ };
556
+ FormoAnalytics.prototype.trackFirstPageHit = function () {
557
+ return __awaiter(this, void 0, void 0, function () {
558
+ return __generator(this, function (_a) {
559
+ if (session.get(CURRENT_URL_KEY) === null) {
560
+ session.set(CURRENT_URL_KEY, window.location.href);
561
+ }
562
+ return [2 /*return*/, this.trackPageHit()];
563
+ });
564
+ });
565
+ };
566
+ FormoAnalytics.prototype.trackPageHits = function () {
567
+ return __awaiter(this, void 0, void 0, function () {
568
+ var oldPushState, oldReplaceState;
569
+ var _this = this;
570
+ return __generator(this, function (_a) {
571
+ oldPushState = history.pushState;
572
+ history.pushState = function pushState() {
573
+ var args = [];
574
+ for (var _i = 0; _i < arguments.length; _i++) {
575
+ args[_i] = arguments[_i];
576
+ }
577
+ var ret = oldPushState.apply(this, args);
578
+ window.dispatchEvent(new window.Event("locationchange"));
579
+ return ret;
580
+ };
581
+ oldReplaceState = history.replaceState;
582
+ history.replaceState = function replaceState() {
583
+ var args = [];
584
+ for (var _i = 0; _i < arguments.length; _i++) {
585
+ args[_i] = arguments[_i];
586
+ }
587
+ var ret = oldReplaceState.apply(this, args);
588
+ window.dispatchEvent(new window.Event("locationchange"));
589
+ return ret;
590
+ };
591
+ window.addEventListener("popstate", function () { return _this.onLocationChange(); });
592
+ window.addEventListener("locationchange", function () { return _this.onLocationChange(); });
593
+ return [2 /*return*/];
594
+ });
595
+ });
596
+ };
597
+ FormoAnalytics.prototype.onLocationChange = function () {
598
+ return __awaiter(this, void 0, void 0, function () {
599
+ var currentUrl;
600
+ return __generator(this, function (_a) {
601
+ currentUrl = session.get(CURRENT_URL_KEY);
602
+ if (currentUrl !== window.location.href) {
603
+ session.set(CURRENT_URL_KEY, window.location.href);
604
+ this.trackPageHit();
605
+ }
606
+ return [2 /*return*/];
607
+ });
608
+ });
609
+ };
610
+ FormoAnalytics.prototype.trackPageHit = function () {
611
+ var _this = this;
612
+ var pathname = window.location.pathname;
613
+ var href = window.location.href;
614
+ var hash = window.location.hash;
615
+ if (!this.config.trackLocalhost && isLocalhost()) {
616
+ return console.warn("FormoAnalytics::trackPageHit: Ignoring event because website is running locally");
617
+ }
618
+ setTimeout(function () { return __awaiter(_this, void 0, void 0, function () {
619
+ return __generator(this, function (_a) {
620
+ this.trackEvent(Event.PAGE, {
621
+ pathname: pathname,
622
+ href: href,
623
+ hash: hash,
624
+ });
625
+ return [2 /*return*/];
626
+ });
627
+ }); }, 300);
628
+ };
629
+ // TODO: refactor this with event queue and flushing
630
+ // https://linear.app/getformo/issue/P-835/sdk-refactor-retries-with-event-queue-and-batching
631
+ FormoAnalytics.prototype.trackEvent = function (action, payload) {
632
+ return __awaiter(this, void 0, void 0, function () {
633
+ var address, requestData, response, error_3;
634
+ var _a;
635
+ return __generator(this, function (_b) {
636
+ switch (_b.label) {
637
+ case 0: return [4 /*yield*/, this.getAddress()];
638
+ case 1:
639
+ address = _b.sent();
640
+ _a = {
641
+ address: address,
642
+ timestamp: new Date().toISOString(),
643
+ action: action,
644
+ version: "1"
645
+ };
646
+ return [4 /*yield*/, this.buildEventPayload(toSnakeCase(payload))];
647
+ case 2:
648
+ requestData = (_a.payload = _b.sent(),
649
+ _a);
650
+ _b.label = 3;
651
+ case 3:
652
+ _b.trys.push([3, 5, , 6]);
653
+ return [4 /*yield*/, axios.post(EVENTS_API_URL, JSON.stringify(requestData), {
654
+ headers: {
655
+ "Content-Type": "application/json",
656
+ Authorization: "Bearer ".concat(this.config.apiKey),
657
+ },
658
+ })];
659
+ case 4:
660
+ response = _b.sent();
661
+ if (response.status >= 200 && response.status < 300) {
662
+ console.log("Event sent successfully: ".concat(this.getActionDescriptor(action, payload)));
663
+ }
664
+ else {
665
+ throw new Error("Failed with status: ".concat(response.status));
666
+ }
667
+ return [3 /*break*/, 6];
668
+ case 5:
669
+ error_3 = _b.sent();
670
+ console.error("Event \"".concat(action, "\" failed. Error: ").concat(error_3));
671
+ return [3 /*break*/, 6];
672
+ case 6: return [2 /*return*/];
673
+ }
674
+ });
675
+ });
676
+ };
677
+ /*
678
+ Utility functions
679
+ */
680
+ FormoAnalytics.prototype.getProviders = function () {
681
+ return __awaiter(this, void 0, void 0, function () {
682
+ var store, providers;
683
+ return __generator(this, function (_a) {
684
+ store = createStore();
685
+ providers = __spreadArray([], store.getProviders(), true);
686
+ // TODO: consider using store.subscribe to detect changes to providers list
687
+ // store.subscribe(providers => (state.providers = providers))
688
+ // Fallback to injected provider if no providers are found
689
+ if (providers.length === 0) {
690
+ return [2 /*return*/, [window === null || window === void 0 ? void 0 : window.ethereum]];
691
+ }
692
+ return [2 /*return*/, providers];
693
+ });
694
+ });
695
+ };
696
+ FormoAnalytics.prototype.identifyAll = function (providers) {
697
+ return __awaiter(this, void 0, void 0, function () {
698
+ var _i, providers_1, _a, provider, info, accounts, _b, accounts_1, address, err_1;
699
+ return __generator(this, function (_c) {
700
+ switch (_c.label) {
701
+ case 0:
702
+ _i = 0, providers_1 = providers;
703
+ _c.label = 1;
704
+ case 1:
705
+ if (!(_i < providers_1.length)) return [3 /*break*/, 13];
706
+ _a = providers_1[_i], provider = _a.provider, info = _a.info;
707
+ _c.label = 2;
708
+ case 2:
709
+ _c.trys.push([2, 11, , 12]);
710
+ return [4 /*yield*/, this.getAccounts(provider)];
711
+ case 3:
712
+ accounts = _c.sent();
713
+ if (!(accounts && accounts.length > 0)) return [3 /*break*/, 8];
714
+ _b = 0, accounts_1 = accounts;
715
+ _c.label = 4;
716
+ case 4:
717
+ if (!(_b < accounts_1.length)) return [3 /*break*/, 7];
718
+ address = accounts_1[_b];
719
+ return [4 /*yield*/, this.identify({
720
+ address: address,
721
+ providerName: info.name,
722
+ rdns: info.rdns,
723
+ })];
724
+ case 5:
725
+ _c.sent();
726
+ _c.label = 6;
727
+ case 6:
728
+ _b++;
729
+ return [3 /*break*/, 4];
730
+ case 7: return [3 /*break*/, 10];
731
+ case 8:
732
+ // Identify without accounts
733
+ return [4 /*yield*/, this.identify({
734
+ address: null,
735
+ providerName: info.name,
736
+ rdns: info.rdns,
737
+ })];
738
+ case 9:
739
+ // Identify without accounts
740
+ _c.sent();
741
+ _c.label = 10;
742
+ case 10: return [3 /*break*/, 12];
743
+ case 11:
744
+ err_1 = _c.sent();
745
+ console.log("identifying all => err", err_1);
746
+ return [3 /*break*/, 12];
747
+ case 12:
748
+ _i++;
749
+ return [3 /*break*/, 1];
750
+ case 13: return [2 /*return*/];
751
+ }
752
+ });
753
+ });
754
+ };
755
+ Object.defineProperty(FormoAnalytics.prototype, "provider", {
756
+ get: function () {
757
+ return this._provider;
758
+ },
759
+ enumerable: false,
760
+ configurable: true
761
+ });
762
+ FormoAnalytics.prototype.getAddress = function () {
763
+ return __awaiter(this, void 0, void 0, function () {
764
+ var accounts, err_2;
765
+ return __generator(this, function (_a) {
766
+ switch (_a.label) {
767
+ case 0:
768
+ if (this.currentConnectedAddress)
769
+ return [2 /*return*/, this.currentConnectedAddress];
770
+ if (!this.provider) {
771
+ console.log("FormoAnalytics::getAddress: the provider is not set");
772
+ return [2 /*return*/, null];
773
+ }
774
+ _a.label = 1;
775
+ case 1:
776
+ _a.trys.push([1, 3, , 4]);
777
+ return [4 /*yield*/, this.getAccounts()];
778
+ case 2:
779
+ accounts = _a.sent();
780
+ if (accounts && accounts.length > 0) {
781
+ return [2 /*return*/, accounts[0]];
782
+ }
783
+ return [3 /*break*/, 4];
784
+ case 3:
785
+ err_2 = _a.sent();
786
+ console.log("Failed to fetch accounts from provider:", err_2);
787
+ return [2 /*return*/, null];
788
+ case 4: return [2 /*return*/, null];
789
+ }
790
+ });
791
+ });
792
+ };
793
+ FormoAnalytics.prototype.getAccounts = function (provider) {
794
+ return __awaiter(this, void 0, void 0, function () {
795
+ var p, res, err_3;
796
+ return __generator(this, function (_a) {
797
+ switch (_a.label) {
798
+ case 0:
799
+ p = provider || this.provider;
800
+ _a.label = 1;
801
+ case 1:
802
+ _a.trys.push([1, 3, , 4]);
803
+ return [4 /*yield*/, (p === null || p === void 0 ? void 0 : p.request({
804
+ method: "eth_accounts",
805
+ }))];
806
+ case 2:
807
+ res = _a.sent();
808
+ if (!res || res.length === 0)
809
+ return [2 /*return*/, null];
810
+ return [2 /*return*/, res];
811
+ case 3:
812
+ err_3 = _a.sent();
813
+ if (err_3.code !== 4001) {
814
+ console.log("FormoAnalytics::getAccounts: eth_accounts threw an error", err_3);
815
+ }
816
+ return [2 /*return*/, null];
817
+ case 4: return [2 /*return*/];
818
+ }
819
+ });
820
+ });
821
+ };
822
+ FormoAnalytics.prototype.getCurrentChainId = function () {
823
+ return __awaiter(this, void 0, void 0, function () {
824
+ var chainIdHex, err_4;
825
+ var _a;
826
+ return __generator(this, function (_b) {
827
+ switch (_b.label) {
828
+ case 0:
829
+ if (!this.provider) {
830
+ console.error("FormoAnalytics::getCurrentChainId: provider not set");
831
+ }
832
+ _b.label = 1;
833
+ case 1:
834
+ _b.trys.push([1, 3, , 4]);
835
+ return [4 /*yield*/, ((_a = this.provider) === null || _a === void 0 ? void 0 : _a.request({
836
+ method: "eth_chainId",
837
+ }))];
838
+ case 2:
839
+ chainIdHex = _b.sent();
840
+ if (!chainIdHex) {
841
+ console.log("FormoAnalytics::fetchChainId: chain id not found");
842
+ return [2 /*return*/, 0];
843
+ }
844
+ return [2 /*return*/, parseInt(chainIdHex, 16)];
845
+ case 3:
846
+ err_4 = _b.sent();
847
+ console.log("FormoAnalytics::fetchChainId: eth_chainId threw an error", err_4);
848
+ return [2 /*return*/, 0];
849
+ case 4: return [2 /*return*/];
850
+ }
851
+ });
852
+ });
853
+ };
854
+ FormoAnalytics.prototype.getLocation = function () {
855
+ try {
856
+ var timezone = Intl.DateTimeFormat().resolvedOptions().timeZone;
857
+ if (timezone in COUNTRY_LIST)
858
+ return COUNTRY_LIST[timezone];
859
+ return timezone;
860
+ }
861
+ catch (error) {
862
+ console.error("Error resolving timezone:", error);
863
+ return "";
864
+ }
865
+ };
866
+ FormoAnalytics.prototype.getLanguage = function () {
867
+ try {
868
+ return ((navigator.languages && navigator.languages.length
869
+ ? navigator.languages[0]
870
+ : navigator.language) || "en");
871
+ }
872
+ catch (error) {
873
+ console.error("Error resolving language:", error);
874
+ return "en";
875
+ }
876
+ };
877
+ // Adds browser properties to the user-supplied payload
878
+ FormoAnalytics.prototype.buildEventPayload = function () {
879
+ return __awaiter(this, arguments, void 0, function (eventSpecificPayload) {
880
+ var url, params, location, language;
881
+ if (eventSpecificPayload === void 0) { eventSpecificPayload = {}; }
882
+ return __generator(this, function (_a) {
883
+ url = new URL(window.location.href);
884
+ params = new URLSearchParams(url.search);
885
+ location = this.getLocation();
886
+ language = this.getLanguage();
887
+ // common browser properties
888
+ return [2 /*return*/, __assign({ "user-agent": window.navigator.userAgent, origin: url.origin, locale: language, location: location, referrer: document.referrer, utm_source: params.get("utm_source"), utm_medium: params.get("utm_medium"), utm_campaign: params.get("utm_campaign"), ref: params.get("ref") }, eventSpecificPayload)];
889
+ });
890
+ });
891
+ };
892
+ FormoAnalytics.prototype.buildSignatureEventPayload = function (method, params, response) {
893
+ var basePayload = {
894
+ chainId: this.currentChainId,
895
+ address: method === "personal_sign"
896
+ ? params[1]
897
+ : params[0],
898
+ };
899
+ if (method === "personal_sign") {
900
+ var message = Buffer.from(params[0].slice(2), "hex").toString("utf8");
901
+ return __assign(__assign(__assign({}, basePayload), { message: message }), (response ? { signatureHash: response } : {}));
902
+ }
903
+ return __assign(__assign(__assign({}, basePayload), { message: params[1] }), (response ? { signatureHash: response } : {}));
904
+ };
905
+ FormoAnalytics.prototype.buildTransactionEventPayload = function (params) {
906
+ return __awaiter(this, void 0, void 0, function () {
907
+ var _a, data, from, to, value, _b;
908
+ var _c;
909
+ return __generator(this, function (_d) {
910
+ switch (_d.label) {
911
+ case 0:
912
+ _a = params[0], data = _a.data, from = _a.from, to = _a.to, value = _a.value;
913
+ _c = {};
914
+ _b = this.currentChainId;
915
+ if (_b) return [3 /*break*/, 2];
916
+ return [4 /*yield*/, this.getCurrentChainId()];
917
+ case 1:
918
+ _b = (_d.sent());
919
+ _d.label = 2;
920
+ case 2: return [2 /*return*/, (_c.chainId = _b,
921
+ _c.data = data,
922
+ _c.address = from,
923
+ _c.to = to,
924
+ _c.value = value,
925
+ _c)];
926
+ }
927
+ });
928
+ });
929
+ };
930
+ FormoAnalytics.prototype.getActionDescriptor = function (action, payload) {
931
+ return "".concat(action).concat(payload.status ? " ".concat(payload.status) : "");
932
+ };
933
+ return FormoAnalytics;
934
+ }());
935
+ export { FormoAnalytics };
936
+ var FormoAnalyticsSession = /** @class */ (function () {
937
+ function FormoAnalyticsSession() {
938
+ }
939
+ FormoAnalyticsSession.prototype.isIdentified = function () {
940
+ return session.get(SESSION_IDENTIFIED_KEY) === true;
941
+ };
942
+ FormoAnalyticsSession.prototype.identify = function () {
943
+ session.set(SESSION_IDENTIFIED_KEY, true);
944
+ };
945
+ return FormoAnalyticsSession;
946
+ }());
947
+ //# sourceMappingURL=FormoAnalytics.js.map