@formo/analytics 1.13.4-alpha.4 → 1.13.4-alpha.6

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