@amplitude/session-replay-browser 1.43.0 → 1.44.1

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 (75) hide show
  1. package/lib/cjs/config/local-config.d.ts +3 -1
  2. package/lib/cjs/config/local-config.d.ts.map +1 -1
  3. package/lib/cjs/config/local-config.js +60 -2
  4. package/lib/cjs/config/local-config.js.map +1 -1
  5. package/lib/cjs/config/types.d.ts +43 -0
  6. package/lib/cjs/config/types.d.ts.map +1 -1
  7. package/lib/cjs/config/types.js.map +1 -1
  8. package/lib/cjs/constants.d.ts +1 -0
  9. package/lib/cjs/constants.d.ts.map +1 -1
  10. package/lib/cjs/constants.js +7 -1
  11. package/lib/cjs/constants.js.map +1 -1
  12. package/lib/cjs/events/base-events-store.d.ts.map +1 -1
  13. package/lib/cjs/events/base-events-store.js +1 -1
  14. package/lib/cjs/events/base-events-store.js.map +1 -1
  15. package/lib/cjs/events/events-idb-store.d.ts +2 -0
  16. package/lib/cjs/events/events-idb-store.d.ts.map +1 -1
  17. package/lib/cjs/events/events-idb-store.js +44 -15
  18. package/lib/cjs/events/events-idb-store.js.map +1 -1
  19. package/lib/cjs/events/events-memory-store.d.ts +2 -0
  20. package/lib/cjs/events/events-memory-store.d.ts.map +1 -1
  21. package/lib/cjs/events/events-memory-store.js +60 -11
  22. package/lib/cjs/events/events-memory-store.js.map +1 -1
  23. package/lib/cjs/session-replay.d.ts.map +1 -1
  24. package/lib/cjs/session-replay.js +27 -25
  25. package/lib/cjs/session-replay.js.map +1 -1
  26. package/lib/cjs/track-destination.d.ts +19 -1
  27. package/lib/cjs/track-destination.d.ts.map +1 -1
  28. package/lib/cjs/track-destination.js +188 -11
  29. package/lib/cjs/track-destination.js.map +1 -1
  30. package/lib/cjs/version.d.ts +1 -1
  31. package/lib/cjs/version.js +1 -1
  32. package/lib/cjs/version.js.map +1 -1
  33. package/lib/cjs/worker/index.js +1 -1
  34. package/lib/esm/config/local-config.d.ts +3 -1
  35. package/lib/esm/config/local-config.d.ts.map +1 -1
  36. package/lib/esm/config/local-config.js +61 -3
  37. package/lib/esm/config/local-config.js.map +1 -1
  38. package/lib/esm/config/types.d.ts +43 -0
  39. package/lib/esm/config/types.d.ts.map +1 -1
  40. package/lib/esm/config/types.js.map +1 -1
  41. package/lib/esm/constants.d.ts +1 -0
  42. package/lib/esm/constants.d.ts.map +1 -1
  43. package/lib/esm/constants.js +6 -0
  44. package/lib/esm/constants.js.map +1 -1
  45. package/lib/esm/events/base-events-store.d.ts.map +1 -1
  46. package/lib/esm/events/base-events-store.js +1 -1
  47. package/lib/esm/events/base-events-store.js.map +1 -1
  48. package/lib/esm/events/events-idb-store.d.ts +2 -0
  49. package/lib/esm/events/events-idb-store.d.ts.map +1 -1
  50. package/lib/esm/events/events-idb-store.js +44 -15
  51. package/lib/esm/events/events-idb-store.js.map +1 -1
  52. package/lib/esm/events/events-memory-store.d.ts +2 -0
  53. package/lib/esm/events/events-memory-store.d.ts.map +1 -1
  54. package/lib/esm/events/events-memory-store.js +61 -12
  55. package/lib/esm/events/events-memory-store.js.map +1 -1
  56. package/lib/esm/session-replay.d.ts.map +1 -1
  57. package/lib/esm/session-replay.js +27 -25
  58. package/lib/esm/session-replay.js.map +1 -1
  59. package/lib/esm/track-destination.d.ts +19 -1
  60. package/lib/esm/track-destination.d.ts.map +1 -1
  61. package/lib/esm/track-destination.js +190 -13
  62. package/lib/esm/track-destination.js.map +1 -1
  63. package/lib/esm/version.d.ts +1 -1
  64. package/lib/esm/version.js +1 -1
  65. package/lib/esm/version.js.map +1 -1
  66. package/lib/esm/worker/index.js +1 -1
  67. package/lib/scripts/index-min.js +1 -1
  68. package/lib/scripts/index-min.js.gz +0 -0
  69. package/lib/scripts/index-min.js.map +1 -1
  70. package/lib/scripts/session-replay-browser-min.js +1 -1
  71. package/lib/scripts/session-replay-browser-min.js.gz +0 -0
  72. package/lib/scripts/session-replay-browser-min.js.map +1 -1
  73. package/lib/scripts/worker-min.js +1 -1
  74. package/lib/scripts/worker-min.js.gz +0 -0
  75. package/package.json +3 -3
@@ -317,13 +317,13 @@ var SessionReplay = /** @class */ (function () {
317
317
  return currentUrl != null ? { url: currentUrl } : undefined;
318
318
  };
319
319
  SessionReplay.prototype._init = function (apiKey, options) {
320
- var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m;
320
+ var _a, _b, _c, _d, _e, _f, _g, _h, _j, _k, _l, _m, _o, _p;
321
321
  return tslib_1.__awaiter(this, void 0, void 0, function () {
322
- var now, _o, _p, joinedConfig, localConfig, remoteConfig, scrollWatcher, managers, storeType, compressionWorkerScript, trackDestinationWorkerScript, globalScope, _q, compressionScript, trackDestinationScript, rrwebEventManager, error_1, typedError, payloadBatcher, interactionEventManager, error_2, typedError, pending_2, pending_1, pending_1_1, _r, event_1, sessionId, messenger, needsUrlTracking;
323
- var e_2, _s;
322
+ var now, _q, _r, joinedConfig, localConfig, remoteConfig, scrollWatcher, managers, storeType, compressionWorkerScript, trackDestinationWorkerScript, globalScope, _s, compressionScript, trackDestinationScript, rrwebEventManager, error_1, typedError, payloadBatcher, interactionEventManager, error_2, typedError, pending_2, pending_1, pending_1_1, _t, event_1, sessionId, messenger, needsUrlTracking;
323
+ var e_2, _u;
324
324
  var _this = this;
325
- return tslib_1.__generator(this, function (_t) {
326
- switch (_t.label) {
325
+ return tslib_1.__generator(this, function (_v) {
326
+ switch (_v.label) {
327
327
  case 0:
328
328
  // Re-init should always tear down any previous URL-change subscription, even when the
329
329
  // next config has no targeting config and we don't subscribe again.
@@ -339,13 +339,13 @@ var SessionReplay = /** @class */ (function () {
339
339
  options.sessionId !== undefined
340
340
  ? (_b = (0, replay_start_time_store_1.getOrInitReplayStartTime)(apiKey, options.sessionId, now, this.loggerProvider)) !== null && _b !== void 0 ? _b : now
341
341
  : now;
342
- _o = this;
342
+ _q = this;
343
343
  return [4 /*yield*/, (0, joined_config_1.createSessionReplayJoinedConfigGenerator)(apiKey, options)];
344
344
  case 1:
345
- _o.joinedConfigGenerator = _t.sent();
345
+ _q.joinedConfigGenerator = _v.sent();
346
346
  return [4 /*yield*/, this.joinedConfigGenerator.generateJoinedConfig()];
347
347
  case 2:
348
- _p = _t.sent(), joinedConfig = _p.joinedConfig, localConfig = _p.localConfig, remoteConfig = _p.remoteConfig;
348
+ _r = _v.sent(), joinedConfig = _r.joinedConfig, localConfig = _r.localConfig, remoteConfig = _r.remoteConfig;
349
349
  this.config = joinedConfig;
350
350
  this.setMetadata(options.sessionId, joinedConfig, localConfig, remoteConfig, (_c = options.version) === null || _c === void 0 ? void 0 : _c.version, version_1.VERSION, (_d = options.version) === null || _d === void 0 ? void 0 : _d.type);
351
351
  this.pageLeaveFns = [];
@@ -369,50 +369,52 @@ var SessionReplay = /** @class */ (function () {
369
369
  if (!(this.config.useWebWorker && globalScope && globalScope.Worker)) return [3 /*break*/, 4];
370
370
  return [4 /*yield*/, Promise.resolve().then(function () { return tslib_1.__importStar(require('./worker')); })];
371
371
  case 3:
372
- _q = _t.sent(), compressionScript = _q.compressionScript, trackDestinationScript = _q.trackDestinationScript;
372
+ _s = _v.sent(), compressionScript = _s.compressionScript, trackDestinationScript = _s.trackDestinationScript;
373
373
  compressionWorkerScript = compressionScript;
374
374
  trackDestinationWorkerScript = trackDestinationScript;
375
- _t.label = 4;
375
+ _v.label = 4;
376
376
  case 4:
377
- _t.trys.push([4, 6, , 7]);
377
+ _v.trys.push([4, 6, , 7]);
378
378
  return [4 /*yield*/, (0, events_manager_1.createEventsManager)({
379
379
  config: this.config,
380
380
  type: 'replay',
381
+ minInterval: (_g = this.config.flushIntervalConfig) === null || _g === void 0 ? void 0 : _g.minIntervalMs,
382
+ maxInterval: (_h = this.config.flushIntervalConfig) === null || _h === void 0 ? void 0 : _h.maxIntervalMs,
381
383
  storeType: storeType,
382
384
  trackDestinationWorkerScript: trackDestinationWorkerScript,
383
385
  shouldSend: function () { return !_this.isBelowMinSessionDuration(); },
384
386
  })];
385
387
  case 5:
386
- rrwebEventManager = _t.sent();
388
+ rrwebEventManager = _v.sent();
387
389
  this.rrwebEventManager = rrwebEventManager;
388
390
  managers.push({ name: 'replay', manager: rrwebEventManager });
389
391
  return [3 /*break*/, 7];
390
392
  case 6:
391
- error_1 = _t.sent();
393
+ error_1 = _v.sent();
392
394
  typedError = error_1;
393
395
  this.loggerProvider.warn("Error occurred while creating replay events manager: ".concat(typedError.toString()));
394
396
  return [3 /*break*/, 7];
395
397
  case 7:
396
- if (!((_g = this.config.interactionConfig) === null || _g === void 0 ? void 0 : _g.enabled)) return [3 /*break*/, 11];
398
+ if (!((_j = this.config.interactionConfig) === null || _j === void 0 ? void 0 : _j.enabled)) return [3 /*break*/, 11];
397
399
  payloadBatcher = this.config.interactionConfig.batch ? click_1.clickBatcher : click_1.clickNonBatcher;
398
- _t.label = 8;
400
+ _v.label = 8;
399
401
  case 8:
400
- _t.trys.push([8, 10, , 11]);
402
+ _v.trys.push([8, 10, , 11]);
401
403
  return [4 /*yield*/, (0, events_manager_1.createEventsManager)({
402
404
  config: this.config,
403
405
  type: 'interaction',
404
- minInterval: (_h = this.config.interactionConfig.trackEveryNms) !== null && _h !== void 0 ? _h : constants_1.INTERACTION_MIN_INTERVAL,
406
+ minInterval: (_k = this.config.interactionConfig.trackEveryNms) !== null && _k !== void 0 ? _k : constants_1.INTERACTION_MIN_INTERVAL,
405
407
  maxInterval: constants_1.INTERACTION_MAX_INTERVAL,
406
408
  payloadBatcher: payloadBatcher,
407
409
  storeType: storeType,
408
410
  trackDestinationWorkerScript: trackDestinationWorkerScript,
409
411
  })];
410
412
  case 9:
411
- interactionEventManager = _t.sent();
413
+ interactionEventManager = _v.sent();
412
414
  managers.push({ name: 'interaction', manager: interactionEventManager });
413
415
  return [3 /*break*/, 11];
414
416
  case 10:
415
- error_2 = _t.sent();
417
+ error_2 = _v.sent();
416
418
  typedError = error_2;
417
419
  this.loggerProvider.warn("Error occurred while creating interaction events manager: ".concat(typedError.toString()));
418
420
  return [3 /*break*/, 11];
@@ -429,14 +431,14 @@ var SessionReplay = /** @class */ (function () {
429
431
  pending_2 = this.pendingEmitEvents.splice(0);
430
432
  try {
431
433
  for (pending_1 = tslib_1.__values(pending_2), pending_1_1 = pending_1.next(); !pending_1_1.done; pending_1_1 = pending_1.next()) {
432
- _r = pending_1_1.value, event_1 = _r.event, sessionId = _r.sessionId;
434
+ _t = pending_1_1.value, event_1 = _t.event, sessionId = _t.sessionId;
433
435
  this.eventCompressor.enqueueEvent(event_1, sessionId);
434
436
  }
435
437
  }
436
438
  catch (e_2_1) { e_2 = { error: e_2_1 }; }
437
439
  finally {
438
440
  try {
439
- if (pending_1_1 && !pending_1_1.done && (_s = pending_1.return)) _s.call(pending_1);
441
+ if (pending_1_1 && !pending_1_1.done && (_u = pending_1.return)) _u.call(pending_1);
440
442
  }
441
443
  finally { if (e_2) throw e_2.error; }
442
444
  }
@@ -473,11 +475,11 @@ var SessionReplay = /** @class */ (function () {
473
475
  ], false);
474
476
  return [4 /*yield*/, this.initializeNetworkObservers()];
475
477
  case 12:
476
- _t.sent();
478
+ _v.sent();
477
479
  // Enable background capture when this page is opened by the Amplitude app
478
480
  // (window.opener exists). Uses the shared messenger singleton so that if
479
481
  // autocapture is also loaded, both share a single messenger and script load.
480
- if ((_j = (0, analytics_core_1.getGlobalScope)()) === null || _j === void 0 ? void 0 : _j.opener) {
482
+ if ((_l = (0, analytics_core_1.getGlobalScope)()) === null || _l === void 0 ? void 0 : _l.opener) {
481
483
  messenger = (0, analytics_core_1.getOrCreateWindowMessenger)();
482
484
  (0, analytics_core_1.enableBackgroundCapture)(messenger);
483
485
  messenger.setup(tslib_1.__assign({ logger: this.loggerProvider }, (this.config.serverZone && { endpoint: analytics_core_1.AMPLITUDE_ORIGINS_MAP[this.config.serverZone] })));
@@ -486,8 +488,8 @@ var SessionReplay = /** @class */ (function () {
486
488
  this.teardownEventListeners(false);
487
489
  return [4 /*yield*/, this.evaluateTargetingAndCapture({ userProperties: options.userProperties, page: this.getCurrentPageForTargeting() }, true)];
488
490
  case 13:
489
- _t.sent();
490
- needsUrlTracking = this.config.targetingConfig || ((_m = (_l = (_k = this.config.privacyConfig) === null || _k === void 0 ? void 0 : _k.urlMaskLevels) === null || _l === void 0 ? void 0 : _l.length) !== null && _m !== void 0 ? _m : 0) > 0;
491
+ _v.sent();
492
+ needsUrlTracking = this.config.targetingConfig || ((_p = (_o = (_m = this.config.privacyConfig) === null || _m === void 0 ? void 0 : _m.urlMaskLevels) === null || _o === void 0 ? void 0 : _o.length) !== null && _p !== void 0 ? _p : 0) > 0;
491
493
  if (needsUrlTracking) {
492
494
  this.setupUrlChangeListener();
493
495
  }
@@ -1 +1 @@
1
- {"version":3,"file":"session-replay.js","sourceRoot":"","sources":["../../src/session-replay.ts"],"names":[],"mappings":";;;;AAAA,4DAYmC;AAEnC,gFAAgF;AAChF,sDAAoG;AACpG,wDAAkF;AASlF,yCAWqB;AACrB,qCASmB;AACnB,8DAA4D;AAC5D,0DAAuF;AACvF,wDAA2D;AAC3D,uCAA4E;AAC5E,yCAA+C;AAC/C,6CAAmD;AACnD,mCAA8C;AAC9C,qEAKmC;AACnC,mEAA0E;AAW1E,uCAA+C;AAC/C,qCAAoC;AAIpC,qEAA+F;AAE/F,+DAA0G;AAI1G;IAmDE;QAAA,iBAEC;QApDD,SAAI,GAAG,mCAAmC,CAAC;QAM3C,yBAAoB,GAAsC,IAAI,CAAC;QAC/D,eAAU,GAAG,CAAC,CAAC;QAEf,0BAAqB,GAAG,KAAK,CAAC;QAI9B,wEAAwE;QACxE,uFAAuF;QACvF,sFAAsF;QACtF,uFAAuF;QACvF,gCAAgC;QAChC,iBAAY,GAAkB,EAAE,CAAC;QAGjC;;;WAGG;QACK,wBAAmB,GAAG,CAAC,CAAC;QAChC,+EAA+E;QACvE,2BAAsB,GAAG,KAAK,CAAC;QAMvC,iDAAiD;QACzC,mBAAc,GAA0B,IAAI,CAAC;QAC7C,yBAAoB,GAAG,KAAK,CAAC;QAC7B,sBAAiB,GAAgE,EAAE,CAAC;QAE5F,gFAAgF;QACxE,mBAAc,GAAG,EAAE,CAAC;QAEpB,yCAAoC,GAAmB,IAAI,CAAC;QAEpE,yFAAyF;QACjF,qBAAgB,GAAwB,IAAI,CAAC;QAC7C,iCAA4B,GAAwC,IAAI,CAAC;QACzE,mCAA8B,GAAwB,IAAI,CAAC;QACnE,qEAAqE;QAC7D,yCAAoC,GAAG,CAAC,CAAC;QAUzC,2BAAsB,GAAG,UAAC,QAAiB;YACjD,IAAM,WAAW,GAAG,IAAA,+BAAc,GAAE,CAAC;YACrC,IAAI,WAAW,EAAE;gBACf,WAAW,CAAC,mBAAmB,CAAC,MAAM,EAAE,KAAI,CAAC,YAAY,CAAC,CAAC;gBAC3D,WAAW,CAAC,mBAAmB,CAAC,OAAO,EAAE,KAAI,CAAC,aAAa,CAAC,CAAC;gBAC7D,CAAC,QAAQ,IAAI,WAAW,CAAC,gBAAgB,CAAC,MAAM,EAAE,KAAI,CAAC,YAAY,CAAC,CAAC;gBACrE,CAAC,QAAQ,IAAI,WAAW,CAAC,gBAAgB,CAAC,OAAO,EAAE,KAAI,CAAC,aAAa,CAAC,CAAC;gBACvE,kFAAkF;gBAClF,4CAA4C;gBAC5C,IAAI,WAAW,CAAC,IAAI,IAAI,YAAY,IAAI,WAAW,CAAC,IAAI,EAAE;oBACxD,WAAW,CAAC,mBAAmB,CAAC,UAAU,EAAE,KAAI,CAAC,iBAAiB,CAAC,CAAC;oBACpE,CAAC,QAAQ,IAAI,WAAW,CAAC,gBAAgB,CAAC,UAAU,EAAE,KAAI,CAAC,iBAAiB,CAAC,CAAC;iBAC/E;qBAAM;oBACL,qFAAqF;oBACrF,0CAA0C;oBAC1C,WAAW,CAAC,mBAAmB,CAAC,cAAc,EAAE,KAAI,CAAC,iBAAiB,CAAC,CAAC;oBACxE,CAAC,QAAQ,IAAI,WAAW,CAAC,gBAAgB,CAAC,cAAc,EAAE,KAAI,CAAC,iBAAiB,CAAC,CAAC;iBACnF;aACF;QACH,CAAC,CAAC;QAmWF,iBAAY,GAAG;YACb,KAAI,CAAC,UAAU,EAAE,CAAC;QACpB,CAAC,CAAC;QAEF,kBAAa,GAAG;YACd,IAAI,KAAI,CAAC,oBAAoB,IAAI,KAAI,CAAC,cAAc,EAAE;gBACpD,IAAI;oBACF,KAAI,CAAC,cAAc,CAAC,gBAAgB,CAAC,IAAI,CAAC,CAAC;iBAC5C;gBAAC,OAAO,KAAK,EAAE;oBACd,KAAI,CAAC,cAAc,CAAC,IAAI,CAAC,wCAAwC,EAAE,KAAK,CAAC,CAAC;iBAC3E;aACF;iBAAM,IAAI,CAAC,KAAI,CAAC,oBAAoB,EAAE;gBACrC,KAAK,KAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;aAC/B;QACH,CAAC,CAAC;QAEF;;;;WAIG;QACK,sBAAiB,GAAG,UAAC,CAA8B;;YACzD,yEAAyE;YACzE,kEAAkE;YAClE,MAAA,KAAI,CAAC,eAAe,0CAAE,UAAU,EAAE,CAAC;YACnC,KAAI,CAAC,UAAU,EAAE,CAAC;YAClB,KAAI,CAAC,YAAY,CAAC,OAAO,CAAC,UAAC,EAAE;gBAC3B,EAAE,CAAC,CAAC,CAAC,CAAC;YACR,CAAC,CAAC,CAAC;QACL,CAAC,CAAC;QAEF,gCAA2B,GAAG,UAC5B,eAA4C,EAC5C,MAAc,EACd,YAAoB,EACpB,0BAAkC;YAFlC,uBAAA,EAAA,cAAc;YACd,6BAAA,EAAA,oBAAoB;YACpB,2CAAA,EAAA,kCAAkC;;;;;;;4BAElC,IAAI,CAAC,IAAI,CAAC,WAAW,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,SAAS,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE;gCACpE,IAAI,IAAI,CAAC,WAAW,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,SAAS,EAAE;oCACnD,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,gFAAgF,CAAC,CAAC;iCAC3G;qCAAM;oCACL,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,qEAAqE,CAAC,CAAC;iCACjG;gCACD,sBAAO;6BACR;4BAED,iDAAiD;4BACjD,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,eAAe,EAAE;gCAChC,IAAI,MAAM,EAAE;oCACV,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,mEAAmE,CAAC,CAAC;iCAC9F;qCAAM;oCACL,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,uEAAuE,CAAC,CAAC;oCACjG,sBAAO;iCACR;6BACF;4BAED,wDAAwD;4BACxD,IAAI,CAAC,mBAAmB,GAAG,eAAe,CAAC;4BAIrC,eAAe,GAAG,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC;4BAC9C,gBAAgB,GAAG,eAAe,IAAI,CAAC,IAAI,CAAC,qBAAqB,CAAC;iCACpE,gBAAgB,EAAhB,wBAAgB;4BAEZ,qBAAqB,GAAG,0BAA0B,CAAC,CAAC,CAAC,IAAI,CAAC,oCAAoC,CAAC,CAAC,CAAC,SAAS,CAAC;4BAC7G,iBAAiB,GAAG,eAAe,CAAC,KAAK,CAAC;4BAC9C,IACE,iBAAiB;gCACjB,MAAM,CAAC,MAAM,CAAC,iCAAgB,CAAC,CAAC,QAAQ,CAAC,iBAAiB,CAAC,UAA8B,CAAC,EAC1F;gCACA,iBAAiB,GAAG,SAAS,CAAC;6BAC/B;4BAEK,OAAO,GAAG,MAAA,MAAA,MAAA,eAAe,CAAC,IAAI,0CAAE,GAAG,mCAAI,MAAA,MAAA,IAAA,+BAAc,GAAE,0CAAE,QAAQ,0CAAE,IAAI,mCAAI,EAAE,CAAC;4BAC9E,gBAAgB,GAAG,MAAA,eAAe,CAAC,IAAI,mCAAI,CAAC,OAAO,KAAK,EAAE,CAAC,CAAC,CAAC,EAAE,GAAG,EAAE,OAAO,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;4BAE1E,qBAAM,IAAA,6CAAyB,EAAC;oCACrD,SAAS,EAAE,IAAI,CAAC,WAAW,CAAC,SAAS;oCACrC,eAAe,iBAAA;oCACf,cAAc,EAAE,IAAI,CAAC,cAAc;oCACnC,MAAM,EAAE,IAAI,CAAC,MAAM,CAAC,MAAM;oCAC1B,eAAe,EAAE;wCACf,cAAc,EAAE,eAAe,CAAC,cAAc;wCAC9C,KAAK,EAAE,iBAAiB;wCACxB,IAAI,EAAE,gBAAgB;qCACvB;oCACD,SAAS,EAAE,0BAA0B;iCACtC,CAAC,EAAA;;4BAXI,cAAc,GAAG,SAWrB;4BAEF,IACE,0BAA0B;gCAC1B,qBAAqB,KAAK,SAAS;gCACnC,qBAAqB,KAAK,IAAI,CAAC,oCAAoC,EACnE;gCACA,IAAI,CAAC,cAAc,CAAC,KAAK,CACvB,sDAA+C,qBAAqB,0BAAgB,IAAI,CAAC,oCAAoC,MAAG,CACjI,CAAC;gCACF,sBAAO;6BACR;4BACD,2EAA2E;4BAC3E,6EAA6E;4BAC7E,4CAA4C;4BAC5C,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC,qBAAqB,IAAI,cAAc,CAAC;4BAE1E,IAAI,CAAC,cAAc,CAAC,KAAK,CACvB,IAAI,CAAC,SAAS,CACZ;gCACE,IAAI,EAAE,gCAAgC;gCACtC,qBAAqB,EAAE,IAAI,CAAC,qBAAqB;gCACjD,KAAK,EAAE,iBAAiB;gCACxB,eAAe,EAAE,eAAe;6BACjC,EACD,IAAI,EACJ,CAAC,CACF,CACF,CAAC;;;iCAGA,MAAM,EAAN,wBAAM;4BACR,KAAK,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;;;iCAClB,CAAA,YAAY,IAAI,CAAC,IAAI,CAAC,oBAAoB,CAAA,EAA1C,wBAA0C;4BACnD,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,2EAA2E,CAAC,CAAC;4BACrG,qBAAM,IAAI,CAAC,YAAY,EAAE,EAAA;;4BAAzB,SAAyB,CAAC;;;;;;SAE7B,CAAC;QAydF,wBAAmB,GAAG,UACpB,SAA2B,EAC3B,SAAsC,EACtC,cAAqB;YADrB,0BAAA,EAAA,cAAsC;YACtC,+BAAA,EAAA,qBAAqB;;;;;;;4BAGf,SAAS,GAA0B,SAAS,CAAC;4BAC3C,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC;iCAEvB,CAAA,MAAM,IAAI,SAAS,KAAK,4BAAgB,CAAC,QAAQ,CAAA,EAAjD,wBAAiD;4BACnD,SAAS,GAAG;gCACV,MAAM,EAAE,IAAA,wBAAc,EAAC,MAAM,CAAC;gCAC9B,OAAO,EAAE,iBAAO;6BACjB,CAAC;iCACE,cAAc,EAAd,wBAAc;4BACQ,qBAAM,IAAA,wBAAc,GAAE,EAAA;;4BAAxC,eAAe,GAAG,SAAsB;4BAC9C,SAAS,yCACJ,eAAe,GACf,SAAS,CACb,CAAC;;;4BAGN,yCAAyC;4BACzC,IAAI,IAAI,CAAC,oBAAoB,IAAI,IAAI,CAAC,cAAc,EAAE;gCACpD,IAAI,CAAC,cAAc,CAAC,cAAc,CAAC,SAAS,wCACvC,SAAS,GACT,SAAS,EACZ,CAAC;6BACJ;iCAAM;gCACL,IAAI,CAAC,cAAc,CAAC,KAAK,CACvB,sDAA+C,SAAS,kCAA+B,CACxF,CAAC;6BACH;;;;4BAED,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,kDAAkD,EAAE,GAAC,CAAC,CAAC;;;;;;SAEpF,CAAC;QAEF,wBAAmB,GAAG;;YACpB,IAAI;gBACF,KAAI,CAAC,cAAc,CAAC,GAAG,CAAC,kCAAkC,CAAC,CAAC;gBAC5D,KAAI,CAAC,oBAAoB,IAAI,KAAI,CAAC,oBAAoB,EAAE,CAAC;gBACzD,KAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC;gBACjC,MAAA,KAAI,CAAC,gBAAgB,0CAAE,IAAI,EAAE,CAAC;gBAC9B,MAAA,KAAI,CAAC,4BAA4B,0CAAE,IAAI,EAAE,CAAC;gBAC1C,KAAI,CAAC,4BAA4B,GAAG,IAAI,CAAC;gBACzC,sEAAsE;gBACtE,sEAAsE;gBACtE,MAAA,KAAI,CAAC,8BAA8B,qDAAI,CAAC;gBACxC,KAAI,CAAC,8BAA8B,GAAG,IAAI,CAAC;aAC5C;YAAC,OAAO,KAAK,EAAE;gBACd,IAAM,UAAU,GAAG,KAAc,CAAC;gBAClC,KAAI,CAAC,cAAc,CAAC,IAAI,CAAC,wDAAiD,UAAU,CAAC,QAAQ,EAAE,CAAE,CAAC,CAAC;aACpG;QACH,CAAC,CAAC;QAzgCA,IAAI,CAAC,cAAc,GAAG,IAAI,2BAAkB,CAAC,IAAI,uBAAM,EAAE,CAAC,CAAC;IAC7D,CAAC;IAED,4BAAI,GAAJ,UAAK,MAAc,EAAE,OAA6B;QAChD,OAAO,IAAA,8BAAa,EAAC,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC,CAAC;IACpD,CAAC;IAuBD;;;;OAIG;IACK,8CAAsB,GAA9B;QAAA,iBAqCC;;QApCC,mFAAmF;QACnF,0EAA0E;QAC1E,MAAA,IAAI,CAAC,gBAAgB,oDAAI,CAAC;QAE1B,IAAM,WAAW,GAAG,IAAA,+BAAc,GAAwB,CAAC;QAC3D,IAAI,CAAC,CAAA,WAAW,aAAX,WAAW,uBAAX,WAAW,CAAE,QAAQ,CAAA,EAAE;YAC1B,OAAO;SACR;QAED,IAAM,YAAY,GAAG,CAAC,CAAC,CAAA,MAAA,IAAI,CAAC,MAAM,0CAAE,eAAe,CAAA,CAAC;QAEpD,IAAM,WAAW,GAAG,UAAC,IAAY;YAC/B,KAAI,CAAC,cAAc,GAAG,IAAI,CAAC;YAE3B,IAAI,YAAY,EAAE;gBAChB,IAAM,YAAY,GAAG,EAAE,KAAI,CAAC,oCAAoC,CAAC;gBACjE,KAAK,KAAI,CAAC,2BAA2B,CACnC;oBACE,cAAc,EAAE,EAAE;oBAClB,KAAK,EAAE,SAAS;oBAChB,IAAI,EAAE,EAAE,GAAG,EAAE,IAAI,EAAE;iBACpB,EACD,KAAK,EACL,KAAK,EACL,IAAI,CACL,CAAC;gBACF,KAAI,CAAC,cAAc,CAAC,KAAK,CAAC,qDAA8C,YAAY,kBAAQ,IAAI,MAAG,CAAC,CAAC;aACtG;QACH,CAAC,CAAC;QAEF,IAAM,WAAW,GAAI,2CAAwC,CAAC,WAAW,EAAE,WAAW,CAAC,CAAC;QAExF,IAAI,CAAC,gBAAgB,GAAG;YACtB,WAAW,EAAE,CAAC;YACd,KAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;QAC/B,CAAC,CAAC;IACJ,CAAC;IAED;;;;;;;;;OASG;IACK,iDAAyB,GAAjC;;QACE,IAAM,oBAAoB,GAAG,MAAA,IAAI,CAAC,MAAM,0CAAE,oBAAoB,CAAC;QAC/D,IAAI,oBAAoB,KAAK,SAAS,IAAI,IAAI,CAAC,gBAAgB,KAAK,SAAS,EAAE;YAC7E,OAAO,KAAK,CAAC;SACd;QACD,OAAO,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,gBAAgB,GAAG,oBAAoB,CAAC;IACnE,CAAC;IAEO,kDAA0B,GAAlC;;QACE,IAAM,UAAU,GAAG,MAAA,MAAA,IAAA,+BAAc,GAAE,0CAAE,QAAQ,0CAAE,IAAI,CAAC;QACpD,OAAO,UAAU,IAAI,IAAI,CAAC,CAAC,CAAC,EAAE,GAAG,EAAE,UAAU,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC;IAC9D,CAAC;IAEe,6BAAK,GAArB,UAAsB,MAAc,EAAE,OAA6B;;;;;;;;;wBACjE,sFAAsF;wBACtF,oEAAoE;wBACpE,MAAA,IAAI,CAAC,gBAAgB,oDAAI,CAAC;wBAE1B,IAAI,CAAC,cAAc,GAAG,IAAI,2BAAkB,CAAC,OAAO,CAAC,cAAc,IAAI,IAAI,uBAAM,EAAE,CAAC,CAAC;wBACrF,MAAM,CAAC,SAAS,CAAC,cAAc,CAAC,IAAI,CAAC,OAAO,EAAE,UAAU,CAAC;4BACvD,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,OAAO,CAAC,QAAoB,CAAC,CAAC;wBAC3D,IAAI,CAAC,cAAc,GAAG,IAAA,uBAAa,GAAE,CAAC;wBACtC,IAAI,CAAC,WAAW,GAAG,IAAI,gCAAkB,CAAC,EAAE,SAAS,EAAE,OAAO,CAAC,SAAS,EAAE,QAAQ,EAAE,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC;wBAIlG,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;wBACvB,IAAA,oDAA0B,EAAC,MAAM,EAAE,GAAG,EAAE,IAAI,CAAC,cAAc,CAAC,CAAC;wBAC7D,IAAI,CAAC,gBAAgB;4BACnB,OAAO,CAAC,SAAS,KAAK,SAAS;gCAC7B,CAAC,CAAC,MAAA,IAAA,kDAAwB,EAAC,MAAM,EAAE,OAAO,CAAC,SAAS,EAAE,GAAG,EAAE,IAAI,CAAC,cAAc,CAAC,mCAAI,GAAG;gCACtF,CAAC,CAAC,GAAG,CAAC;wBACV,KAAA,IAAI,CAAA;wBAAyB,qBAAM,IAAA,wDAAwC,EAAC,MAAM,EAAE,OAAO,CAAC,EAAA;;wBAA5F,GAAK,qBAAqB,GAAG,SAA+D,CAAC;wBACzC,qBAAM,IAAI,CAAC,qBAAqB,CAAC,oBAAoB,EAAE,EAAA;;wBAArG,KAA8C,SAAuD,EAAnG,YAAY,kBAAA,EAAE,WAAW,iBAAA,EAAE,YAAY,kBAAA;wBAC/C,IAAI,CAAC,MAAM,GAAG,YAAY,CAAC;wBAE3B,IAAI,CAAC,WAAW,CACd,OAAO,CAAC,SAAS,EACjB,YAAY,EACZ,WAAW,EACX,YAAY,EACZ,MAAA,OAAO,CAAC,OAAO,0CAAE,OAAO,EACxB,iBAAO,EACP,MAAA,OAAO,CAAC,OAAO,0CAAE,IAAI,CACtB,CAAC;wBAEF,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;wBAEvB,IAAI,OAAO,CAAC,SAAS,KAAI,MAAA,IAAI,CAAC,MAAM,CAAC,iBAAiB,0CAAE,OAAO,CAAA,EAAE;4BACzD,aAAa,GAAG,sBAAa,CAAC,OAAO,CACzC;gCACE,SAAS,EAAE,OAAO,CAAC,SAAS;gCAC5B,IAAI,EAAE,aAAa;6BACpB,EACD,IAAI,CAAC,MAAM,CACZ,CAAC;4BACF,IAAI,CAAC,YAAY,GAAG,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC;4BAC1F,IAAI,CAAC,UAAU,GAAG,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;4BACzD,IAAI,CAAC,YAAY,GAAG,IAAI,oBAAY,CAAC,IAAI,CAAC,cAAc,EAAE,aAAa,CAAC,CAAC;yBAC1E;wBAEK,QAAQ,GAA+C,EAAE,CAAC;wBAC1D,SAAS,GAAK,IAAI,CAAC,MAAM,UAAhB,CAAiB;wBAChC,IAAI,SAAS,KAAK,KAAK,IAAI,CAAC,CAAA,MAAA,IAAA,+BAAc,GAAE,0CAAE,SAAS,CAAA,EAAE;4BACvD,SAAS,GAAG,QAAQ,CAAC;4BACrB,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,2EAA2E,CAAC,CAAC;yBACvG;wBACD,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,gBAAS,SAAS,wBAAqB,CAAC,CAAC;wBAG3D,WAAW,GAAG,IAAA,+BAAc,GAAE,CAAC;6BACjC,CAAA,IAAI,CAAC,MAAM,CAAC,YAAY,IAAI,WAAW,IAAI,WAAW,CAAC,MAAM,CAAA,EAA7D,wBAA6D;wBACT,8FAAa,UAAU,QAAC;;wBAAxE,KAAgD,SAAwB,EAAtE,iBAAiB,uBAAA,EAAE,sBAAsB,4BAAA;wBACjD,uBAAuB,GAAG,iBAAiB,CAAC;wBAC5C,4BAA4B,GAAG,sBAAsB,CAAC;;;;wBAKlC,qBAAM,IAAA,oCAAmB,EAAW;gCACtD,MAAM,EAAE,IAAI,CAAC,MAAM;gCACnB,IAAI,EAAE,QAAQ;gCACd,SAAS,WAAA;gCACT,4BAA4B,8BAAA;gCAC5B,UAAU,EAAE,cAAM,OAAA,CAAC,KAAI,CAAC,yBAAyB,EAAE,EAAjC,CAAiC;6BACpD,CAAC,EAAA;;wBANF,iBAAiB,GAAG,SAMlB,CAAC;wBACH,IAAI,CAAC,iBAAiB,GAAG,iBAAiB,CAAC;wBAC3C,QAAQ,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,QAAQ,EAAE,OAAO,EAAE,iBAAiB,EAAE,CAAC,CAAC;;;;wBAExD,UAAU,GAAG,OAAc,CAAC;wBAClC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,+DAAwD,UAAU,CAAC,QAAQ,EAAE,CAAE,CAAC,CAAC;;;6BAGxG,CAAA,MAAA,IAAI,CAAC,MAAM,CAAC,iBAAiB,0CAAE,OAAO,CAAA,EAAtC,yBAAsC;wBAClC,cAAc,GAAG,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC,CAAC,oBAAY,CAAC,CAAC,CAAC,uBAAe,CAAC;;;;wBAE1D,qBAAM,IAAA,oCAAmB,EAAgB;gCACvE,MAAM,EAAE,IAAI,CAAC,MAAM;gCACnB,IAAI,EAAE,aAAa;gCACnB,WAAW,EAAE,MAAA,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,aAAa,mCAAI,oCAAwB;gCACpF,WAAW,EAAE,oCAAwB;gCACrC,cAAc,gBAAA;gCACd,SAAS,WAAA;gCACT,4BAA4B,8BAAA;6BAC7B,CAAC,EAAA;;wBARI,uBAAuB,GAAG,SAQ9B;wBACF,QAAQ,CAAC,IAAI,CAAC,EAAE,IAAI,EAAE,aAAa,EAAE,OAAO,EAAE,uBAAuB,EAAE,CAAC,CAAC;;;;wBAEnE,UAAU,GAAG,OAAc,CAAC;wBAClC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,oEAA6D,UAAU,CAAC,QAAQ,EAAE,CAAE,CAAC,CAAC;;;wBAInH,IAAI,CAAC,aAAa,QAAO,iCAAiB,YAAjB,iCAAiB,iDAAsC,QAAQ,aAAC,CAAC;wBAC1F,+BAA+B;wBAC/B,IAAI,IAAI,CAAC,eAAe,EAAE;4BACxB,IAAI,CAAC,eAAe,CAAC,SAAS,EAAE,CAAC;yBAClC;wBAED,IAAI,CAAC,eAAe,GAAG,IAAI,kCAAe,CACxC,IAAI,CAAC,aAAa,EAClB,IAAI,CAAC,MAAM,EACX,IAAI,CAAC,WAAW,EAAE,EAClB,uBAAuB,EACvB,cAAM,OAAA,KAAI,CAAC,UAAU,EAAE,EAAjB,CAAiB,CACxB,CAAC;wBAEF,wEAAwE;wBACxE,4EAA4E;wBAC5E,IAAI,IAAI,CAAC,iBAAiB,CAAC,MAAM,GAAG,CAAC,EAAE;4BAC/B,YAAU,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC;;gCACjD,KAAmC,YAAA,iBAAA,SAAO,CAAA,qFAAE;oCAAjC,sBAAoB,EAAlB,kBAAK,EAAE,SAAS,eAAA;oCAC3B,IAAI,CAAC,eAAe,CAAC,YAAY,CAAC,OAAK,EAAE,SAAS,CAAC,CAAC;iCACrD;;;;;;;;;yBACF;wBAED,0EAA0E;wBAC1E,+EAA+E;wBAC/E,EAAE;wBACF,qEAAqE;wBACrE,kFAAkF;wBAClF,yEAAyE;wBACzE,gFAAgF;wBAChF,sEAAsE;wBACtE,IAAI,CAAC,YAAY,kEACZ,IAAI,CAAC,YAAY;4BACpB;;gCACE,IAAI,CAAC,KAAI,CAAC,MAAM,IAAI,CAAC,CAAA,MAAA,KAAI,CAAC,WAAW,0CAAE,SAAS,CAAA,IAAI,CAAC,iBAAiB;oCAAE,OAAO;gCAC/E,IAAM,MAAM,GAAG,iBAAiB,CAAC,eAAe,EAAE,CAAC;gCACnD,IAAI,CAAC,MAAM,CAAC,MAAM;oCAAE,OAAO;gCAC3B,IAAM,QAAQ,GAAG,KAAI,CAAC,WAAW,EAAE,CAAC;gCACpC,IAAI,CAAC,QAAQ;oCAAE,OAAO;gCACtB,IAAI,KAAI,CAAC,yBAAyB,EAAE;oCAAE,OAAO;gCAC7C,iBAAiB,CAAC,gBAAgB,CAAC,UAAU,CAAC;oCAC5C,MAAM,QAAA;oCACN,SAAS,EAAE,KAAI,CAAC,WAAW,CAAC,SAAS;oCACrC,QAAQ,UAAA;oCACR,MAAM,EAAE,KAAI,CAAC,MAAM,CAAC,MAAM;oCAC1B,UAAU,EAAE,KAAI,CAAC,MAAM,CAAC,UAAU;iCACnC,CAAC,CAAC;4BACL,CAAC;iCACF,CAAC;wBAEF,qBAAM,IAAI,CAAC,0BAA0B,EAAE,EAAA;;wBAAvC,SAAuC,CAAC;wBAExC,0EAA0E;wBAC1E,yEAAyE;wBACzE,6EAA6E;wBAC7E,IAAI,MAAA,IAAA,+BAAc,GAAE,0CAAE,MAAM,EAAE;4BACtB,SAAS,GAAG,IAAA,2CAA0B,GAAE,CAAC;4BAC/C,IAAA,wCAAuB,EAAC,SAAS,CAAC,CAAC;4BACnC,SAAS,CAAC,KAAK,oBACb,MAAM,EAAE,IAAI,CAAC,cAAc,IACxB,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,IAAI,EAAE,QAAQ,EAAE,sCAAqB,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,EAAE,CAAC,EAC1F,CAAC;yBACJ;wBAED,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,+CAA+C,CAAC,CAAC;wBAEzE,IAAI,CAAC,sBAAsB,CAAC,KAAK,CAAC,CAAC;wBAEnC,qBAAM,IAAI,CAAC,2BAA2B,CACpC,EAAE,cAAc,EAAE,OAAO,CAAC,cAAc,EAAE,IAAI,EAAE,IAAI,CAAC,0BAA0B,EAAE,EAAE,EACnF,IAAI,CACL,EAAA;;wBAHD,SAGC,CAAC;wBAEI,gBAAgB,GAAG,IAAI,CAAC,MAAM,CAAC,eAAe,IAAI,CAAC,MAAA,MAAA,MAAA,IAAI,CAAC,MAAM,CAAC,aAAa,0CAAE,aAAa,0CAAE,MAAM,mCAAI,CAAC,CAAC,GAAG,CAAC,CAAC;wBACpH,IAAI,gBAAgB,EAAE;4BACpB,IAAI,CAAC,sBAAsB,EAAE,CAAC;yBAC/B;;;;;KACF;IAED,oCAAY,GAAZ,UAAa,SAA0B,EAAE,QAAiB;QACxD,OAAO,IAAA,8BAAa,EAAC,IAAI,CAAC,iBAAiB,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC,CAAC;IACpE,CAAC;IAEK,yCAAiB,GAAvB,UACE,SAA0B,EAC1B,QAAiB,EACjB,OAAqD;;;;;;;wBAErD,gFAAgF;wBAChF,IAAI,CAAC,oCAAoC,EAAE,CAAC;wBAC5C,IAAI,CAAC,qBAAqB,GAAG,KAAK,CAAC;wBACnC,IAAI,CAAC,wBAAwB,GAAG,SAAS,CAAC,CAAC,2CAA2C;wBAEhF,iBAAiB,GAAG,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC;wBACzE,IAAI,iBAAiB,EAAE;4BACrB,IAAI,CAAC,UAAU,CAAC,iBAAiB,CAAC,CAAC;yBACpC;wBAEK,eAAe,GAAG,iBAAiB,KAAK,SAAS,CAAC;wBAExD,kFAAkF;wBAClF,qFAAqF;wBACrF,oFAAoF;wBACpF,sFAAsF;wBACtF,uFAAuF;wBACvF,IAAI,eAAe,EAAE;4BACnB,MAAA,IAAI,CAAC,iBAAiB,0CAAE,uBAAuB,EAAE,CAAC;yBACnD;wBAEK,mBAAmB,GAAG,QAAQ,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;wBAC3D,IAAI,CAAC,WAAW,GAAG,IAAI,gCAAkB,CAAC;4BACxC,SAAS,EAAE,SAAS;4BACpB,QAAQ,EAAE,mBAAmB;yBAC9B,CAAC,CAAC;wBAEH,oFAAoF;wBACpF,wFAAwF;wBACxF,wFAAwF;wBACxF,uCAAuC;wBACvC,IAAI,eAAe,EAAE;4BACnB,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;4BACnC,IAAI,CAAC,mBAAmB,GAAG,CAAC,CAAC;4BAC7B,IAAI,CAAC,sBAAsB,GAAG,KAAK,CAAC;4BACpC,IAAI,MAAA,IAAI,CAAC,MAAM,0CAAE,MAAM,EAAE;gCACvB,IAAA,4CAAkB,EAAC,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,SAAS,EAAE,IAAI,CAAC,gBAAgB,EAAE,IAAI,CAAC,cAAc,CAAC,CAAC;gCAC9F,IAAI,iBAAiB,KAAK,SAAS,EAAE;oCACnC,IAAA,+CAAqB,EAAC,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,iBAAiB,EAAE,IAAI,CAAC,cAAc,CAAC,CAAC;iCACnF;6BACF;yBACF;6BAIG,CAAA,IAAI,CAAC,qBAAqB,IAAI,iBAAiB,CAAA,EAA/C,wBAA+C;wBACxB,qBAAM,IAAI,CAAC,qBAAqB,CAAC,oBAAoB,EAAE,EAAA;;wBAAxE,YAAY,GAAK,CAAA,SAAuD,CAAA,aAA5D;wBACpB,IAAI,CAAC,MAAM,GAAG,YAAY,CAAC;;;6BAGzB,CAAA,MAAA,IAAI,CAAC,MAAM,0CAAE,eAAe,CAAA,EAA5B,wBAA4B;wBAC9B,qBAAM,IAAI,CAAC,2BAA2B,CACpC,EAAE,cAAc,EAAE,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,cAAc,EAAE,IAAI,EAAE,IAAI,CAAC,0BAA0B,EAAE,EAAE,EACpF,KAAK,EACL,IAAI,CACL,EAAA;;wBAJD,SAIC,CAAC;;4BAEF,qBAAM,IAAI,CAAC,YAAY,EAAE,EAAA;;wBAAzB,SAAyB,CAAC;;;;;;KAE7B;IAED,kDAA0B,GAA1B;;QACE,IAAM,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC;QAC3B,IAAM,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC;QACrC,IAAI,CAAC,MAAM,IAAI,CAAC,WAAW,EAAE;YAC3B,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,gFAAgF,CAAC,CAAC;YAC3G,OAAO,EAAE,CAAC;SACX;QAED,IAAM,YAAY,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC;QAC5C,IAAI,eAAe,GAAqC,EAAE,CAAC;QAE3D,IAAI,YAAY,EAAE;YAChB,eAAe;gBACb,GAAC,2CAA+B,IAAG,WAAW,CAAC,eAAe,CAAC,CAAC,CAAC,WAAW,CAAC,eAAe,CAAC,CAAC,CAAC,IAAI;mBACpG,CAAC;YACF,IAAI,MAAM,CAAC,SAAS,EAAE;gBACpB,eAAe,CAAC,yCAA6B,CAAC,GAAG,IAAI,CAAC,SAAS,CAAC;oBAC9D,OAAO,EAAE,IAAA,iCAAgB,EAAC,MAAM,CAAC,MAAM,CAAC,CAAC,QAAQ,EAAE;iBACpD,CAAC,CAAC;aACJ;SACF;QAED,KAAK,IAAI,CAAC,mBAAmB,CAC3B,4BAAgB,CAAC,YAAY,EAC7B;YACE,YAAY,cAAA;YACZ,eAAe,EAAE,eAAe;SACjC,EACD,IAAI,CAAC,UAAU,KAAK,EAAE,CACvB,CAAC;QACF,IAAI,IAAI,CAAC,UAAU,KAAK,EAAE,EAAE;YAC1B,IAAI,CAAC,UAAU,GAAG,CAAC,CAAC;SACrB;QACD,IAAI,CAAC,UAAU,EAAE,CAAC;QAElB,OAAO,eAAe,CAAC;IACzB,CAAC;IAiID,kCAAU,GAAV,UAAW,SAA2B;;QACpC,IAAM,eAAe,GAAG,SAAS,KAAI,MAAA,IAAI,CAAC,WAAW,0CAAE,SAAS,CAAA,CAAC;QACjE,IAAM,QAAQ,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QACpC,IAAI,IAAI,CAAC,aAAa,IAAI,eAAe,IAAI,QAAQ,EAAE;YACrD,IAAI,IAAI,CAAC,yBAAyB,EAAE,EAAE;gBACpC,+EAA+E;gBAC/E,qDAAqD;gBACrD,IAAM,SAAS,GAAG,IAAI,CAAC,gBAA0B,CAAC;gBAClD,IAAM,KAAK,GAAI,IAAI,CAAC,MAAoC,CAAC,oBAA8B,CAAC;gBACxF,IAAI,CAAC,cAAc,CAAC,GAAG,CACrB,kBAAW,eAAe,iCAAuB,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,iCAAuB,KAAK,QAAK,CACzG,CAAC;gBACF,8EAA8E;gBAC9E,gFAAgF;gBAChF,2EAA2E;gBAC3E,+EAA+E;gBAC/E,2EAA2E;gBAC3E,IAAI,CAAC,mBAAmB,EAAE,CAAC;gBAC3B,OAAO;aACR;YACD,iFAAiF;YACjF,gFAAgF;YAChF,mEAAmE;YACnE,EAAE;YACF,0EAA0E;YAC1E,iFAAiF;YACjF,kFAAkF;YAClF,kFAAkF;YAClF,sCAAsC;YACtC,IACE,CAAC,IAAI,CAAC,sBAAsB;gBAC5B,CAAA,MAAA,IAAI,CAAC,MAAM,0CAAE,oBAAoB,MAAK,SAAS;gBAC/C,IAAI,CAAC,oBAAoB;gBACzB,IAAI,CAAC,cAAc,EACnB;gBACA,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC;gBACnC,IAAM,SAAS,GAAG,IAAI,CAAC,gBAAgB,KAAK,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC,SAAS,CAAC;gBACvG,KAAK,IAAI,CAAC,mBAAmB,CAC3B,4BAAgB,CAAC,oBAAoB,EACrC;oBACE,SAAS,EAAE,eAAe;oBAC1B,mBAAmB,EAAE,IAAI,CAAC,mBAAmB;oBAC7C,SAAS,WAAA;oBACT,oBAAoB,EAAE,IAAI,CAAC,MAAM,CAAC,oBAAoB;iBACvD,EACD,KAAK,CACN,CAAC;aACH;YACD,IAAI,CAAC,aAAa,CAAC,yBAAyB,CAAC,EAAE,SAAS,EAAE,eAAe,EAAE,QAAQ,UAAA,EAAE,CAAC,CAAC;SACxF;IACH,CAAC;IAEK,kCAAU,GAAhB,UAAiB,sBAA8B;;QAA9B,uCAAA,EAAA,8BAA8B;;;;gBAC7C,IAAI,CAAC,CAAA,MAAA,IAAI,CAAC,WAAW,0CAAE,SAAS,CAAA,EAAE;oBAChC,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,0DAA0D,CAAC,CAAC;oBACpF,sBAAO,OAAO,CAAC,OAAO,EAAE,EAAC;iBAC1B;gBAEK,QAAQ,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;gBACpC,IAAI,CAAC,QAAQ,EAAE;oBACb,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,yDAAyD,CAAC,CAAC;oBACnF,sBAAO,OAAO,CAAC,OAAO,EAAE,EAAC;iBAC1B;gBACD,IAAI,CAAC,aAAa,IAAI,sBAAsB,IAAI,KAAK,IAAI,CAAC,aAAa,CAAC,gBAAgB,CAAC,EAAE,QAAQ,UAAA,EAAE,CAAC,CAAC;gBAEvG,sBAAO,IAAI,CAAC,YAAY,EAAE,EAAC;;;KAC5B;IAED,oCAAY,GAAZ;;QACE,IAAI,mBAAwC,CAAC;QAC7C,IAAI,MAAA,IAAI,CAAC,MAAM,0CAAE,YAAY,EAAE;YAC7B,IAAM,aAAa,GAAG,IAAA,sCAAqB,EAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,aAAa,CAAC;YACpF,mBAAmB,GAAG,aAAa,CAAC,WAAW,EAAE,CAAC,MAAM,CAAC;SAC1D;QAED,OAAO,mBAAmB,KAAK,SAAS,CAAC,CAAC,CAAC,mBAAmB,CAAC,CAAC,CAAC,MAAA,IAAI,CAAC,MAAM,0CAAE,MAAM,CAAC;IACvF,CAAC;IAED,uCAAe,GAAf;QACE,IAAI,CAAC,IAAI,CAAC,WAAW,IAAI,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC,SAAS,EAAE;YACpE,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,sFAAsF,CAAC,CAAC;YACjH,OAAO,KAAK,CAAC;SACd;QAED,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE;YAC/B,IAAI,CAAC,cAAc,CAAC,GAAG,CACrB,kBAAW,IAAI,CAAC,WAAW,CAAC,SAAS,qHAAkH,CACxJ,CAAC;YACF,OAAO,KAAK,CAAC;SACd;QAED,IAAI,IAAI,CAAC,YAAY,EAAE,EAAE;YACvB,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,yBAAkB,IAAI,CAAC,WAAW,CAAC,SAAS,4CAAyC,CAAC,CAAC;YAC/G,OAAO,KAAK,CAAC;SACd;QAED,IAAI,YAAY,GAAG,KAAK,CAAC;QACzB,IAAI,OAAO,GAAG,EAAE,CAAC;QACjB,IAAI,OAAO,GAAG,KAAK,CAAC;QAEpB,gGAAgG;QAChG,wEAAwE;QACxE,IAAI,IAAI,CAAC,MAAM,CAAC,eAAe,EAAE;YAC/B,IAAI,CAAC,IAAI,CAAC,qBAAqB,EAAE;gBAC/B,OAAO,GAAG,4CAAqC,IAAI,CAAC,WAAW,CAAC,SAAS,+CAA4C,CAAC;gBACtH,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;gBACjC,YAAY,GAAG,KAAK,CAAC;gBACrB,OAAO,GAAG,KAAK,CAAC;aACjB;iBAAM;gBACL,OAAO,GAAG,wCAAiC,IAAI,CAAC,WAAW,CAAC,SAAS,2CAAwC,CAAC;gBAC9G,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;gBACjC,YAAY,GAAG,IAAI,CAAC;gBACpB,OAAO,GAAG,IAAI,CAAC;aAChB;SACF;aAAM;YACL,IAAM,UAAU,GAAG,IAAA,4BAAiB,EAAC,IAAI,CAAC,WAAW,CAAC,SAAS,EAAE,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC;YACzF,IAAI,CAAC,UAAU,EAAE;gBACf,OAAO,GAAG,yBAAkB,IAAI,CAAC,WAAW,CAAC,SAAS,0CAAuC,CAAC;gBAC9F,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;gBACjC,YAAY,GAAG,KAAK,CAAC;gBACrB,OAAO,GAAG,KAAK,CAAC;aACjB;iBAAM;gBACL,YAAY,GAAG,IAAI,CAAC;gBACpB,OAAO,GAAG,IAAI,CAAC;aAChB;SACF;QAED,gFAAgF;QAChF,IAAI,IAAI,CAAC,wBAAwB,KAAK,YAAY,IAAI,IAAI,CAAC,MAAM,CAAC,eAAe,EAAE;YACjF,KAAK,IAAI,CAAC,mBAAmB,CAAC,4BAAgB,CAAC,kBAAkB,EAAE;gBACjE,OAAO,SAAA;gBACP,SAAS,EAAE,IAAI,CAAC,WAAW,CAAC,SAAS;gBACrC,OAAO,SAAA;gBACP,eAAe,EAAE,IAAI,CAAC,mBAAmB;aAC1C,CAAC,CAAC;YACH,IAAI,CAAC,wBAAwB,GAAG,YAAY,CAAC;SAC9C;QAED,OAAO,YAAY,CAAC;IACtB,CAAC;IAED,yCAAiB,GAAjB;;QACE,0FAA0F;QAC1F,4DAA4D;QAC5D,6DAA6D;QAC7D,IAAM,aAAa,GAAG,MAAA,MAAA,MAAA,IAAI,CAAC,MAAM,0CAAE,aAAa,0CAAE,aAAa,mCAAI,EAAE,CAAC;QACtE,IAAI,aAAa,CAAC,MAAM,KAAK,CAAC,EAAE;YAC9B,OAAO,SAAS,CAAC;SAClB;QACD,OAAO,aAAa,CAAC;IACvB,CAAC;IAED,4CAAoB,GAApB;;QACE,IAAM,aAAa,GAAG,MAAA,IAAI,CAAC,MAAM,0CAAE,aAAa,CAAC;QACjD,IAAM,cAAc,GAAG,aAAa,CAAC,CAAC,CAAC,IAAA,+BAAqB,EAAC,IAAI,CAAC,cAAc,EAAE,aAAa,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;QAE7G,IAAI,cAAc,KAAK,cAAc,EAAE;YACrC,OAAO,GAAG,CAAC;SACZ;QAED,6EAA6E;QAC7E,wEAAwE;QACxE,2EAA2E;QAC3E,yEAAyE;QACzE,IAAI,MAAA,aAAa,aAAb,aAAa,uBAAb,aAAa,CAAE,aAAa,0CAAE,IAAI,CAAC,UAAC,IAAI,IAAK,OAAA,IAAI,CAAC,SAAS,KAAK,cAAc,EAAjC,CAAiC,CAAC,EAAE;YACnF,OAAO,GAAG,CAAC;SACZ;QAED,uFAAuF;QACvF,mFAAmF;QACnF,yEAAyE;QACzE,IAAM,aAAa,GAAG,aAAa,aAAb,aAAa,uBAAb,aAAa,CAAE,aAAa,CAAC;QACnD,IAAI,CAAA,aAAa,aAAb,aAAa,uBAAb,aAAa,CAAE,gBAAgB,MAAK,cAAc,IAAI,aAAa,IAAI,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE;YACnG,OAAO,GAAG,CAAC;SACZ;QAED,IAAM,YAAY,GAAG,aAAa,aAAb,aAAa,uBAAb,aAAa,CAAE,YAAY,CAAC;QACjD,IAAI,CAAC,YAAY,EAAE;YACjB,OAAO;SACR;QAED,OAAO,YAAiC,CAAC;IAC3C,CAAC;IAEK,2CAAmB,GAAzB,UAA0B,aAAwC;;;;;;;wBAC1D,OAAO,GAAG,EAAE,CAAC;wBAEnB,0BAA0B;wBAC1B,IAAI;4BACI,iBAAiB,GAAG,IAAA,6CAAuB,EAAC;gCAChD,cAAc,EAAE,CAAA,MAAA,MAAA,IAAI,CAAC,MAAM,0CAAE,iBAAiB,0CAAE,cAAc,KAAI,EAAE;gCACpE,aAAa,EAAE,CAAA,MAAA,IAAI,CAAC,MAAM,0CAAE,sBAAsB,KAAI,KAAK;gCAC3D,eAAe,EAAE,MAAA,IAAI,CAAC,MAAM,0CAAE,wBAAwB;gCACtD,oBAAoB,EAAE,MAAA,IAAI,CAAC,MAAM,0CAAE,oBAAoB;6BACxD,CAAC,CAAC;4BAEH,OAAO,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;yBACjC;wBAAC,OAAO,KAAK,EAAE;4BACd,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,uCAAuC,EAAE,KAAK,CAAC,CAAC;yBAC1E;6BAaG,CAAA,MAAA,aAAa,aAAb,aAAa,uBAAb,aAAa,CAAE,OAAO,0CAAE,OAAO,CAAA,EAA/B,wBAA+B;;;;wBAGI,8FAAa,wCAAwC,QAAC;;wBAAjF,sBAAsB,GAAK,CAAA,SAAsD,CAAA,uBAA3D;wBAC9B,OAAO,CAAC,IAAI,CAAC,sBAAsB,CAAC,EAAE,KAAK,EAAE,aAAa,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;;;;wBAE9E,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,gCAAgC,EAAE,OAAK,CAAC,CAAC;;4BAItE,sBAAO,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS,EAAC;;;;KACjD;IAEa,yCAAiB,GAA/B;;;;;;wBACE,IAAI,IAAI,CAAC,cAAc,EAAE;4BACvB,sBAAO,IAAI,CAAC,cAAc,EAAC;yBAC5B;;;;wBAGoB,8FAAa,yBAAyB,QAAC;;wBAAlD,MAAM,GAAK,CAAA,SAAuC,CAAA,OAA5C;wBACd,IAAI,CAAC,cAAc,GAAG,MAAM,CAAC;wBAC7B,sBAAO,MAAM,EAAC;;;wBAEd,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,qCAAqC,EAAE,OAAK,CAAC,CAAC;wBACvE,sBAAO,IAAI,EAAC;;;;;KAEf;IAEK,oCAAY,GAAlB,UAAmB,iBAAwB;QAAxB,kCAAA,EAAA,wBAAwB;;;;;;wBACzC,IAAI,IAAI,CAAC,oBAAoB,EAAE;4BAC7B,IAAI,CAAC,oCAAoC,GAAG,iBAAiB,CAAC;4BAC9D,sBAAO;yBACR;wBACD,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC;;;;wBAE/B,qBAAM,IAAI,CAAC,aAAa,CAAC,iBAAiB,CAAC,EAAA;;wBAA3C,SAA2C,CAAC;;;6BACrC,CAAA,IAAI,CAAC,oCAAoC,KAAK,IAAI,CAAA;wBACjD,WAAW,GAAG,IAAI,CAAC,oCAAoC,CAAC;wBAC9D,IAAI,CAAC,oCAAoC,GAAG,IAAI,CAAC;wBACjD,qBAAM,IAAI,CAAC,aAAa,CAAC,WAAW,CAAC,EAAA;;wBAArC,SAAqC,CAAC;;;;wBAGxC,IAAI,CAAC,oBAAoB,GAAG,KAAK,CAAC;wBAClC,IAAI,CAAC,oCAAoC,GAAG,IAAI,CAAC;;;;;;KAEpD;IAEa,qCAAa,GAA3B,UAA4B,iBAAwB;;QAAxB,kCAAA,EAAA,wBAAwB;;;;;;;;wBAC5C,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC;wBACrB,YAAY,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC;wBACtC,SAAS,GAAG,MAAA,IAAI,CAAC,WAAW,0CAAE,SAAS,CAAC;wBAC9C,IAAI,CAAC,YAAY,IAAI,CAAC,SAAS,IAAI,CAAC,MAAM,EAAE;4BAC1C,sBAAO;yBACR;wBACD,IAAI,CAAC,mBAAmB,EAAE,CAAC;wBAEJ,qBAAM,IAAI,CAAC,iBAAiB,EAAE,EAAA;;wBAA/C,cAAc,GAAG,SAA8B;wBAErD,iDAAiD;wBACjD,IAAI,CAAC,cAAc,EAAE;4BACnB,sBAAO;yBACR;wBAED,qBAAM,IAAI,CAAC,0BAA0B,EAAE,EAAA;;wBAAvC,SAAuC,CAAC;wBAElC,oBAAoB,GAAG,MAAA,MAAM,CAAC,aAAa,0CAAE,OAAO,CAAC;wBACrD,QAAQ,GAAG,IAAA,sBAAY,EAAC,MAAM,CAAC,UAAU,EAAE,MAAM,CAAC,cAAc,CAAC,CAAC;wBAClE,WAAW,GAAG,CAAC,qCAAyB,EAAE,iCAAqB,EAAE,sCAA0B,EAAE,QAAQ,CAAC,CAAC;wBAC7G,MAAA,IAAI,CAAC,gBAAgB,0CAAE,KAAK,CAAC,UAAC,KAA0B;4BACtD,IAAI,WAAW,CAAC,IAAI,CAAC,UAAC,GAAG,IAAK,OAAA,KAAK,CAAC,GAAG,CAAC,UAAU,CAAC,GAAG,CAAC,EAAzB,CAAyB,CAAC;gCAAE,OAAO;4BACjE,KAAK,KAAI,CAAC,mBAAmB,CAAC,4BAAgB,CAAC,aAAa,EAAE,KAAK,CAAC,CAAC;wBACvE,CAAC,EAAE,oBAAoB,CAAC,CAAC;wBACjB,iBAAiB,GAAoB,MAAM,kBAA1B,EAAE,aAAa,GAAK,MAAM,cAAX,CAAY;wBAE9C,KAAK,GAAG,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,OAAO;4BACtC,CAAC,CAAC;gCACE,gBAAgB,EACd,IAAI,CAAC,aAAa;qCAClB,MAAA,IAAI,CAAC,YAAY,0CAAE,UAAU,CAAC;wCAC5B,aAAa,EAAE,IAAI,CAAC,aAAa;wCACjC,SAAS,WAAA;wCACT,UAAU,EAAE,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC;wCACvC,MAAM,EAAE,cAAc,CAAC,MAAM;wCAC7B,cAAc,EAAE,MAAA,iBAAiB,CAAC,cAAc,mCAAI,EAAE;wCACtD,kBAAkB,EAAE,MAAA,MAAM,CAAC,iBAAiB,0CAAE,WAAW;qCAC1D,CAAC,CAAA;gCACJ,MAAM,EAAE,IAAI,CAAC,UAAU;6BACxB;4BACH,CAAC,CAAC,EAAE,CAAC;wBAED,cAAc,GAClB,CAAA,iBAAiB,aAAjB,iBAAiB,uBAAjB,iBAAiB,CAAE,OAAO,KAAI,iBAAiB,CAAC,cAAc,CAAC,CAAC,CAAC,iBAAiB,CAAC,cAAc,CAAC,CAAC,CAAC,EAAE,CAAC;wBAEzG,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,+CAAwC,SAAS,MAAG,CAAC,CAAC;;;;wBAGtE,yBAAyB,GAAG,CAAC,CAAC,CAAA,MAAA,MAAM,CAAC,kBAAkB,0CAAE,OAAO,CAAA,CAAC;wBACjE,kBAAkB,GAAG,CAAA,MAAA,MAAM,CAAC,kBAAkB,0CAAE,kBAAkB,MAAK,KAAK,CAAC;wBAC7E,SAAS,GAAG,yBAAyB,IAAI,IAAA,iCAAU,GAAE,CAAC;wBAE5D,IAAI,SAAS,IAAI,kBAAkB,EAAE;4BACnC,yEAAyE;4BACzE,qFAAqF;4BACrF,IAAI,CAAC,8BAA8B,GAAG,IAAA,6CAAsB,EAAC;gCAC3D,OAAO,EAAE,cAAM,OAAA,KAAI,CAAC,wBAAwB,CAAC,cAAc,EAAE,SAAS,EAAE,MAAM,EAAE,KAAK,CAAC,EAAvE,CAAuE;gCACtF,MAAM,EAAE;oCACN,IAAI;wCACF,4EAA4E;wCAC5E,2EAA2E;wCAC3E,mDAAmD;wCACnD,KAAI,CAAC,oBAAoB,IAAI,KAAI,CAAC,oBAAoB,EAAE,CAAC;wCACzD,KAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC;qCAClC;oCAAC,OAAO,KAAK,EAAE;wCACd,IAAM,UAAU,GAAG,KAAc,CAAC;wCAClC,KAAI,CAAC,cAAc,CAAC,IAAI,CACtB,qEAA8D,UAAU,CAAC,QAAQ,EAAE,CAAE,CACtF,CAAC;qCACH;gCACH,CAAC;6BACF,CAAC,CAAC;4BACH,sBAAO;yBACR;wBAED,KAAA,IAAI,CAAA;wBAAwB,KAAA,cAAc,CAAA;mDACrC,IAAI,CAAC,uBAAuB,CAC7B,MAAM,EACN,KAAK,EACL,UAAC,KAAoB;gCACnB,IAAI,KAAI,CAAC,YAAY,EAAE,EAAE;oCACvB,KAAI,CAAC,cAAc,CAAC,GAAG,CAAC,yBAAkB,SAAS,4CAAyC,CAAC,CAAC;oCAC9F,KAAI,CAAC,mBAAmB,EAAE,CAAC;oCAC3B,KAAI,CAAC,UAAU,EAAE,CAAC;oCAClB,OAAO;iCACR;gCAED,IAAI,KAAK,CAAC,IAAI,KAAK,uBAAc,CAAC,IAAI,EAAE;oCACtC,KAAK,CAAC,IAAI,CAAC,IAAI,GAAG,IAAA,oBAAU,EAAC,KAAK,CAAC,IAAI,CAAC,IAAI,EAAE,cAAc,CAAC,CAAC;iCAC/D;gCAED,IAAI,KAAI,CAAC,eAAe,EAAE;oCACxB,mFAAmF;oCACnF,KAAI,CAAC,eAAe,CAAC,YAAY,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;iCACrD;qCAAM;oCACL,qEAAqE;oCACrE,4EAA4E;oCAC5E,KAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,EAAE,KAAK,OAAA,EAAE,SAAS,WAAA,EAAE,CAAC,CAAC;iCACnD;4BACH,CAAC,EACD,gCAAgC,CACjC;;wBACQ,qBAAM,IAAI,CAAC,mBAAmB,CAAC,aAAa,CAAC,EAAA;;wBA3BxD,GAAK,oBAAoB,GAAG,6DA2B1B,UAAO,GAAE,SAA6C,EACtD,2BAAwB,GAAE,yBAAyB,UACnD,CAAC;wBAEH,IAAI,yBAAyB,IAAI,CAAC,SAAS,IAAI,kBAAkB,EAAE;4BACjE,IAAI,CAAC,IAAI,CAAC,4BAA4B,EAAE;gCACtC,IAAI,CAAC,4BAA4B,GAAG,IAAI,mDAA4B,EAAE,CAAC;6BACxE;4BACD,IAAI,CAAC,4BAA4B,CAAC,KAAK,EAAE,CAAC;yBAC3C;wBAED,KAAK,IAAI,CAAC,mBAAmB,CAAC,4BAAgB,CAAC,UAAU,CAAC,CAAC;wBAC3D,IAAI,iBAAiB,EAAE;4BACrB,KAAK,IAAI,CAAC,mBAAmB,CAAC,4BAAgB,CAAC,QAAQ,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;yBACzE;;;;wBAED,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,sCAAsC,EAAE,OAAK,CAAC,CAAC;;;;;;KAE3E;IAEO,+CAAuB,GAA/B,UACE,MAAiC,EACjC,KAA0D,EAC1D,IAAoC,EACpC,cAAsB;QAJxB,iBAqDC;QA/CS,IAAA,aAAa,GAAK,MAAM,cAAX,CAAY;QACjC,2CACE,IAAI,MAAA,EACJ,gBAAgB,EAAE,MAAM,CAAC,sBAAsB,EAC/C,KAAK,OAAA,EACL,aAAa,EAAE,IAAI,EACnB,aAAa,EAAE,2BAAe,EAC9B,UAAU,EAAE,uBAAW,EACvB,aAAa,EAAE,IAAI,CAAC,iBAAiB,EAAwB,EAC7D,qCAAqC,EAAE,MAAM,CAAC,qCAAqC,EACnF,WAAW,EAAE,IAAA,gBAAM,EAAC,OAAO,EAAE,aAAa,EAAE,cAAM,OAAA,KAAI,CAAC,cAAc,EAAnB,CAAmB,CAAC,EACtE,UAAU,EAAE,IAAA,gBAAM,EAAC,MAAM,EAAE,aAAa,EAAE,cAAM,OAAA,KAAI,CAAC,cAAc,EAAnB,CAAmB,CAAC,EACpE,eAAe,EAAE,IAAA,yBAAe,EAAC,aAAa,EAAE,cAAM,OAAA,KAAI,CAAC,cAAc,EAAnB,CAAmB,CAAC,EAC1E,gBAAgB,EAAE,IAAI,CAAC,oBAAoB,EAAE,IAC1C,CAAC,MAAM,CAAC,sBAAsB,KAAK,SAAS,IAAI,EAAE,gBAAgB,EAAE,MAAM,CAAC,sBAAsB,EAAE,CAAC,KACvG,YAAY,EAAE,KAAK,EACnB,yBAAyB,EAAE,MAAM,CAAC,yBAAyB;YAC3D,kEAAkE;YAClE,6EAA6E;YAC7E,cAAc,EAAE;gBACd,MAAM,EAAE,IAAI;gBACZ,OAAO,EAAE,IAAI;gBACb,WAAW,EAAE,IAAI;gBACjB,cAAc,EAAE,IAAI;gBACpB,oBAAoB,EAAE,IAAI;gBAC1B,cAAc,EAAE,IAAI;gBACpB,cAAc,EAAE,IAAI;gBACpB,iBAAiB,EAAE,IAAI;gBACvB,kBAAkB,EAAE,IAAI;gBACxB,oBAAoB,EAAE,IAAI;aAC3B,EACD,YAAY,EAAE,UAAC,KAAc;gBAC3B,IAAM,UAAU,GAAG,KAAyC,CAAC;gBAC7D,wGAAwG;gBACxG,IAAI,UAAU,CAAC,OAAO,CAAC,QAAQ,CAAC,YAAY,CAAC,IAAI,UAAU,CAAC,OAAO,CAAC,QAAQ,CAAC,eAAe,CAAC,EAAE;oBAC7F,MAAM,UAAU,CAAC;iBAClB;gBACD,oGAAoG;gBACpG,gFAAgF;gBAChF,IAAI,UAAU,CAAC,UAAU,EAAE;oBACzB,MAAM,UAAU,CAAC;iBAClB;gBACD,KAAI,CAAC,cAAc,CAAC,IAAI,CAAC,cAAc,EAAE,UAAU,CAAC,QAAQ,EAAE,CAAC,CAAC;gBAChE,kFAAkF;gBAClF,OAAO,IAAI,CAAC;YACd,CAAC,IACD;IACJ,CAAC;IAEO,gDAAwB,GAAhC,UACE,cAA8B,EAC9B,SAA0B,EAC1B,MAAiC,EACjC,KAA0D;QAE1D,8EAA8E;QAC9E,kFAAkF;QAClF,oFAAoF;QACpF,qFAAqF;QACrF,gEAAgE;QAChE,IAAI;YACF,kFAAkF;YAClF,mFAAmF;YACnF,mFAAmF;YACnF,IAAI,CAAC,oBAAoB,IAAI,IAAI,CAAC,oBAAoB,EAAE,CAAC;YACzD,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC;YACjC,IAAI,CAAC,oBAAoB,GAAG,cAAc,uCACrC,IAAI,CAAC,uBAAuB,CAC7B,MAAM,EACN,KAAK,EACL;gBACE,uEAAuE;YACzE,CAAC,EACD,+CAA+C,CAChD,KACD,wBAAwB,EAAE,IAAI,IAC9B,CAAC;YACH,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,oEAA6D,SAAS,MAAG,CAAC,CAAC;SACpG;QAAC,OAAO,KAAK,EAAE;YACd,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,2DAA2D,EAAE,KAAK,CAAC,CAAC;SAC9F;IACH,CAAC;IA0DD,mCAAW,GAAX;;QACE,OAAO,MAAA,IAAI,CAAC,WAAW,0CAAE,QAAQ,CAAC;IACpC,CAAC;IAED,oCAAY,GAAZ;;QACE,OAAO,MAAA,IAAI,CAAC,WAAW,0CAAE,SAAS,CAAC;IACrC,CAAC;IAEK,6BAAK,GAAX,UAAY,QAAgB;;QAAhB,yBAAA,EAAA,gBAAgB;;;gBAC1B,gFAAgF;gBAChF,gFAAgF;gBAChF,mFAAmF;gBACnF,iFAAiF;gBACjF,8CAA8C;gBAC9C,sBAAO,MAAA,IAAI,CAAC,aAAa,0CAAE,KAAK,CAAC,QAAQ,CAAC,EAAC;;;KAC5C;IAED,gCAAQ,GAAR;;QACE,MAAA,IAAI,CAAC,gBAAgB,oDAAI,CAAC;QAC1B,MAAA,IAAI,CAAC,8BAA8B,oDAAI,CAAC;QACxC,IAAI,CAAC,8BAA8B,GAAG,IAAI,CAAC;QAC3C,IAAI,CAAC,sBAAsB,CAAC,IAAI,CAAC,CAAC;QAClC,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAC3B,IAAI,CAAC,UAAU,EAAE,CAAC;IACpB,CAAC;IAEO,kCAAU,GAAlB,UAAmB,OAA2B;QAC5C,IAAI,OAAO,KAAK,QAAQ,EAAE;YACxB,OAAO,0CAA0C,CAAC;SACnD;QAED,IAAI,OAAO,KAAK,SAAS,EAAE;YACzB,OAAO,0CAA0C,CAAC;SACnD;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAEO,mCAAW,GAAnB,UACE,SAAsC,EACtC,YAAuC,EACvC,WAAqC,EACrC,YAAmD,EACnD,gBAAoC,EACpC,oBAAwC,EACxC,OAA2B;QAE3B,IAAM,SAAS,GAAG,CAAA,SAAS,aAAT,SAAS,uBAAT,SAAS,CAAE,QAAQ,EAAE,EAAC,CAAC,CAAC,IAAA,iCAAgB,EAAC,SAAS,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;QAE7F,IAAI,CAAC,QAAQ,GAAG;YACd,YAAY,cAAA;YACZ,WAAW,aAAA;YACX,YAAY,cAAA;YACZ,SAAS,WAAA;YACT,SAAS,WAAA;YACT,UAAU,EAAE,YAAY,CAAC,UAAU;YACnC,aAAa,EAAE,IAAI,CAAC,UAAU,CAAC,OAAO,CAAC;YACvC,gBAAgB,kBAAA;YAChB,iBAAiB,EAAE,mCAAmC;YACtD,oBAAoB,sBAAA;SACrB,CAAC;IACJ,CAAC;IAEa,kDAA0B,GAAxC;;;;;;;6BACM,CAAA,CAAA,MAAA,MAAA,MAAA,IAAI,CAAC,MAAM,0CAAE,aAAa,0CAAE,OAAO,0CAAE,OAAO,KAAI,CAAC,IAAI,CAAC,gBAAgB,CAAA,EAAtE,wBAAsE;;;;wBAElB,8FAAa,aAAa,QAAC;;wBAArD,qBAAqB,GAAK,CAAA,SAA2B,CAAA,iBAAhC;wBAC/C,IAAI,CAAC,gBAAgB,GAAG,IAAI,qBAAqB,EAAE,CAAC;;;;wBAEpD,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,mDAAmD,EAAE,OAAK,CAAC,CAAC;;;;;;KAG1F;IACH,oBAAC;AAAD,CAAC,AAxoCD,IAwoCC;AAxoCY,sCAAa","sourcesContent":["import {\n getAnalyticsConnector,\n getGlobalScope,\n ILogger,\n Logger,\n LogLevel,\n returnWrapper,\n SpecialEventType,\n generateHashCode,\n getOrCreateWindowMessenger,\n enableBackgroundCapture,\n AMPLITUDE_ORIGINS_MAP,\n} from '@amplitude/analytics-core';\n\n// Import only specific types to avoid pulling in the entire rrweb-types package\nimport { eventWithTime, EventType as RRWebEventType, scrollCallback } from '@amplitude/rrweb-types';\nimport { createSessionReplayJoinedConfigGenerator } from './config/joined-config';\nimport {\n LoggingConfig,\n SessionReplayJoinedConfig,\n SessionReplayJoinedConfigGenerator,\n SessionReplayLocalConfig,\n SessionReplayMetadata,\n SessionReplayRemoteConfig,\n} from './config/types';\nimport {\n BLOCK_CLASS,\n CustomRRwebEvent,\n DEFAULT_SESSION_REPLAY_PROPERTY,\n INTERACTION_MAX_INTERVAL,\n INTERACTION_MIN_INTERVAL,\n MASK_TEXT_CLASS,\n SESSION_REPLAY_DEBUG_PROPERTY,\n SESSION_REPLAY_EU_URL,\n SESSION_REPLAY_SERVER_URL,\n SESSION_REPLAY_STAGING_URL,\n} from './constants';\nimport {\n getServerUrl,\n getDebugConfig,\n getEffectiveMaskLevel,\n getPageUrl,\n getStorageSize,\n getCurrentUrl,\n maskFn,\n maskAttributeFn,\n} from './helpers';\nimport { EventCompressor } from './events/event-compressor';\nimport { createEventsManager, EventsManagerWithBeacon } from './events/events-manager';\nimport { MultiEventManager } from './events/multi-manager';\nimport { clickBatcher, ClickHandler, clickNonBatcher } from './hooks/click';\nimport { ScrollWatcher } from './hooks/scroll';\nimport { SessionIdentifiers } from './identifiers';\nimport { SafeLoggerProvider } from './logger';\nimport {\n getOrInitReplayStartTime,\n pruneStaleReplayStartTimes,\n removeReplayStartTime,\n setReplayStartTime,\n} from './replay-start-time-store';\nimport { evaluateTargetingAndStore } from './targeting/targeting-manager';\nimport {\n AmplitudeSessionReplay,\n SessionReplayEventsManager as AmplitudeSessionReplayEventsManager,\n DebugInfo,\n EventsManagerWithType,\n EventType,\n SessionIdentifiers as ISessionIdentifiers,\n SessionReplayOptions,\n SessionReplayTargetingInput,\n} from './typings/session-replay';\nimport { isSessionInSample } from './sampling';\nimport { VERSION } from './version';\n\n// Import only the type for NetworkRequestEvent to keep type safety\nimport type { NetworkObservers, NetworkRequestEvent } from './observers';\nimport { createUrlTrackingPlugin, subscribeToUrlChanges } from './plugins/url-tracking-plugin';\nimport type { RecordFunction } from './utils/rrweb';\nimport { isInIframe, CrossOriginIframeCoordinator, listenForParentSignals } from './cross-origin-iframes';\n\ntype PageLeaveFn = (e: PageTransitionEvent | Event) => void;\n\nexport class SessionReplay implements AmplitudeSessionReplay {\n name = '@amplitude/session-replay-browser';\n config: SessionReplayJoinedConfig | undefined;\n joinedConfigGenerator: SessionReplayJoinedConfigGenerator | undefined;\n identifiers: ISessionIdentifiers | undefined;\n eventsManager?: AmplitudeSessionReplayEventsManager<'replay' | 'interaction', string>;\n loggerProvider: ILogger;\n recordCancelCallback: ReturnType<RecordFunction> | null = null;\n eventCount = 0;\n eventCompressor: EventCompressor | undefined;\n sessionTargetingMatch = false;\n private lastTargetingParams?: SessionReplayTargetingInput;\n private lastShouldRecordDecision?: boolean;\n\n // Public on purpose. `pageLeaveFns` is iterated by `pageLeaveListener`,\n // `rrwebEventManager` is dereferenced in `asyncSetSessionId` to drop the beacon buffer\n // at a session boundary, and `sessionStartTime` drives `isBelowMinSessionDuration()`.\n // Tests also stub/inspect these — privatizing them would break both production callers\n // and the gate's test coverage.\n pageLeaveFns: PageLeaveFn[] = [];\n sessionStartTime: number | undefined;\n rrwebEventManager: EventsManagerWithBeacon<'replay'> | undefined;\n /**\n * Count of sendEvents() calls suppressed by the min-session-duration gate for the\n * current session. Drives the REPLAY_GATE_DECISION rrweb event on first send-after-pass.\n */\n private suppressedSendCount = 0;\n /** True once REPLAY_GATE_DECISION has been emitted for the current session. */\n private hasEmittedGateDecision = false;\n private scrollHook?: scrollCallback;\n private clickHandler?: ClickHandler;\n private networkObservers?: NetworkObservers;\n private metadata: SessionReplayMetadata | undefined;\n\n // Cache the dynamically imported record function\n private recordFunction: RecordFunction | null = null;\n private recordEventsInFlight = false;\n private pendingEmitEvents: Array<{ event: eventWithTime; sessionId: string | number }> = [];\n\n /** Current page URL, kept in sync with SPA navigations for URL-based masking */\n private currentPageUrl = '';\n\n private recordEventsPendingShouldLogMetadata: boolean | null = null;\n\n /** Cleanup for URL change listener used to re-evaluate targeting on SPA route changes */\n private urlChangeCleanup: (() => void) | null = null;\n private crossOriginIframeCoordinator: CrossOriginIframeCoordinator | null = null;\n private crossOriginParentSignalCleanup: (() => void) | null = null;\n /** Monotonic counter to ignore stale URL-change targeting results */\n private latestUrlChangeTargetingEvaluationId = 0;\n\n constructor() {\n this.loggerProvider = new SafeLoggerProvider(new Logger());\n }\n\n init(apiKey: string, options: SessionReplayOptions) {\n return returnWrapper(this._init(apiKey, options));\n }\n\n private teardownEventListeners = (teardown: boolean) => {\n const globalScope = getGlobalScope();\n if (globalScope) {\n globalScope.removeEventListener('blur', this.blurListener);\n globalScope.removeEventListener('focus', this.focusListener);\n !teardown && globalScope.addEventListener('blur', this.blurListener);\n !teardown && globalScope.addEventListener('focus', this.focusListener);\n // prefer pagehide to unload events, this is the standard going forward. it is not\n // 100% reliable, but is bfcache-compatible.\n if (globalScope.self && 'onpagehide' in globalScope.self) {\n globalScope.removeEventListener('pagehide', this.pageLeaveListener);\n !teardown && globalScope.addEventListener('pagehide', this.pageLeaveListener);\n } else {\n // this has performance implications, but is the only way we can reliably send events\n // in browser that don't support pagehide.\n globalScope.removeEventListener('beforeunload', this.pageLeaveListener);\n !teardown && globalScope.addEventListener('beforeunload', this.pageLeaveListener);\n }\n }\n };\n\n /**\n * Subscribes to SPA URL changes via the URL tracking plugin. Always keeps\n * `currentPageUrl` in sync (needed for URL-based masking). When a targeting\n * config is present it also re-evaluates targeting on every navigation.\n */\n private setupUrlChangeListener(): void {\n // If init() runs multiple times, remove the previous URL-change subscription first\n // so we don't leak callbacks and trigger duplicate targeting evaluations.\n this.urlChangeCleanup?.();\n\n const globalScope = getGlobalScope() as Window | undefined;\n if (!globalScope?.location) {\n return;\n }\n\n const hasTargeting = !!this.config?.targetingConfig;\n\n const onUrlChange = (href: string): void => {\n this.currentPageUrl = href;\n\n if (hasTargeting) {\n const evaluationId = ++this.latestUrlChangeTargetingEvaluationId;\n void this.evaluateTargetingAndCapture(\n {\n userProperties: {},\n event: undefined,\n page: { url: href },\n },\n false,\n false,\n true,\n );\n this.loggerProvider.debug(`Queued URL-change targeting re-evaluation #${evaluationId} for ${href}.`);\n }\n };\n type UrlUnsubscribe = (scope: Window | undefined, cb: (href: string) => void) => () => void;\n const unsubscribe = (subscribeToUrlChanges as UrlUnsubscribe)(globalScope, onUrlChange);\n\n this.urlChangeCleanup = (): void => {\n unsubscribe();\n this.urlChangeCleanup = null;\n };\n }\n\n /**\n * Single source of truth for the min_session_duration_ms gate. Returns true when the\n * current session has not yet reached the configured threshold (and we have both a\n * configured value and a recorded start time). Returns false when there's no\n * threshold, no recorded start time, or the threshold has been met — i.e. it\n * answers \"should this batch be suppressed?\".\n *\n * Centralizing the check means future changes (clock-skew tolerance, switching to\n * performance.now(), etc.) are one-line edits.\n */\n private isBelowMinSessionDuration(): boolean {\n const minSessionDurationMs = this.config?.minSessionDurationMs;\n if (minSessionDurationMs === undefined || this.sessionStartTime === undefined) {\n return false;\n }\n return Date.now() - this.sessionStartTime < minSessionDurationMs;\n }\n\n private getCurrentPageForTargeting(): SessionReplayTargetingInput['page'] {\n const currentUrl = getGlobalScope()?.location?.href;\n return currentUrl != null ? { url: currentUrl } : undefined;\n }\n\n protected async _init(apiKey: string, options: SessionReplayOptions) {\n // Re-init should always tear down any previous URL-change subscription, even when the\n // next config has no targeting config and we don't subscribe again.\n this.urlChangeCleanup?.();\n\n this.loggerProvider = new SafeLoggerProvider(options.loggerProvider || new Logger());\n Object.prototype.hasOwnProperty.call(options, 'logLevel') &&\n this.loggerProvider.enable(options.logLevel as LogLevel);\n this.currentPageUrl = getCurrentUrl();\n this.identifiers = new SessionIdentifiers({ sessionId: options.sessionId, deviceId: options.deviceId });\n // Persist replay start time per sessionId so the min_session_duration_ms gate\n // measures replay duration (survives page reloads within a session) rather than\n // page-load duration. Storage failures fall back to a transient Date.now().\n const now = Date.now();\n pruneStaleReplayStartTimes(apiKey, now, this.loggerProvider);\n this.sessionStartTime =\n options.sessionId !== undefined\n ? getOrInitReplayStartTime(apiKey, options.sessionId, now, this.loggerProvider) ?? now\n : now;\n this.joinedConfigGenerator = await createSessionReplayJoinedConfigGenerator(apiKey, options);\n const { joinedConfig, localConfig, remoteConfig } = await this.joinedConfigGenerator.generateJoinedConfig();\n this.config = joinedConfig;\n\n this.setMetadata(\n options.sessionId,\n joinedConfig,\n localConfig,\n remoteConfig,\n options.version?.version,\n VERSION,\n options.version?.type,\n );\n\n this.pageLeaveFns = [];\n\n if (options.sessionId && this.config.interactionConfig?.enabled) {\n const scrollWatcher = ScrollWatcher.default(\n {\n sessionId: options.sessionId,\n type: 'interaction',\n },\n this.config,\n );\n this.pageLeaveFns = [scrollWatcher.send(this.getDeviceId.bind(this)).bind(scrollWatcher)];\n this.scrollHook = scrollWatcher.hook.bind(scrollWatcher);\n this.clickHandler = new ClickHandler(this.loggerProvider, scrollWatcher);\n }\n\n const managers: EventsManagerWithType<EventType, string>[] = [];\n let { storeType } = this.config;\n if (storeType === 'idb' && !getGlobalScope()?.indexedDB) {\n storeType = 'memory';\n this.loggerProvider.warn('Could not use preferred indexedDB storage, reverting to in memory option.');\n }\n this.loggerProvider.log(`Using ${storeType} for event storage.`);\n let compressionWorkerScript: string | undefined;\n let trackDestinationWorkerScript: string | undefined;\n const globalScope = getGlobalScope();\n if (this.config.useWebWorker && globalScope && globalScope.Worker) {\n const { compressionScript, trackDestinationScript } = await import('./worker');\n compressionWorkerScript = compressionScript;\n trackDestinationWorkerScript = trackDestinationScript;\n }\n\n let rrwebEventManager: EventsManagerWithBeacon<'replay'> | undefined;\n try {\n rrwebEventManager = await createEventsManager<'replay'>({\n config: this.config,\n type: 'replay',\n storeType,\n trackDestinationWorkerScript,\n shouldSend: () => !this.isBelowMinSessionDuration(),\n });\n this.rrwebEventManager = rrwebEventManager;\n managers.push({ name: 'replay', manager: rrwebEventManager });\n } catch (error) {\n const typedError = error as Error;\n this.loggerProvider.warn(`Error occurred while creating replay events manager: ${typedError.toString()}`);\n }\n\n if (this.config.interactionConfig?.enabled) {\n const payloadBatcher = this.config.interactionConfig.batch ? clickBatcher : clickNonBatcher;\n try {\n const interactionEventManager = await createEventsManager<'interaction'>({\n config: this.config,\n type: 'interaction',\n minInterval: this.config.interactionConfig.trackEveryNms ?? INTERACTION_MIN_INTERVAL,\n maxInterval: INTERACTION_MAX_INTERVAL,\n payloadBatcher,\n storeType,\n trackDestinationWorkerScript,\n });\n managers.push({ name: 'interaction', manager: interactionEventManager });\n } catch (error) {\n const typedError = error as Error;\n this.loggerProvider.warn(`Error occurred while creating interaction events manager: ${typedError.toString()}`);\n }\n }\n\n this.eventsManager = new MultiEventManager<'replay' | 'interaction', string>(...managers);\n // To prevent too many threads.\n if (this.eventCompressor) {\n this.eventCompressor.terminate();\n }\n\n this.eventCompressor = new EventCompressor(\n this.eventsManager,\n this.config,\n this.getDeviceId(),\n compressionWorkerScript,\n () => this.sendEvents(),\n );\n\n // Flush any events that arrived while eventCompressor was not yet ready\n // (e.g. a concurrent setSessionId() call that raced _init()'s async setup).\n if (this.pendingEmitEvents.length > 0) {\n const pending = this.pendingEmitEvents.splice(0);\n for (const { event, sessionId } of pending) {\n this.eventCompressor.enqueueEvent(event, sessionId);\n }\n }\n\n // Register beacon fallback for page exit. sendBeacon survives page unload\n // and delivers any incremental events that haven't been flushed via fetch yet.\n //\n // Known cross-session race: if asyncSetSessionId fired and its async\n // storeCurrentSequence hasn't resolved before unload, the beacon buffer can still\n // hold previous-session events. The gate below reads the *new* session's\n // sessionStartTime, so legitimately-sendable old-session events get suppressed.\n // Follow-up: track start time per buffered batch instead of globally.\n this.pageLeaveFns = [\n ...this.pageLeaveFns,\n () => {\n if (!this.config || !this.identifiers?.sessionId || !rrwebEventManager) return;\n const events = rrwebEventManager.getBeaconEvents();\n if (!events.length) return;\n const deviceId = this.getDeviceId();\n if (!deviceId) return;\n if (this.isBelowMinSessionDuration()) return;\n rrwebEventManager.trackDestination.sendBeacon({\n events,\n sessionId: this.identifiers.sessionId,\n deviceId,\n apiKey: this.config.apiKey,\n serverZone: this.config.serverZone,\n });\n },\n ];\n\n await this.initializeNetworkObservers();\n\n // Enable background capture when this page is opened by the Amplitude app\n // (window.opener exists). Uses the shared messenger singleton so that if\n // autocapture is also loaded, both share a single messenger and script load.\n if (getGlobalScope()?.opener) {\n const messenger = getOrCreateWindowMessenger();\n enableBackgroundCapture(messenger);\n messenger.setup({\n logger: this.loggerProvider,\n ...(this.config.serverZone && { endpoint: AMPLITUDE_ORIGINS_MAP[this.config.serverZone] }),\n });\n }\n\n this.loggerProvider.log('Installing @amplitude/session-replay-browser.');\n\n this.teardownEventListeners(false);\n\n await this.evaluateTargetingAndCapture(\n { userProperties: options.userProperties, page: this.getCurrentPageForTargeting() },\n true,\n );\n\n const needsUrlTracking = this.config.targetingConfig || (this.config.privacyConfig?.urlMaskLevels?.length ?? 0) > 0;\n if (needsUrlTracking) {\n this.setupUrlChangeListener();\n }\n }\n\n setSessionId(sessionId: string | number, deviceId?: string) {\n return returnWrapper(this.asyncSetSessionId(sessionId, deviceId));\n }\n\n async asyncSetSessionId(\n sessionId: string | number,\n deviceId?: string,\n options?: { userProperties?: { [key: string]: any } },\n ) {\n // Invalidate any in-flight URL-change re-evaluations from the previous session.\n this.latestUrlChangeTargetingEvaluationId++;\n this.sessionTargetingMatch = false;\n this.lastShouldRecordDecision = undefined; // Reset targeting decision for new session\n\n const previousSessionId = this.identifiers && this.identifiers.sessionId;\n if (previousSessionId) {\n this.sendEvents(previousSessionId);\n }\n\n const isSessionChange = previousSessionId !== sessionId;\n\n // Drop any beacon-buffered events from the previous session BEFORE installing the\n // new identifiers / start time. Otherwise the page-leave beacon path could attribute\n // old-session events to the new session id, and the gate (using the new start time)\n // would compute the wrong elapsed duration. Skip on a redundant same-sessionId call —\n // the buffer belongs to the *continuing* session and should ship via beacon as normal.\n if (isSessionChange) {\n this.rrwebEventManager?.dropPendingBeaconEvents();\n }\n\n const deviceIdForReplayId = deviceId || this.getDeviceId();\n this.identifiers = new SessionIdentifiers({\n sessionId: sessionId,\n deviceId: deviceIdForReplayId,\n });\n\n // Gate state and persisted start time only get reset on an actual session boundary.\n // A redundant setSessionId(currentId) call would otherwise overwrite both the in-memory\n // and stored start time with a fresh Date.now() — restarting the gate clock for what is\n // supposed to be a continuing session.\n if (isSessionChange) {\n this.sessionStartTime = Date.now();\n this.suppressedSendCount = 0;\n this.hasEmittedGateDecision = false;\n if (this.config?.apiKey) {\n setReplayStartTime(this.config.apiKey, sessionId, this.sessionStartTime, this.loggerProvider);\n if (previousSessionId !== undefined) {\n removeReplayStartTime(this.config.apiKey, previousSessionId, this.loggerProvider);\n }\n }\n }\n\n // If there is no previous session id, SDK is being initialized for the first time,\n // and config was just fetched in initialization, so no need to fetch it a second time\n if (this.joinedConfigGenerator && previousSessionId) {\n const { joinedConfig } = await this.joinedConfigGenerator.generateJoinedConfig();\n this.config = joinedConfig;\n }\n\n if (this.config?.targetingConfig) {\n await this.evaluateTargetingAndCapture(\n { userProperties: options?.userProperties, page: this.getCurrentPageForTargeting() },\n false,\n true,\n );\n } else {\n await this.recordEvents();\n }\n }\n\n getSessionReplayProperties() {\n const config = this.config;\n const identifiers = this.identifiers;\n if (!config || !identifiers) {\n this.loggerProvider.warn('Session replay init has not been called, cannot get session replay properties.');\n return {};\n }\n\n const shouldRecord = this.getShouldRecord();\n let eventProperties: { [key: string]: string | null } = {};\n\n if (shouldRecord) {\n eventProperties = {\n [DEFAULT_SESSION_REPLAY_PROPERTY]: identifiers.sessionReplayId ? identifiers.sessionReplayId : null,\n };\n if (config.debugMode) {\n eventProperties[SESSION_REPLAY_DEBUG_PROPERTY] = JSON.stringify({\n appHash: generateHashCode(config.apiKey).toString(),\n });\n }\n }\n\n void this.addCustomRRWebEvent(\n CustomRRwebEvent.GET_SR_PROPS,\n {\n shouldRecord,\n eventProperties: eventProperties,\n },\n this.eventCount === 10,\n );\n if (this.eventCount === 10) {\n this.eventCount = 0;\n }\n this.eventCount++;\n\n return eventProperties;\n }\n\n blurListener = () => {\n this.sendEvents();\n };\n\n focusListener = () => {\n if (this.recordCancelCallback && this.recordFunction) {\n try {\n this.recordFunction.takeFullSnapshot(true);\n } catch (error) {\n this.loggerProvider.warn('Failed to take full snapshot on focus:', error);\n }\n } else if (!this.recordEventsInFlight) {\n void this.recordEvents(false);\n }\n };\n\n /**\n * This is an instance member so that if init is called multiple times\n * it doesn't add another listener to the page leave event. This is to\n * prevent duplicate listener actions from firing.\n */\n private pageLeaveListener = (e: PageTransitionEvent | Event) => {\n // Synchronously drain any events still queued in the requestIdleCallback\n // pipeline so they are available to send before the page unloads.\n this.eventCompressor?.flushQueue();\n this.sendEvents();\n this.pageLeaveFns.forEach((fn) => {\n fn(e);\n });\n };\n\n evaluateTargetingAndCapture = async (\n targetingParams: SessionReplayTargetingInput,\n isInit = false,\n forceRestart = false,\n forceTargetingReevaluation = false,\n ) => {\n if (!this.identifiers || !this.identifiers.sessionId || !this.config) {\n if (this.identifiers && !this.identifiers.sessionId) {\n this.loggerProvider.log('Session ID has not been set yet, cannot evaluate targeting for Session Replay.');\n } else {\n this.loggerProvider.warn('Session replay init has not been called, cannot evaluate targeting.');\n }\n return;\n }\n\n // Handle cases where there's no targeting config\n if (!this.config.targetingConfig) {\n if (isInit) {\n this.loggerProvider.log('Targeting config has not been set yet, cannot evaluate targeting.');\n } else {\n this.loggerProvider.log('No targeting config set, skipping initialization/recording for event.');\n return;\n }\n }\n\n // Store targeting parameters for use in getShouldRecord\n this.lastTargetingParams = targetingParams;\n\n // Re-evaluate only until we get the first match in this session.\n // Once matched, keep recording for the rest of the session.\n const targetingConfig = this.config.targetingConfig;\n const shouldReEvaluate = targetingConfig && !this.sessionTargetingMatch;\n if (shouldReEvaluate) {\n // Capture URL-change evaluation id so out-of-order async completions can be discarded.\n const urlChangeEvaluationId = forceTargetingReevaluation ? this.latestUrlChangeTargetingEvaluationId : undefined;\n let eventForTargeting = targetingParams.event;\n if (\n eventForTargeting &&\n Object.values(SpecialEventType).includes(eventForTargeting.event_type as SpecialEventType)\n ) {\n eventForTargeting = undefined;\n }\n\n const pageUrl = targetingParams.page?.url ?? getGlobalScope()?.location?.href ?? '';\n const pageForTargeting = targetingParams.page ?? (pageUrl !== '' ? { url: pageUrl } : undefined);\n\n const targetingMatch = await evaluateTargetingAndStore({\n sessionId: this.identifiers.sessionId,\n targetingConfig,\n loggerProvider: this.loggerProvider,\n apiKey: this.config.apiKey,\n targetingParams: {\n userProperties: targetingParams.userProperties,\n event: eventForTargeting,\n page: pageForTargeting,\n },\n urlChange: forceTargetingReevaluation,\n });\n\n if (\n forceTargetingReevaluation &&\n urlChangeEvaluationId !== undefined &&\n urlChangeEvaluationId !== this.latestUrlChangeTargetingEvaluationId\n ) {\n this.loggerProvider.debug(\n `Ignoring stale URL-change targeting result #${urlChangeEvaluationId}; latest is #${this.latestUrlChangeTargetingEvaluationId}.`,\n );\n return;\n }\n // Keep targeting match monotonic within a session: once true, always true.\n // This avoids races where an older in-flight evaluation resolves false after\n // a newer evaluation already resolved true.\n this.sessionTargetingMatch = this.sessionTargetingMatch || targetingMatch;\n\n this.loggerProvider.debug(\n JSON.stringify(\n {\n name: 'targeted replay capture config',\n sessionTargetingMatch: this.sessionTargetingMatch,\n event: eventForTargeting,\n targetingParams: targetingParams,\n },\n null,\n 2,\n ),\n );\n }\n\n if (isInit) {\n void this.initialize(true);\n } else if (forceRestart || !this.recordCancelCallback) {\n this.loggerProvider.log('Recording events for session due to forceRestart or no ongoing recording.');\n await this.recordEvents();\n }\n };\n\n sendEvents(sessionId?: string | number) {\n const sessionIdToSend = sessionId || this.identifiers?.sessionId;\n const deviceId = this.getDeviceId();\n if (this.eventsManager && sessionIdToSend && deviceId) {\n if (this.isBelowMinSessionDuration()) {\n // Safe to dereference: isBelowMinSessionDuration() only returns true when both\n // this.config and this.sessionStartTime are defined.\n const startTime = this.sessionStartTime as number;\n const minMs = (this.config as SessionReplayJoinedConfig).minSessionDurationMs as number;\n this.loggerProvider.log(\n `Session ${sessionIdToSend} not sent: duration ${Date.now() - startTime}ms is below minimum ${minMs}ms.`,\n );\n // We deliberately do NOT clear the beacon buffer here. Blur/visibility-change\n // can call sendEvents() mid-session; if the session later crosses the threshold\n // those buffered events are legitimately sendable via beacon on page exit.\n // Cross-session leak is prevented in asyncSetSessionId, which drops the buffer\n // at the session transition. The page-leave path also gates independently.\n this.suppressedSendCount++;\n return;\n }\n // On the first send-after-pass for the session, emit a custom rrweb event so the\n // payload itself carries the gate verdict. Lets backend ingestion diff intended\n // vs actual replay counts to spot start-time-tracking regressions.\n //\n // Gate on recording being active too: addCustomRRWebEvent is a no-op when\n // recordCancelCallback/recordFunction aren't set yet (e.g. a blur-listener-fired\n // sendEvents reaches us before _recordEvents() activates on a reloaded long-lived\n // session). Tripping hasEmittedGateDecision unconditionally would lose the signal\n // for that session's first real send.\n if (\n !this.hasEmittedGateDecision &&\n this.config?.minSessionDurationMs !== undefined &&\n this.recordCancelCallback &&\n this.recordFunction\n ) {\n this.hasEmittedGateDecision = true;\n const elapsedMs = this.sessionStartTime !== undefined ? Date.now() - this.sessionStartTime : undefined;\n void this.addCustomRRWebEvent(\n CustomRRwebEvent.REPLAY_GATE_DECISION,\n {\n sessionId: sessionIdToSend,\n suppressedSendCount: this.suppressedSendCount,\n elapsedMs,\n minSessionDurationMs: this.config.minSessionDurationMs,\n },\n false,\n );\n }\n this.eventsManager.sendCurrentSequenceEvents({ sessionId: sessionIdToSend, deviceId });\n }\n }\n\n async initialize(shouldSendStoredEvents = false) {\n if (!this.identifiers?.sessionId) {\n this.loggerProvider.log(`Session is not being recorded due to lack of session id.`);\n return Promise.resolve();\n }\n\n const deviceId = this.getDeviceId();\n if (!deviceId) {\n this.loggerProvider.log(`Session is not being recorded due to lack of device id.`);\n return Promise.resolve();\n }\n this.eventsManager && shouldSendStoredEvents && void this.eventsManager.sendStoredEvents({ deviceId });\n\n return this.recordEvents();\n }\n\n shouldOptOut() {\n let identityStoreOptOut: boolean | undefined;\n if (this.config?.instanceName) {\n const identityStore = getAnalyticsConnector(this.config.instanceName).identityStore;\n identityStoreOptOut = identityStore.getIdentity().optOut;\n }\n\n return identityStoreOptOut !== undefined ? identityStoreOptOut : this.config?.optOut;\n }\n\n getShouldRecord() {\n if (!this.identifiers || !this.config || !this.identifiers.sessionId) {\n this.loggerProvider.warn(`Session is not being recorded due to lack of config, please call sessionReplay.init.`);\n return false;\n }\n\n if (!this.config.captureEnabled) {\n this.loggerProvider.log(\n `Session ${this.identifiers.sessionId} not being captured due to capture being disabled for project or because the remote config could not be fetched.`,\n );\n return false;\n }\n\n if (this.shouldOptOut()) {\n this.loggerProvider.log(`Opting session ${this.identifiers.sessionId} out of recording due to optOut config.`);\n return false;\n }\n\n let shouldRecord = false;\n let message = '';\n let matched = false;\n\n // If targetingConfig exists, we'll use the sessionTargetingMatch to determine whether to record\n // Otherwise, we'll evaluate the session against the overall sample rate\n if (this.config.targetingConfig) {\n if (!this.sessionTargetingMatch) {\n message = `Not capturing replays for session ${this.identifiers.sessionId} due to not matching targeting conditions.`;\n this.loggerProvider.log(message);\n shouldRecord = false;\n matched = false;\n } else {\n message = `Capturing replays for session ${this.identifiers.sessionId} due to matching targeting conditions.`;\n this.loggerProvider.log(message);\n shouldRecord = true;\n matched = true;\n }\n } else {\n const isInSample = isSessionInSample(this.identifiers.sessionId, this.config.sampleRate);\n if (!isInSample) {\n message = `Opting session ${this.identifiers.sessionId} out of recording due to sample rate.`;\n this.loggerProvider.log(message);\n shouldRecord = false;\n matched = false;\n } else {\n shouldRecord = true;\n matched = true;\n }\n }\n\n // Only send custom rrweb event for targeting decision when the decision changes\n if (this.lastShouldRecordDecision !== shouldRecord && this.config.targetingConfig) {\n void this.addCustomRRWebEvent(CustomRRwebEvent.TARGETING_DECISION, {\n message,\n sessionId: this.identifiers.sessionId,\n matched,\n targetingParams: this.lastTargetingParams,\n });\n this.lastShouldRecordDecision = shouldRecord;\n }\n\n return shouldRecord;\n }\n\n getBlockSelectors(): string | string[] | undefined {\n // For some reason, this defaults to empty array ([]) if undefined in the compiled script.\n // Empty arrays cause errors when being evaluated in Safari.\n // Force the selector to be undefined if it's an empty array.\n const blockSelector = this.config?.privacyConfig?.blockSelector ?? [];\n if (blockSelector.length === 0) {\n return undefined;\n }\n return blockSelector;\n }\n\n getMaskTextSelectors(): string | undefined {\n const privacyConfig = this.config?.privacyConfig;\n const effectiveLevel = privacyConfig ? getEffectiveMaskLevel(this.currentPageUrl, privacyConfig) : undefined;\n\n if (effectiveLevel === 'conservative') {\n return '*';\n }\n\n // If any urlMaskLevels rule uses 'conservative', always route all text nodes\n // through maskTextFn so the dynamic URL getter can decide at call time.\n // Without this, rrweb's static maskTextSelector would miss text nodes when\n // the user navigates from a non-conservative page to a conservative one.\n if (privacyConfig?.urlMaskLevels?.some((rule) => rule.maskLevel === 'conservative')) {\n return '*';\n }\n\n // If defaultMaskLevel is 'conservative' and URL rules exist, always route text through\n // maskTextFn — a page matching no rule falls back to the conservative default, and\n // rrweb must be set up at start to call maskTextFn for those text nodes.\n const urlMaskLevels = privacyConfig?.urlMaskLevels;\n if (privacyConfig?.defaultMaskLevel === 'conservative' && urlMaskLevels && urlMaskLevels.length > 0) {\n return '*';\n }\n\n const maskSelector = privacyConfig?.maskSelector;\n if (!maskSelector) {\n return;\n }\n\n return maskSelector as unknown as string;\n }\n\n async getRecordingPlugins(loggingConfig: LoggingConfig | undefined) {\n const plugins = [];\n\n // Add URL tracking plugin\n try {\n const urlTrackingPlugin = createUrlTrackingPlugin({\n ugcFilterRules: this.config?.interactionConfig?.ugcFilterRules || [],\n enablePolling: this.config?.enableUrlChangePolling || false,\n pollingInterval: this.config?.urlChangePollingInterval,\n captureDocumentTitle: this.config?.captureDocumentTitle,\n });\n\n plugins.push(urlTrackingPlugin);\n } catch (error) {\n this.loggerProvider.warn('Failed to create URL tracking plugin:', error);\n }\n\n // Default plugin settings -\n // {\n // level: ['info', 'log', 'warn', 'error'],\n // lengthThreshold: 10000,\n // stringifyOptions: {\n // stringLengthLimit: undefined,\n // numOfKeysLimit: 50,\n // depthOfLimit: 4,\n // },\n // logger: window.console,\n // }\n if (loggingConfig?.console?.enabled) {\n try {\n // Dynamic import keeps console plugin separate and only loads when needed\n const { getRecordConsolePlugin } = await import('@amplitude/rrweb-plugin-console-record');\n plugins.push(getRecordConsolePlugin({ level: loggingConfig.console.levels }));\n } catch (error) {\n this.loggerProvider.warn('Failed to load console plugin:', error);\n }\n }\n\n return plugins.length > 0 ? plugins : undefined;\n }\n\n private async getRecordFunction(): Promise<RecordFunction | null> {\n if (this.recordFunction) {\n return this.recordFunction;\n }\n\n try {\n const { record } = await import('@amplitude/rrweb-record');\n this.recordFunction = record;\n return record;\n } catch (error) {\n this.loggerProvider.warn('Failed to load rrweb-record module:', error);\n return null;\n }\n }\n\n async recordEvents(shouldLogMetadata = true) {\n if (this.recordEventsInFlight) {\n this.recordEventsPendingShouldLogMetadata = shouldLogMetadata;\n return;\n }\n this.recordEventsInFlight = true;\n try {\n await this._recordEvents(shouldLogMetadata);\n while (this.recordEventsPendingShouldLogMetadata !== null) {\n const pendingArgs = this.recordEventsPendingShouldLogMetadata;\n this.recordEventsPendingShouldLogMetadata = null;\n await this._recordEvents(pendingArgs);\n }\n } finally {\n this.recordEventsInFlight = false;\n this.recordEventsPendingShouldLogMetadata = null;\n }\n }\n\n private async _recordEvents(shouldLogMetadata = true) {\n const config = this.config;\n const shouldRecord = this.getShouldRecord();\n const sessionId = this.identifiers?.sessionId;\n if (!shouldRecord || !sessionId || !config) {\n return;\n }\n this.stopRecordingEvents();\n\n const recordFunction = await this.getRecordFunction();\n\n // May be undefined if cannot import rrweb-record\n if (!recordFunction) {\n return;\n }\n\n await this.initializeNetworkObservers();\n\n const networkLoggingConfig = config.loggingConfig?.network;\n const trackUrl = getServerUrl(config.serverZone, config.trackServerUrl);\n const ignoredUrls = [SESSION_REPLAY_SERVER_URL, SESSION_REPLAY_EU_URL, SESSION_REPLAY_STAGING_URL, trackUrl];\n this.networkObservers?.start((event: NetworkRequestEvent) => {\n if (ignoredUrls.some((url) => event.url.startsWith(url))) return;\n void this.addCustomRRWebEvent(CustomRRwebEvent.FETCH_REQUEST, event);\n }, networkLoggingConfig);\n const { interactionConfig, loggingConfig } = config;\n\n const hooks = interactionConfig?.enabled\n ? {\n mouseInteraction:\n this.eventsManager &&\n this.clickHandler?.createHook({\n eventsManager: this.eventsManager,\n sessionId,\n deviceIdFn: this.getDeviceId.bind(this),\n mirror: recordFunction.mirror,\n ugcFilterRules: interactionConfig.ugcFilterRules ?? [],\n performanceOptions: config.performanceConfig?.interaction,\n }),\n scroll: this.scrollHook,\n }\n : {};\n\n const ugcFilterRules =\n interactionConfig?.enabled && interactionConfig.ugcFilterRules ? interactionConfig.ugcFilterRules : [];\n\n this.loggerProvider.log(`Session Replay capture beginning for ${sessionId}.`);\n\n try {\n const crossOriginIframesEnabled = !!config.crossOriginIframes?.enabled;\n const coordinateChildren = config.crossOriginIframes?.coordinateChildren !== false;\n const childMode = crossOriginIframesEnabled && isInIframe();\n\n if (childMode && coordinateChildren) {\n // Child mode: don't self-start; wait for a start signal from the parent.\n // (The previous listener, if any, was already removed by stopRecordingEvents above.)\n this.crossOriginParentSignalCleanup = listenForParentSignals({\n onStart: () => this._recordEventsInChildMode(recordFunction, sessionId, config, hooks),\n onStop: () => {\n try {\n // Only cancel the rrweb recording — do NOT call stopRecordingEvents() here,\n // which would clear crossOriginParentSignalCleanup and make the child deaf\n // to subsequent start/stop cycles from the parent.\n this.recordCancelCallback && this.recordCancelCallback();\n this.recordCancelCallback = null;\n } catch (error) {\n const typedError = error as Error;\n this.loggerProvider.warn(\n `Error occurred while stopping child iframe replay capture: ${typedError.toString()}`,\n );\n }\n },\n });\n return;\n }\n\n this.recordCancelCallback = recordFunction({\n ...this.buildRRWebRecordOptions(\n config,\n hooks,\n (event: eventWithTime) => {\n if (this.shouldOptOut()) {\n this.loggerProvider.log(`Opting session ${sessionId} out of recording due to optOut config.`);\n this.stopRecordingEvents();\n this.sendEvents();\n return;\n }\n\n if (event.type === RRWebEventType.Meta) {\n event.data.href = getPageUrl(event.data.href, ugcFilterRules);\n }\n\n if (this.eventCompressor) {\n // Schedule processing during idle time if the browser supports requestIdleCallback\n this.eventCompressor.enqueueEvent(event, sessionId);\n } else {\n // eventCompressor is not yet ready (concurrent call racing _init()).\n // Buffer the event so it can be flushed once the compressor is initialized.\n this.pendingEmitEvents.push({ event, sessionId });\n }\n },\n 'Error while capturing replay: ',\n ),\n plugins: await this.getRecordingPlugins(loggingConfig),\n recordCrossOriginIframes: crossOriginIframesEnabled,\n });\n\n if (crossOriginIframesEnabled && !childMode && coordinateChildren) {\n if (!this.crossOriginIframeCoordinator) {\n this.crossOriginIframeCoordinator = new CrossOriginIframeCoordinator();\n }\n this.crossOriginIframeCoordinator.start();\n }\n\n void this.addCustomRRWebEvent(CustomRRwebEvent.DEBUG_INFO);\n if (shouldLogMetadata) {\n void this.addCustomRRWebEvent(CustomRRwebEvent.METADATA, this.metadata);\n }\n } catch (error) {\n this.loggerProvider.warn('Failed to initialize session replay:', error);\n }\n }\n\n private buildRRWebRecordOptions(\n config: SessionReplayJoinedConfig,\n hooks: { mouseInteraction?: any; scroll?: scrollCallback },\n emit: (event: eventWithTime) => void,\n errorLogPrefix: string,\n ): Parameters<RecordFunction>[0] {\n const { privacyConfig } = config;\n return {\n emit,\n inlineStylesheet: config.shouldInlineStylesheet,\n hooks,\n maskAllInputs: true,\n maskTextClass: MASK_TEXT_CLASS,\n blockClass: BLOCK_CLASS,\n blockSelector: this.getBlockSelectors() as string | undefined,\n applyBackgroundColorToBlockedElements: config.applyBackgroundColorToBlockedElements,\n maskInputFn: maskFn('input', privacyConfig, () => this.currentPageUrl),\n maskTextFn: maskFn('text', privacyConfig, () => this.currentPageUrl),\n maskAttributeFn: maskAttributeFn(privacyConfig, () => this.currentPageUrl),\n maskTextSelector: this.getMaskTextSelectors(),\n ...(config.fullSnapshotIntervalMs !== undefined && { checkoutEveryNms: config.fullSnapshotIntervalMs }),\n recordCanvas: false,\n captureAdoptedStyleSheets: config.captureAdoptedStyleSheets,\n // Strip nodes that are never rendered by the rrweb replay player.\n // None of these affect visual fidelity; omitting them reduces snapshot size.\n slimDOMOptions: {\n script: true,\n comment: true,\n headFavicon: true,\n headWhitespace: true,\n headMetaDescKeywords: true,\n headMetaSocial: true,\n headMetaRobots: true,\n headMetaHttpEquiv: true,\n headMetaAuthorship: true,\n headMetaVerification: true,\n },\n errorHandler: (error: unknown) => {\n const typedError = error as Error & { _external_?: boolean };\n // styled-components relies on this error being thrown and bubbled up, rrweb is otherwise suppressing it\n if (typedError.message.includes('insertRule') && typedError.message.includes('CSSStyleSheet')) {\n throw typedError;\n }\n // rrweb monkey-patches window functions like CSSStyleSheet.insertRule; errors from external callers\n // (e.g. styled-components) must be re-thrown so they aren't silently swallowed.\n if (typedError._external_) {\n throw typedError;\n }\n this.loggerProvider.warn(errorLogPrefix, typedError.toString());\n // Return true so that we don't clutter user's consoles with internal rrweb errors\n return true;\n },\n };\n }\n\n private _recordEventsInChildMode(\n recordFunction: RecordFunction,\n sessionId: string | number,\n config: SessionReplayJoinedConfig,\n hooks: { mouseInteraction?: any; scroll?: scrollCallback },\n ) {\n // In child mode, rrweb detects window.parent !== window and routes events via\n // postMessage to the parent instead of calling emit. The emit callback is unused.\n // Note: recording plugins (URL tracking, console capture) are intentionally omitted\n // here — the child's events are merged into the parent stream, so URL changes inside\n // the iframe should not be recorded as parent page-view events.\n try {\n // Stop only the previous rrweb recording. Do NOT call stopRecordingEvents() here:\n // that would clear crossOriginParentSignalCleanup — the very listener that invoked\n // this method — making the child permanently deaf to subsequent stop/start cycles.\n this.recordCancelCallback && this.recordCancelCallback();\n this.recordCancelCallback = null;\n this.recordCancelCallback = recordFunction({\n ...this.buildRRWebRecordOptions(\n config,\n hooks,\n () => {\n // no-op: child events are forwarded to parent via postMessage by rrweb\n },\n 'Error while capturing replay (child iframe): ',\n ),\n recordCrossOriginIframes: true, // child mode is only entered when crossOriginIframes.enabled is true\n });\n this.loggerProvider.log(`Session Replay child iframe capture beginning for session ${sessionId}.`);\n } catch (error) {\n this.loggerProvider.warn('Failed to initialize session replay in child iframe mode:', error);\n }\n }\n\n addCustomRRWebEvent = async (\n eventName: CustomRRwebEvent,\n eventData: { [key: string]: any } = {},\n addStorageInfo = true,\n ) => {\n try {\n let debugInfo: DebugInfo | undefined = undefined;\n const config = this.config;\n // Only add debug info for non-metadata events\n if (config && eventName !== CustomRRwebEvent.METADATA) {\n debugInfo = {\n config: getDebugConfig(config),\n version: VERSION,\n };\n if (addStorageInfo) {\n const storageSizeData = await getStorageSize();\n debugInfo = {\n ...storageSizeData,\n ...debugInfo,\n };\n }\n }\n // Check first to ensure we are recording\n if (this.recordCancelCallback && this.recordFunction) {\n this.recordFunction.addCustomEvent(eventName, {\n ...eventData,\n ...debugInfo,\n });\n } else {\n this.loggerProvider.debug(\n `Not able to add custom replay capture event ${eventName} due to no ongoing recording.`,\n );\n }\n } catch (e) {\n this.loggerProvider.debug('Error while adding custom replay capture event: ', e);\n }\n };\n\n stopRecordingEvents = () => {\n try {\n this.loggerProvider.log('Session Replay capture stopping.');\n this.recordCancelCallback && this.recordCancelCallback();\n this.recordCancelCallback = null;\n this.networkObservers?.stop();\n this.crossOriginIframeCoordinator?.stop();\n this.crossOriginIframeCoordinator = null;\n // Remove the child-mode parent signal listener so a later mode change\n // (e.g. crossOriginIframes disabled) does not leave a stale listener.\n this.crossOriginParentSignalCleanup?.();\n this.crossOriginParentSignalCleanup = null;\n } catch (error) {\n const typedError = error as Error;\n this.loggerProvider.warn(`Error occurred while stopping replay capture: ${typedError.toString()}`);\n }\n };\n\n getDeviceId() {\n return this.identifiers?.deviceId;\n }\n\n getSessionId() {\n return this.identifiers?.sessionId;\n }\n\n async flush(useRetry = false) {\n // Intentionally not gated on min_session_duration_ms. flush() forwards payloads\n // already queued in trackDestination, and every code path that queues into it —\n // sendCurrentSequenceEvents, addEvent's batch-split path, sendStoredEvents reading\n // sequencesToSend from IDB — has already passed the gate at the time of queuing.\n // A duplicate gate here would be unreachable.\n return this.eventsManager?.flush(useRetry);\n }\n\n shutdown() {\n this.urlChangeCleanup?.();\n this.crossOriginParentSignalCleanup?.();\n this.crossOriginParentSignalCleanup = null;\n this.teardownEventListeners(true);\n this.stopRecordingEvents();\n this.sendEvents();\n }\n\n private mapSDKType(sdkType: string | undefined) {\n if (sdkType === 'plugin') {\n return '@amplitude/plugin-session-replay-browser';\n }\n\n if (sdkType === 'segment') {\n return '@amplitude/segment-session-replay-plugin';\n }\n\n return null;\n }\n\n private setMetadata(\n sessionId: string | number | undefined,\n joinedConfig: SessionReplayJoinedConfig,\n localConfig: SessionReplayLocalConfig,\n remoteConfig: SessionReplayRemoteConfig | undefined,\n replaySDKVersion: string | undefined,\n standaloneSDKVersion: string | undefined,\n sdkType: string | undefined,\n ) {\n const hashValue = sessionId?.toString() ? generateHashCode(sessionId.toString()) : undefined;\n\n this.metadata = {\n joinedConfig,\n localConfig,\n remoteConfig,\n sessionId,\n hashValue,\n sampleRate: joinedConfig.sampleRate,\n replaySDKType: this.mapSDKType(sdkType),\n replaySDKVersion,\n standaloneSDKType: '@amplitude/session-replay-browser',\n standaloneSDKVersion,\n };\n }\n\n private async initializeNetworkObservers(): Promise<void> {\n if (this.config?.loggingConfig?.network?.enabled && !this.networkObservers) {\n try {\n const { NetworkObservers: NetworkObserversClass } = await import('./observers');\n this.networkObservers = new NetworkObserversClass();\n } catch (error) {\n this.loggerProvider.warn('Failed to import or instantiate NetworkObservers:', error);\n }\n }\n }\n}\n"]}
1
+ {"version":3,"file":"session-replay.js","sourceRoot":"","sources":["../../src/session-replay.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n getAnalyticsConnector,\n getGlobalScope,\n ILogger,\n Logger,\n LogLevel,\n returnWrapper,\n SpecialEventType,\n generateHashCode,\n getOrCreateWindowMessenger,\n enableBackgroundCapture,\n AMPLITUDE_ORIGINS_MAP,\n} from '@amplitude/analytics-core';\n\n// Import only specific types to avoid pulling in the entire rrweb-types package\nimport { eventWithTime, EventType as RRWebEventType, scrollCallback } from '@amplitude/rrweb-types';\nimport { createSessionReplayJoinedConfigGenerator } from './config/joined-config';\nimport {\n LoggingConfig,\n SessionReplayJoinedConfig,\n SessionReplayJoinedConfigGenerator,\n SessionReplayLocalConfig,\n SessionReplayMetadata,\n SessionReplayRemoteConfig,\n} from './config/types';\nimport {\n BLOCK_CLASS,\n CustomRRwebEvent,\n DEFAULT_SESSION_REPLAY_PROPERTY,\n INTERACTION_MAX_INTERVAL,\n INTERACTION_MIN_INTERVAL,\n MASK_TEXT_CLASS,\n SESSION_REPLAY_DEBUG_PROPERTY,\n SESSION_REPLAY_EU_URL,\n SESSION_REPLAY_SERVER_URL,\n SESSION_REPLAY_STAGING_URL,\n} from './constants';\nimport {\n getServerUrl,\n getDebugConfig,\n getEffectiveMaskLevel,\n getPageUrl,\n getStorageSize,\n getCurrentUrl,\n maskFn,\n maskAttributeFn,\n} from './helpers';\nimport { EventCompressor } from './events/event-compressor';\nimport { createEventsManager, EventsManagerWithBeacon } from './events/events-manager';\nimport { MultiEventManager } from './events/multi-manager';\nimport { clickBatcher, ClickHandler, clickNonBatcher } from './hooks/click';\nimport { ScrollWatcher } from './hooks/scroll';\nimport { SessionIdentifiers } from './identifiers';\nimport { SafeLoggerProvider } from './logger';\nimport {\n getOrInitReplayStartTime,\n pruneStaleReplayStartTimes,\n removeReplayStartTime,\n setReplayStartTime,\n} from './replay-start-time-store';\nimport { evaluateTargetingAndStore } from './targeting/targeting-manager';\nimport {\n AmplitudeSessionReplay,\n SessionReplayEventsManager as AmplitudeSessionReplayEventsManager,\n DebugInfo,\n EventsManagerWithType,\n EventType,\n SessionIdentifiers as ISessionIdentifiers,\n SessionReplayOptions,\n SessionReplayTargetingInput,\n} from './typings/session-replay';\nimport { isSessionInSample } from './sampling';\nimport { VERSION } from './version';\n\n// Import only the type for NetworkRequestEvent to keep type safety\nimport type { NetworkObservers, NetworkRequestEvent } from './observers';\nimport { createUrlTrackingPlugin, subscribeToUrlChanges } from './plugins/url-tracking-plugin';\nimport type { RecordFunction } from './utils/rrweb';\nimport { isInIframe, CrossOriginIframeCoordinator, listenForParentSignals } from './cross-origin-iframes';\n\ntype PageLeaveFn = (e: PageTransitionEvent | Event) => void;\n\nexport class SessionReplay implements AmplitudeSessionReplay {\n name = '@amplitude/session-replay-browser';\n config: SessionReplayJoinedConfig | undefined;\n joinedConfigGenerator: SessionReplayJoinedConfigGenerator | undefined;\n identifiers: ISessionIdentifiers | undefined;\n eventsManager?: AmplitudeSessionReplayEventsManager<'replay' | 'interaction', string>;\n loggerProvider: ILogger;\n recordCancelCallback: ReturnType<RecordFunction> | null = null;\n eventCount = 0;\n eventCompressor: EventCompressor | undefined;\n sessionTargetingMatch = false;\n private lastTargetingParams?: SessionReplayTargetingInput;\n private lastShouldRecordDecision?: boolean;\n\n // Public on purpose. `pageLeaveFns` is iterated by `pageLeaveListener`,\n // `rrwebEventManager` is dereferenced in `asyncSetSessionId` to drop the beacon buffer\n // at a session boundary, and `sessionStartTime` drives `isBelowMinSessionDuration()`.\n // Tests also stub/inspect these — privatizing them would break both production callers\n // and the gate's test coverage.\n pageLeaveFns: PageLeaveFn[] = [];\n sessionStartTime: number | undefined;\n rrwebEventManager: EventsManagerWithBeacon<'replay'> | undefined;\n /**\n * Count of sendEvents() calls suppressed by the min-session-duration gate for the\n * current session. Drives the REPLAY_GATE_DECISION rrweb event on first send-after-pass.\n */\n private suppressedSendCount = 0;\n /** True once REPLAY_GATE_DECISION has been emitted for the current session. */\n private hasEmittedGateDecision = false;\n private scrollHook?: scrollCallback;\n private clickHandler?: ClickHandler;\n private networkObservers?: NetworkObservers;\n private metadata: SessionReplayMetadata | undefined;\n\n // Cache the dynamically imported record function\n private recordFunction: RecordFunction | null = null;\n private recordEventsInFlight = false;\n private pendingEmitEvents: Array<{ event: eventWithTime; sessionId: string | number }> = [];\n\n /** Current page URL, kept in sync with SPA navigations for URL-based masking */\n private currentPageUrl = '';\n\n private recordEventsPendingShouldLogMetadata: boolean | null = null;\n\n /** Cleanup for URL change listener used to re-evaluate targeting on SPA route changes */\n private urlChangeCleanup: (() => void) | null = null;\n private crossOriginIframeCoordinator: CrossOriginIframeCoordinator | null = null;\n private crossOriginParentSignalCleanup: (() => void) | null = null;\n /** Monotonic counter to ignore stale URL-change targeting results */\n private latestUrlChangeTargetingEvaluationId = 0;\n\n constructor() {\n this.loggerProvider = new SafeLoggerProvider(new Logger());\n }\n\n init(apiKey: string, options: SessionReplayOptions) {\n return returnWrapper(this._init(apiKey, options));\n }\n\n private teardownEventListeners = (teardown: boolean) => {\n const globalScope = getGlobalScope();\n if (globalScope) {\n globalScope.removeEventListener('blur', this.blurListener);\n globalScope.removeEventListener('focus', this.focusListener);\n !teardown && globalScope.addEventListener('blur', this.blurListener);\n !teardown && globalScope.addEventListener('focus', this.focusListener);\n // prefer pagehide to unload events, this is the standard going forward. it is not\n // 100% reliable, but is bfcache-compatible.\n if (globalScope.self && 'onpagehide' in globalScope.self) {\n globalScope.removeEventListener('pagehide', this.pageLeaveListener);\n !teardown && globalScope.addEventListener('pagehide', this.pageLeaveListener);\n } else {\n // this has performance implications, but is the only way we can reliably send events\n // in browser that don't support pagehide.\n globalScope.removeEventListener('beforeunload', this.pageLeaveListener);\n !teardown && globalScope.addEventListener('beforeunload', this.pageLeaveListener);\n }\n }\n };\n\n /**\n * Subscribes to SPA URL changes via the URL tracking plugin. Always keeps\n * `currentPageUrl` in sync (needed for URL-based masking). When a targeting\n * config is present it also re-evaluates targeting on every navigation.\n */\n private setupUrlChangeListener(): void {\n // If init() runs multiple times, remove the previous URL-change subscription first\n // so we don't leak callbacks and trigger duplicate targeting evaluations.\n this.urlChangeCleanup?.();\n\n const globalScope = getGlobalScope() as Window | undefined;\n if (!globalScope?.location) {\n return;\n }\n\n const hasTargeting = !!this.config?.targetingConfig;\n\n const onUrlChange = (href: string): void => {\n this.currentPageUrl = href;\n\n if (hasTargeting) {\n const evaluationId = ++this.latestUrlChangeTargetingEvaluationId;\n void this.evaluateTargetingAndCapture(\n {\n userProperties: {},\n event: undefined,\n page: { url: href },\n },\n false,\n false,\n true,\n );\n this.loggerProvider.debug(`Queued URL-change targeting re-evaluation #${evaluationId} for ${href}.`);\n }\n };\n type UrlUnsubscribe = (scope: Window | undefined, cb: (href: string) => void) => () => void;\n const unsubscribe = (subscribeToUrlChanges as UrlUnsubscribe)(globalScope, onUrlChange);\n\n this.urlChangeCleanup = (): void => {\n unsubscribe();\n this.urlChangeCleanup = null;\n };\n }\n\n /**\n * Single source of truth for the min_session_duration_ms gate. Returns true when the\n * current session has not yet reached the configured threshold (and we have both a\n * configured value and a recorded start time). Returns false when there's no\n * threshold, no recorded start time, or the threshold has been met — i.e. it\n * answers \"should this batch be suppressed?\".\n *\n * Centralizing the check means future changes (clock-skew tolerance, switching to\n * performance.now(), etc.) are one-line edits.\n */\n private isBelowMinSessionDuration(): boolean {\n const minSessionDurationMs = this.config?.minSessionDurationMs;\n if (minSessionDurationMs === undefined || this.sessionStartTime === undefined) {\n return false;\n }\n return Date.now() - this.sessionStartTime < minSessionDurationMs;\n }\n\n private getCurrentPageForTargeting(): SessionReplayTargetingInput['page'] {\n const currentUrl = getGlobalScope()?.location?.href;\n return currentUrl != null ? { url: currentUrl } : undefined;\n }\n\n protected async _init(apiKey: string, options: SessionReplayOptions) {\n // Re-init should always tear down any previous URL-change subscription, even when the\n // next config has no targeting config and we don't subscribe again.\n this.urlChangeCleanup?.();\n\n this.loggerProvider = new SafeLoggerProvider(options.loggerProvider || new Logger());\n Object.prototype.hasOwnProperty.call(options, 'logLevel') &&\n this.loggerProvider.enable(options.logLevel as LogLevel);\n this.currentPageUrl = getCurrentUrl();\n this.identifiers = new SessionIdentifiers({ sessionId: options.sessionId, deviceId: options.deviceId });\n // Persist replay start time per sessionId so the min_session_duration_ms gate\n // measures replay duration (survives page reloads within a session) rather than\n // page-load duration. Storage failures fall back to a transient Date.now().\n const now = Date.now();\n pruneStaleReplayStartTimes(apiKey, now, this.loggerProvider);\n this.sessionStartTime =\n options.sessionId !== undefined\n ? getOrInitReplayStartTime(apiKey, options.sessionId, now, this.loggerProvider) ?? now\n : now;\n this.joinedConfigGenerator = await createSessionReplayJoinedConfigGenerator(apiKey, options);\n const { joinedConfig, localConfig, remoteConfig } = await this.joinedConfigGenerator.generateJoinedConfig();\n this.config = joinedConfig;\n\n this.setMetadata(\n options.sessionId,\n joinedConfig,\n localConfig,\n remoteConfig,\n options.version?.version,\n VERSION,\n options.version?.type,\n );\n\n this.pageLeaveFns = [];\n\n if (options.sessionId && this.config.interactionConfig?.enabled) {\n const scrollWatcher = ScrollWatcher.default(\n {\n sessionId: options.sessionId,\n type: 'interaction',\n },\n this.config,\n );\n this.pageLeaveFns = [scrollWatcher.send(this.getDeviceId.bind(this)).bind(scrollWatcher)];\n this.scrollHook = scrollWatcher.hook.bind(scrollWatcher);\n this.clickHandler = new ClickHandler(this.loggerProvider, scrollWatcher);\n }\n\n const managers: EventsManagerWithType<EventType, string>[] = [];\n let { storeType } = this.config;\n if (storeType === 'idb' && !getGlobalScope()?.indexedDB) {\n storeType = 'memory';\n this.loggerProvider.warn('Could not use preferred indexedDB storage, reverting to in memory option.');\n }\n this.loggerProvider.log(`Using ${storeType} for event storage.`);\n let compressionWorkerScript: string | undefined;\n let trackDestinationWorkerScript: string | undefined;\n const globalScope = getGlobalScope();\n if (this.config.useWebWorker && globalScope && globalScope.Worker) {\n const { compressionScript, trackDestinationScript } = await import('./worker');\n compressionWorkerScript = compressionScript;\n trackDestinationWorkerScript = trackDestinationScript;\n }\n\n let rrwebEventManager: EventsManagerWithBeacon<'replay'> | undefined;\n try {\n rrwebEventManager = await createEventsManager<'replay'>({\n config: this.config,\n type: 'replay',\n minInterval: this.config.flushIntervalConfig?.minIntervalMs,\n maxInterval: this.config.flushIntervalConfig?.maxIntervalMs,\n storeType,\n trackDestinationWorkerScript,\n shouldSend: () => !this.isBelowMinSessionDuration(),\n });\n this.rrwebEventManager = rrwebEventManager;\n managers.push({ name: 'replay', manager: rrwebEventManager });\n } catch (error) {\n const typedError = error as Error;\n this.loggerProvider.warn(`Error occurred while creating replay events manager: ${typedError.toString()}`);\n }\n\n if (this.config.interactionConfig?.enabled) {\n const payloadBatcher = this.config.interactionConfig.batch ? clickBatcher : clickNonBatcher;\n try {\n const interactionEventManager = await createEventsManager<'interaction'>({\n config: this.config,\n type: 'interaction',\n minInterval: this.config.interactionConfig.trackEveryNms ?? INTERACTION_MIN_INTERVAL,\n maxInterval: INTERACTION_MAX_INTERVAL,\n payloadBatcher,\n storeType,\n trackDestinationWorkerScript,\n });\n managers.push({ name: 'interaction', manager: interactionEventManager });\n } catch (error) {\n const typedError = error as Error;\n this.loggerProvider.warn(`Error occurred while creating interaction events manager: ${typedError.toString()}`);\n }\n }\n\n this.eventsManager = new MultiEventManager<'replay' | 'interaction', string>(...managers);\n // To prevent too many threads.\n if (this.eventCompressor) {\n this.eventCompressor.terminate();\n }\n\n this.eventCompressor = new EventCompressor(\n this.eventsManager,\n this.config,\n this.getDeviceId(),\n compressionWorkerScript,\n () => this.sendEvents(),\n );\n\n // Flush any events that arrived while eventCompressor was not yet ready\n // (e.g. a concurrent setSessionId() call that raced _init()'s async setup).\n if (this.pendingEmitEvents.length > 0) {\n const pending = this.pendingEmitEvents.splice(0);\n for (const { event, sessionId } of pending) {\n this.eventCompressor.enqueueEvent(event, sessionId);\n }\n }\n\n // Register beacon fallback for page exit. sendBeacon survives page unload\n // and delivers any incremental events that haven't been flushed via fetch yet.\n //\n // Known cross-session race: if asyncSetSessionId fired and its async\n // storeCurrentSequence hasn't resolved before unload, the beacon buffer can still\n // hold previous-session events. The gate below reads the *new* session's\n // sessionStartTime, so legitimately-sendable old-session events get suppressed.\n // Follow-up: track start time per buffered batch instead of globally.\n this.pageLeaveFns = [\n ...this.pageLeaveFns,\n () => {\n if (!this.config || !this.identifiers?.sessionId || !rrwebEventManager) return;\n const events = rrwebEventManager.getBeaconEvents();\n if (!events.length) return;\n const deviceId = this.getDeviceId();\n if (!deviceId) return;\n if (this.isBelowMinSessionDuration()) return;\n rrwebEventManager.trackDestination.sendBeacon({\n events,\n sessionId: this.identifiers.sessionId,\n deviceId,\n apiKey: this.config.apiKey,\n serverZone: this.config.serverZone,\n });\n },\n ];\n\n await this.initializeNetworkObservers();\n\n // Enable background capture when this page is opened by the Amplitude app\n // (window.opener exists). Uses the shared messenger singleton so that if\n // autocapture is also loaded, both share a single messenger and script load.\n if (getGlobalScope()?.opener) {\n const messenger = getOrCreateWindowMessenger();\n enableBackgroundCapture(messenger);\n messenger.setup({\n logger: this.loggerProvider,\n ...(this.config.serverZone && { endpoint: AMPLITUDE_ORIGINS_MAP[this.config.serverZone] }),\n });\n }\n\n this.loggerProvider.log('Installing @amplitude/session-replay-browser.');\n\n this.teardownEventListeners(false);\n\n await this.evaluateTargetingAndCapture(\n { userProperties: options.userProperties, page: this.getCurrentPageForTargeting() },\n true,\n );\n\n const needsUrlTracking = this.config.targetingConfig || (this.config.privacyConfig?.urlMaskLevels?.length ?? 0) > 0;\n if (needsUrlTracking) {\n this.setupUrlChangeListener();\n }\n }\n\n setSessionId(sessionId: string | number, deviceId?: string) {\n return returnWrapper(this.asyncSetSessionId(sessionId, deviceId));\n }\n\n async asyncSetSessionId(\n sessionId: string | number,\n deviceId?: string,\n options?: { userProperties?: { [key: string]: any } },\n ) {\n // Invalidate any in-flight URL-change re-evaluations from the previous session.\n this.latestUrlChangeTargetingEvaluationId++;\n this.sessionTargetingMatch = false;\n this.lastShouldRecordDecision = undefined; // Reset targeting decision for new session\n\n const previousSessionId = this.identifiers && this.identifiers.sessionId;\n if (previousSessionId) {\n this.sendEvents(previousSessionId);\n }\n\n const isSessionChange = previousSessionId !== sessionId;\n\n // Drop any beacon-buffered events from the previous session BEFORE installing the\n // new identifiers / start time. Otherwise the page-leave beacon path could attribute\n // old-session events to the new session id, and the gate (using the new start time)\n // would compute the wrong elapsed duration. Skip on a redundant same-sessionId call —\n // the buffer belongs to the *continuing* session and should ship via beacon as normal.\n if (isSessionChange) {\n this.rrwebEventManager?.dropPendingBeaconEvents();\n }\n\n const deviceIdForReplayId = deviceId || this.getDeviceId();\n this.identifiers = new SessionIdentifiers({\n sessionId: sessionId,\n deviceId: deviceIdForReplayId,\n });\n\n // Gate state and persisted start time only get reset on an actual session boundary.\n // A redundant setSessionId(currentId) call would otherwise overwrite both the in-memory\n // and stored start time with a fresh Date.now() — restarting the gate clock for what is\n // supposed to be a continuing session.\n if (isSessionChange) {\n this.sessionStartTime = Date.now();\n this.suppressedSendCount = 0;\n this.hasEmittedGateDecision = false;\n if (this.config?.apiKey) {\n setReplayStartTime(this.config.apiKey, sessionId, this.sessionStartTime, this.loggerProvider);\n if (previousSessionId !== undefined) {\n removeReplayStartTime(this.config.apiKey, previousSessionId, this.loggerProvider);\n }\n }\n }\n\n // If there is no previous session id, SDK is being initialized for the first time,\n // and config was just fetched in initialization, so no need to fetch it a second time\n if (this.joinedConfigGenerator && previousSessionId) {\n const { joinedConfig } = await this.joinedConfigGenerator.generateJoinedConfig();\n this.config = joinedConfig;\n }\n\n if (this.config?.targetingConfig) {\n await this.evaluateTargetingAndCapture(\n { userProperties: options?.userProperties, page: this.getCurrentPageForTargeting() },\n false,\n true,\n );\n } else {\n await this.recordEvents();\n }\n }\n\n getSessionReplayProperties() {\n const config = this.config;\n const identifiers = this.identifiers;\n if (!config || !identifiers) {\n this.loggerProvider.warn('Session replay init has not been called, cannot get session replay properties.');\n return {};\n }\n\n const shouldRecord = this.getShouldRecord();\n let eventProperties: { [key: string]: string | null } = {};\n\n if (shouldRecord) {\n eventProperties = {\n [DEFAULT_SESSION_REPLAY_PROPERTY]: identifiers.sessionReplayId ? identifiers.sessionReplayId : null,\n };\n if (config.debugMode) {\n eventProperties[SESSION_REPLAY_DEBUG_PROPERTY] = JSON.stringify({\n appHash: generateHashCode(config.apiKey).toString(),\n });\n }\n }\n\n void this.addCustomRRWebEvent(\n CustomRRwebEvent.GET_SR_PROPS,\n {\n shouldRecord,\n eventProperties: eventProperties,\n },\n this.eventCount === 10,\n );\n if (this.eventCount === 10) {\n this.eventCount = 0;\n }\n this.eventCount++;\n\n return eventProperties;\n }\n\n blurListener = () => {\n this.sendEvents();\n };\n\n focusListener = () => {\n if (this.recordCancelCallback && this.recordFunction) {\n try {\n this.recordFunction.takeFullSnapshot(true);\n } catch (error) {\n this.loggerProvider.warn('Failed to take full snapshot on focus:', error);\n }\n } else if (!this.recordEventsInFlight) {\n void this.recordEvents(false);\n }\n };\n\n /**\n * This is an instance member so that if init is called multiple times\n * it doesn't add another listener to the page leave event. This is to\n * prevent duplicate listener actions from firing.\n */\n private pageLeaveListener = (e: PageTransitionEvent | Event) => {\n // Synchronously drain any events still queued in the requestIdleCallback\n // pipeline so they are available to send before the page unloads.\n this.eventCompressor?.flushQueue();\n this.sendEvents();\n this.pageLeaveFns.forEach((fn) => {\n fn(e);\n });\n };\n\n evaluateTargetingAndCapture = async (\n targetingParams: SessionReplayTargetingInput,\n isInit = false,\n forceRestart = false,\n forceTargetingReevaluation = false,\n ) => {\n if (!this.identifiers || !this.identifiers.sessionId || !this.config) {\n if (this.identifiers && !this.identifiers.sessionId) {\n this.loggerProvider.log('Session ID has not been set yet, cannot evaluate targeting for Session Replay.');\n } else {\n this.loggerProvider.warn('Session replay init has not been called, cannot evaluate targeting.');\n }\n return;\n }\n\n // Handle cases where there's no targeting config\n if (!this.config.targetingConfig) {\n if (isInit) {\n this.loggerProvider.log('Targeting config has not been set yet, cannot evaluate targeting.');\n } else {\n this.loggerProvider.log('No targeting config set, skipping initialization/recording for event.');\n return;\n }\n }\n\n // Store targeting parameters for use in getShouldRecord\n this.lastTargetingParams = targetingParams;\n\n // Re-evaluate only until we get the first match in this session.\n // Once matched, keep recording for the rest of the session.\n const targetingConfig = this.config.targetingConfig;\n const shouldReEvaluate = targetingConfig && !this.sessionTargetingMatch;\n if (shouldReEvaluate) {\n // Capture URL-change evaluation id so out-of-order async completions can be discarded.\n const urlChangeEvaluationId = forceTargetingReevaluation ? this.latestUrlChangeTargetingEvaluationId : undefined;\n let eventForTargeting = targetingParams.event;\n if (\n eventForTargeting &&\n Object.values(SpecialEventType).includes(eventForTargeting.event_type as SpecialEventType)\n ) {\n eventForTargeting = undefined;\n }\n\n const pageUrl = targetingParams.page?.url ?? getGlobalScope()?.location?.href ?? '';\n const pageForTargeting = targetingParams.page ?? (pageUrl !== '' ? { url: pageUrl } : undefined);\n\n const targetingMatch = await evaluateTargetingAndStore({\n sessionId: this.identifiers.sessionId,\n targetingConfig,\n loggerProvider: this.loggerProvider,\n apiKey: this.config.apiKey,\n targetingParams: {\n userProperties: targetingParams.userProperties,\n event: eventForTargeting,\n page: pageForTargeting,\n },\n urlChange: forceTargetingReevaluation,\n });\n\n if (\n forceTargetingReevaluation &&\n urlChangeEvaluationId !== undefined &&\n urlChangeEvaluationId !== this.latestUrlChangeTargetingEvaluationId\n ) {\n this.loggerProvider.debug(\n `Ignoring stale URL-change targeting result #${urlChangeEvaluationId}; latest is #${this.latestUrlChangeTargetingEvaluationId}.`,\n );\n return;\n }\n // Keep targeting match monotonic within a session: once true, always true.\n // This avoids races where an older in-flight evaluation resolves false after\n // a newer evaluation already resolved true.\n this.sessionTargetingMatch = this.sessionTargetingMatch || targetingMatch;\n\n this.loggerProvider.debug(\n JSON.stringify(\n {\n name: 'targeted replay capture config',\n sessionTargetingMatch: this.sessionTargetingMatch,\n event: eventForTargeting,\n targetingParams: targetingParams,\n },\n null,\n 2,\n ),\n );\n }\n\n if (isInit) {\n void this.initialize(true);\n } else if (forceRestart || !this.recordCancelCallback) {\n this.loggerProvider.log('Recording events for session due to forceRestart or no ongoing recording.');\n await this.recordEvents();\n }\n };\n\n sendEvents(sessionId?: string | number) {\n const sessionIdToSend = sessionId || this.identifiers?.sessionId;\n const deviceId = this.getDeviceId();\n if (this.eventsManager && sessionIdToSend && deviceId) {\n if (this.isBelowMinSessionDuration()) {\n // Safe to dereference: isBelowMinSessionDuration() only returns true when both\n // this.config and this.sessionStartTime are defined.\n const startTime = this.sessionStartTime as number;\n const minMs = (this.config as SessionReplayJoinedConfig).minSessionDurationMs as number;\n this.loggerProvider.log(\n `Session ${sessionIdToSend} not sent: duration ${Date.now() - startTime}ms is below minimum ${minMs}ms.`,\n );\n // We deliberately do NOT clear the beacon buffer here. Blur/visibility-change\n // can call sendEvents() mid-session; if the session later crosses the threshold\n // those buffered events are legitimately sendable via beacon on page exit.\n // Cross-session leak is prevented in asyncSetSessionId, which drops the buffer\n // at the session transition. The page-leave path also gates independently.\n this.suppressedSendCount++;\n return;\n }\n // On the first send-after-pass for the session, emit a custom rrweb event so the\n // payload itself carries the gate verdict. Lets backend ingestion diff intended\n // vs actual replay counts to spot start-time-tracking regressions.\n //\n // Gate on recording being active too: addCustomRRWebEvent is a no-op when\n // recordCancelCallback/recordFunction aren't set yet (e.g. a blur-listener-fired\n // sendEvents reaches us before _recordEvents() activates on a reloaded long-lived\n // session). Tripping hasEmittedGateDecision unconditionally would lose the signal\n // for that session's first real send.\n if (\n !this.hasEmittedGateDecision &&\n this.config?.minSessionDurationMs !== undefined &&\n this.recordCancelCallback &&\n this.recordFunction\n ) {\n this.hasEmittedGateDecision = true;\n const elapsedMs = this.sessionStartTime !== undefined ? Date.now() - this.sessionStartTime : undefined;\n void this.addCustomRRWebEvent(\n CustomRRwebEvent.REPLAY_GATE_DECISION,\n {\n sessionId: sessionIdToSend,\n suppressedSendCount: this.suppressedSendCount,\n elapsedMs,\n minSessionDurationMs: this.config.minSessionDurationMs,\n },\n false,\n );\n }\n this.eventsManager.sendCurrentSequenceEvents({ sessionId: sessionIdToSend, deviceId });\n }\n }\n\n async initialize(shouldSendStoredEvents = false) {\n if (!this.identifiers?.sessionId) {\n this.loggerProvider.log(`Session is not being recorded due to lack of session id.`);\n return Promise.resolve();\n }\n\n const deviceId = this.getDeviceId();\n if (!deviceId) {\n this.loggerProvider.log(`Session is not being recorded due to lack of device id.`);\n return Promise.resolve();\n }\n this.eventsManager && shouldSendStoredEvents && void this.eventsManager.sendStoredEvents({ deviceId });\n\n return this.recordEvents();\n }\n\n shouldOptOut() {\n let identityStoreOptOut: boolean | undefined;\n if (this.config?.instanceName) {\n const identityStore = getAnalyticsConnector(this.config.instanceName).identityStore;\n identityStoreOptOut = identityStore.getIdentity().optOut;\n }\n\n return identityStoreOptOut !== undefined ? identityStoreOptOut : this.config?.optOut;\n }\n\n getShouldRecord() {\n if (!this.identifiers || !this.config || !this.identifiers.sessionId) {\n this.loggerProvider.warn(`Session is not being recorded due to lack of config, please call sessionReplay.init.`);\n return false;\n }\n\n if (!this.config.captureEnabled) {\n this.loggerProvider.log(\n `Session ${this.identifiers.sessionId} not being captured due to capture being disabled for project or because the remote config could not be fetched.`,\n );\n return false;\n }\n\n if (this.shouldOptOut()) {\n this.loggerProvider.log(`Opting session ${this.identifiers.sessionId} out of recording due to optOut config.`);\n return false;\n }\n\n let shouldRecord = false;\n let message = '';\n let matched = false;\n\n // If targetingConfig exists, we'll use the sessionTargetingMatch to determine whether to record\n // Otherwise, we'll evaluate the session against the overall sample rate\n if (this.config.targetingConfig) {\n if (!this.sessionTargetingMatch) {\n message = `Not capturing replays for session ${this.identifiers.sessionId} due to not matching targeting conditions.`;\n this.loggerProvider.log(message);\n shouldRecord = false;\n matched = false;\n } else {\n message = `Capturing replays for session ${this.identifiers.sessionId} due to matching targeting conditions.`;\n this.loggerProvider.log(message);\n shouldRecord = true;\n matched = true;\n }\n } else {\n const isInSample = isSessionInSample(this.identifiers.sessionId, this.config.sampleRate);\n if (!isInSample) {\n message = `Opting session ${this.identifiers.sessionId} out of recording due to sample rate.`;\n this.loggerProvider.log(message);\n shouldRecord = false;\n matched = false;\n } else {\n shouldRecord = true;\n matched = true;\n }\n }\n\n // Only send custom rrweb event for targeting decision when the decision changes\n if (this.lastShouldRecordDecision !== shouldRecord && this.config.targetingConfig) {\n void this.addCustomRRWebEvent(CustomRRwebEvent.TARGETING_DECISION, {\n message,\n sessionId: this.identifiers.sessionId,\n matched,\n targetingParams: this.lastTargetingParams,\n });\n this.lastShouldRecordDecision = shouldRecord;\n }\n\n return shouldRecord;\n }\n\n getBlockSelectors(): string | string[] | undefined {\n // For some reason, this defaults to empty array ([]) if undefined in the compiled script.\n // Empty arrays cause errors when being evaluated in Safari.\n // Force the selector to be undefined if it's an empty array.\n const blockSelector = this.config?.privacyConfig?.blockSelector ?? [];\n if (blockSelector.length === 0) {\n return undefined;\n }\n return blockSelector;\n }\n\n getMaskTextSelectors(): string | undefined {\n const privacyConfig = this.config?.privacyConfig;\n const effectiveLevel = privacyConfig ? getEffectiveMaskLevel(this.currentPageUrl, privacyConfig) : undefined;\n\n if (effectiveLevel === 'conservative') {\n return '*';\n }\n\n // If any urlMaskLevels rule uses 'conservative', always route all text nodes\n // through maskTextFn so the dynamic URL getter can decide at call time.\n // Without this, rrweb's static maskTextSelector would miss text nodes when\n // the user navigates from a non-conservative page to a conservative one.\n if (privacyConfig?.urlMaskLevels?.some((rule) => rule.maskLevel === 'conservative')) {\n return '*';\n }\n\n // If defaultMaskLevel is 'conservative' and URL rules exist, always route text through\n // maskTextFn — a page matching no rule falls back to the conservative default, and\n // rrweb must be set up at start to call maskTextFn for those text nodes.\n const urlMaskLevels = privacyConfig?.urlMaskLevels;\n if (privacyConfig?.defaultMaskLevel === 'conservative' && urlMaskLevels && urlMaskLevels.length > 0) {\n return '*';\n }\n\n const maskSelector = privacyConfig?.maskSelector;\n if (!maskSelector) {\n return;\n }\n\n return maskSelector as unknown as string;\n }\n\n async getRecordingPlugins(loggingConfig: LoggingConfig | undefined) {\n const plugins = [];\n\n // Add URL tracking plugin\n try {\n const urlTrackingPlugin = createUrlTrackingPlugin({\n ugcFilterRules: this.config?.interactionConfig?.ugcFilterRules || [],\n enablePolling: this.config?.enableUrlChangePolling || false,\n pollingInterval: this.config?.urlChangePollingInterval,\n captureDocumentTitle: this.config?.captureDocumentTitle,\n });\n\n plugins.push(urlTrackingPlugin);\n } catch (error) {\n this.loggerProvider.warn('Failed to create URL tracking plugin:', error);\n }\n\n // Default plugin settings -\n // {\n // level: ['info', 'log', 'warn', 'error'],\n // lengthThreshold: 10000,\n // stringifyOptions: {\n // stringLengthLimit: undefined,\n // numOfKeysLimit: 50,\n // depthOfLimit: 4,\n // },\n // logger: window.console,\n // }\n if (loggingConfig?.console?.enabled) {\n try {\n // Dynamic import keeps console plugin separate and only loads when needed\n const { getRecordConsolePlugin } = await import('@amplitude/rrweb-plugin-console-record');\n plugins.push(getRecordConsolePlugin({ level: loggingConfig.console.levels }));\n } catch (error) {\n this.loggerProvider.warn('Failed to load console plugin:', error);\n }\n }\n\n return plugins.length > 0 ? plugins : undefined;\n }\n\n private async getRecordFunction(): Promise<RecordFunction | null> {\n if (this.recordFunction) {\n return this.recordFunction;\n }\n\n try {\n const { record } = await import('@amplitude/rrweb-record');\n this.recordFunction = record;\n return record;\n } catch (error) {\n this.loggerProvider.warn('Failed to load rrweb-record module:', error);\n return null;\n }\n }\n\n async recordEvents(shouldLogMetadata = true) {\n if (this.recordEventsInFlight) {\n this.recordEventsPendingShouldLogMetadata = shouldLogMetadata;\n return;\n }\n this.recordEventsInFlight = true;\n try {\n await this._recordEvents(shouldLogMetadata);\n while (this.recordEventsPendingShouldLogMetadata !== null) {\n const pendingArgs = this.recordEventsPendingShouldLogMetadata;\n this.recordEventsPendingShouldLogMetadata = null;\n await this._recordEvents(pendingArgs);\n }\n } finally {\n this.recordEventsInFlight = false;\n this.recordEventsPendingShouldLogMetadata = null;\n }\n }\n\n private async _recordEvents(shouldLogMetadata = true) {\n const config = this.config;\n const shouldRecord = this.getShouldRecord();\n const sessionId = this.identifiers?.sessionId;\n if (!shouldRecord || !sessionId || !config) {\n return;\n }\n this.stopRecordingEvents();\n\n const recordFunction = await this.getRecordFunction();\n\n // May be undefined if cannot import rrweb-record\n if (!recordFunction) {\n return;\n }\n\n await this.initializeNetworkObservers();\n\n const networkLoggingConfig = config.loggingConfig?.network;\n const trackUrl = getServerUrl(config.serverZone, config.trackServerUrl);\n const ignoredUrls = [SESSION_REPLAY_SERVER_URL, SESSION_REPLAY_EU_URL, SESSION_REPLAY_STAGING_URL, trackUrl];\n this.networkObservers?.start((event: NetworkRequestEvent) => {\n if (ignoredUrls.some((url) => event.url.startsWith(url))) return;\n void this.addCustomRRWebEvent(CustomRRwebEvent.FETCH_REQUEST, event);\n }, networkLoggingConfig);\n const { interactionConfig, loggingConfig } = config;\n\n const hooks = interactionConfig?.enabled\n ? {\n mouseInteraction:\n this.eventsManager &&\n this.clickHandler?.createHook({\n eventsManager: this.eventsManager,\n sessionId,\n deviceIdFn: this.getDeviceId.bind(this),\n mirror: recordFunction.mirror,\n ugcFilterRules: interactionConfig.ugcFilterRules ?? [],\n performanceOptions: config.performanceConfig?.interaction,\n }),\n scroll: this.scrollHook,\n }\n : {};\n\n const ugcFilterRules =\n interactionConfig?.enabled && interactionConfig.ugcFilterRules ? interactionConfig.ugcFilterRules : [];\n\n this.loggerProvider.log(`Session Replay capture beginning for ${sessionId}.`);\n\n try {\n const crossOriginIframesEnabled = !!config.crossOriginIframes?.enabled;\n const coordinateChildren = config.crossOriginIframes?.coordinateChildren !== false;\n const childMode = crossOriginIframesEnabled && isInIframe();\n\n if (childMode && coordinateChildren) {\n // Child mode: don't self-start; wait for a start signal from the parent.\n // (The previous listener, if any, was already removed by stopRecordingEvents above.)\n this.crossOriginParentSignalCleanup = listenForParentSignals({\n onStart: () => this._recordEventsInChildMode(recordFunction, sessionId, config, hooks),\n onStop: () => {\n try {\n // Only cancel the rrweb recording — do NOT call stopRecordingEvents() here,\n // which would clear crossOriginParentSignalCleanup and make the child deaf\n // to subsequent start/stop cycles from the parent.\n this.recordCancelCallback && this.recordCancelCallback();\n this.recordCancelCallback = null;\n } catch (error) {\n const typedError = error as Error;\n this.loggerProvider.warn(\n `Error occurred while stopping child iframe replay capture: ${typedError.toString()}`,\n );\n }\n },\n });\n return;\n }\n\n this.recordCancelCallback = recordFunction({\n ...this.buildRRWebRecordOptions(\n config,\n hooks,\n (event: eventWithTime) => {\n if (this.shouldOptOut()) {\n this.loggerProvider.log(`Opting session ${sessionId} out of recording due to optOut config.`);\n this.stopRecordingEvents();\n this.sendEvents();\n return;\n }\n\n if (event.type === RRWebEventType.Meta) {\n event.data.href = getPageUrl(event.data.href, ugcFilterRules);\n }\n\n if (this.eventCompressor) {\n // Schedule processing during idle time if the browser supports requestIdleCallback\n this.eventCompressor.enqueueEvent(event, sessionId);\n } else {\n // eventCompressor is not yet ready (concurrent call racing _init()).\n // Buffer the event so it can be flushed once the compressor is initialized.\n this.pendingEmitEvents.push({ event, sessionId });\n }\n },\n 'Error while capturing replay: ',\n ),\n plugins: await this.getRecordingPlugins(loggingConfig),\n recordCrossOriginIframes: crossOriginIframesEnabled,\n });\n\n if (crossOriginIframesEnabled && !childMode && coordinateChildren) {\n if (!this.crossOriginIframeCoordinator) {\n this.crossOriginIframeCoordinator = new CrossOriginIframeCoordinator();\n }\n this.crossOriginIframeCoordinator.start();\n }\n\n void this.addCustomRRWebEvent(CustomRRwebEvent.DEBUG_INFO);\n if (shouldLogMetadata) {\n void this.addCustomRRWebEvent(CustomRRwebEvent.METADATA, this.metadata);\n }\n } catch (error) {\n this.loggerProvider.warn('Failed to initialize session replay:', error);\n }\n }\n\n private buildRRWebRecordOptions(\n config: SessionReplayJoinedConfig,\n hooks: { mouseInteraction?: any; scroll?: scrollCallback },\n emit: (event: eventWithTime) => void,\n errorLogPrefix: string,\n ): Parameters<RecordFunction>[0] {\n const { privacyConfig } = config;\n return {\n emit,\n inlineStylesheet: config.shouldInlineStylesheet,\n hooks,\n maskAllInputs: true,\n maskTextClass: MASK_TEXT_CLASS,\n blockClass: BLOCK_CLASS,\n blockSelector: this.getBlockSelectors() as string | undefined,\n applyBackgroundColorToBlockedElements: config.applyBackgroundColorToBlockedElements,\n maskInputFn: maskFn('input', privacyConfig, () => this.currentPageUrl),\n maskTextFn: maskFn('text', privacyConfig, () => this.currentPageUrl),\n maskAttributeFn: maskAttributeFn(privacyConfig, () => this.currentPageUrl),\n maskTextSelector: this.getMaskTextSelectors(),\n ...(config.fullSnapshotIntervalMs !== undefined && { checkoutEveryNms: config.fullSnapshotIntervalMs }),\n recordCanvas: false,\n captureAdoptedStyleSheets: config.captureAdoptedStyleSheets,\n // Strip nodes that are never rendered by the rrweb replay player.\n // None of these affect visual fidelity; omitting them reduces snapshot size.\n slimDOMOptions: {\n script: true,\n comment: true,\n headFavicon: true,\n headWhitespace: true,\n headMetaDescKeywords: true,\n headMetaSocial: true,\n headMetaRobots: true,\n headMetaHttpEquiv: true,\n headMetaAuthorship: true,\n headMetaVerification: true,\n },\n errorHandler: (error: unknown) => {\n const typedError = error as Error & { _external_?: boolean };\n // styled-components relies on this error being thrown and bubbled up, rrweb is otherwise suppressing it\n if (typedError.message.includes('insertRule') && typedError.message.includes('CSSStyleSheet')) {\n throw typedError;\n }\n // rrweb monkey-patches window functions like CSSStyleSheet.insertRule; errors from external callers\n // (e.g. styled-components) must be re-thrown so they aren't silently swallowed.\n if (typedError._external_) {\n throw typedError;\n }\n this.loggerProvider.warn(errorLogPrefix, typedError.toString());\n // Return true so that we don't clutter user's consoles with internal rrweb errors\n return true;\n },\n };\n }\n\n private _recordEventsInChildMode(\n recordFunction: RecordFunction,\n sessionId: string | number,\n config: SessionReplayJoinedConfig,\n hooks: { mouseInteraction?: any; scroll?: scrollCallback },\n ) {\n // In child mode, rrweb detects window.parent !== window and routes events via\n // postMessage to the parent instead of calling emit. The emit callback is unused.\n // Note: recording plugins (URL tracking, console capture) are intentionally omitted\n // here — the child's events are merged into the parent stream, so URL changes inside\n // the iframe should not be recorded as parent page-view events.\n try {\n // Stop only the previous rrweb recording. Do NOT call stopRecordingEvents() here:\n // that would clear crossOriginParentSignalCleanup — the very listener that invoked\n // this method — making the child permanently deaf to subsequent stop/start cycles.\n this.recordCancelCallback && this.recordCancelCallback();\n this.recordCancelCallback = null;\n this.recordCancelCallback = recordFunction({\n ...this.buildRRWebRecordOptions(\n config,\n hooks,\n () => {\n // no-op: child events are forwarded to parent via postMessage by rrweb\n },\n 'Error while capturing replay (child iframe): ',\n ),\n recordCrossOriginIframes: true, // child mode is only entered when crossOriginIframes.enabled is true\n });\n this.loggerProvider.log(`Session Replay child iframe capture beginning for session ${sessionId}.`);\n } catch (error) {\n this.loggerProvider.warn('Failed to initialize session replay in child iframe mode:', error);\n }\n }\n\n addCustomRRWebEvent = async (\n eventName: CustomRRwebEvent,\n eventData: { [key: string]: any } = {},\n addStorageInfo = true,\n ) => {\n try {\n let debugInfo: DebugInfo | undefined = undefined;\n const config = this.config;\n // Only add debug info for non-metadata events\n if (config && eventName !== CustomRRwebEvent.METADATA) {\n debugInfo = {\n config: getDebugConfig(config),\n version: VERSION,\n };\n if (addStorageInfo) {\n const storageSizeData = await getStorageSize();\n debugInfo = {\n ...storageSizeData,\n ...debugInfo,\n };\n }\n }\n // Check first to ensure we are recording\n if (this.recordCancelCallback && this.recordFunction) {\n this.recordFunction.addCustomEvent(eventName, {\n ...eventData,\n ...debugInfo,\n });\n } else {\n this.loggerProvider.debug(\n `Not able to add custom replay capture event ${eventName} due to no ongoing recording.`,\n );\n }\n } catch (e) {\n this.loggerProvider.debug('Error while adding custom replay capture event: ', e);\n }\n };\n\n stopRecordingEvents = () => {\n try {\n this.loggerProvider.log('Session Replay capture stopping.');\n this.recordCancelCallback && this.recordCancelCallback();\n this.recordCancelCallback = null;\n this.networkObservers?.stop();\n this.crossOriginIframeCoordinator?.stop();\n this.crossOriginIframeCoordinator = null;\n // Remove the child-mode parent signal listener so a later mode change\n // (e.g. crossOriginIframes disabled) does not leave a stale listener.\n this.crossOriginParentSignalCleanup?.();\n this.crossOriginParentSignalCleanup = null;\n } catch (error) {\n const typedError = error as Error;\n this.loggerProvider.warn(`Error occurred while stopping replay capture: ${typedError.toString()}`);\n }\n };\n\n getDeviceId() {\n return this.identifiers?.deviceId;\n }\n\n getSessionId() {\n return this.identifiers?.sessionId;\n }\n\n async flush(useRetry = false) {\n // Intentionally not gated on min_session_duration_ms. flush() forwards payloads\n // already queued in trackDestination, and every code path that queues into it —\n // sendCurrentSequenceEvents, addEvent's batch-split path, sendStoredEvents reading\n // sequencesToSend from IDB — has already passed the gate at the time of queuing.\n // A duplicate gate here would be unreachable.\n return this.eventsManager?.flush(useRetry);\n }\n\n shutdown() {\n this.urlChangeCleanup?.();\n this.crossOriginParentSignalCleanup?.();\n this.crossOriginParentSignalCleanup = null;\n this.teardownEventListeners(true);\n this.stopRecordingEvents();\n this.sendEvents();\n }\n\n private mapSDKType(sdkType: string | undefined) {\n if (sdkType === 'plugin') {\n return '@amplitude/plugin-session-replay-browser';\n }\n\n if (sdkType === 'segment') {\n return '@amplitude/segment-session-replay-plugin';\n }\n\n return null;\n }\n\n private setMetadata(\n sessionId: string | number | undefined,\n joinedConfig: SessionReplayJoinedConfig,\n localConfig: SessionReplayLocalConfig,\n remoteConfig: SessionReplayRemoteConfig | undefined,\n replaySDKVersion: string | undefined,\n standaloneSDKVersion: string | undefined,\n sdkType: string | undefined,\n ) {\n const hashValue = sessionId?.toString() ? generateHashCode(sessionId.toString()) : undefined;\n\n this.metadata = {\n joinedConfig,\n localConfig,\n remoteConfig,\n sessionId,\n hashValue,\n sampleRate: joinedConfig.sampleRate,\n replaySDKType: this.mapSDKType(sdkType),\n replaySDKVersion,\n standaloneSDKType: '@amplitude/session-replay-browser',\n standaloneSDKVersion,\n };\n }\n\n private async initializeNetworkObservers(): Promise<void> {\n if (this.config?.loggingConfig?.network?.enabled && !this.networkObservers) {\n try {\n const { NetworkObservers: NetworkObserversClass } = await import('./observers');\n this.networkObservers = new NetworkObserversClass();\n } catch (error) {\n this.loggerProvider.warn('Failed to import or instantiate NetworkObservers:', error);\n }\n }\n }\n}\n"]}
@@ -12,6 +12,7 @@ export declare class SessionReplayTrackDestination implements AmplitudeSessionRe
12
12
  storageKey: string;
13
13
  trackServerUrl?: string;
14
14
  retryTimeout: number;
15
+ private enableTransportCompression;
15
16
  private scheduled;
16
17
  payloadBatcher: PayloadBatcher;
17
18
  queue: SessionReplayDestinationContext[];
@@ -19,12 +20,15 @@ export declare class SessionReplayTrackDestination implements AmplitudeSessionRe
19
20
  private sendIdCounter;
20
21
  private pendingWorkerRequests;
21
22
  private flushPauseUntilMs;
23
+ private mergeOnNextFlush;
24
+ private mergeLogFiredThisPause;
22
25
  private killedSessions;
23
- constructor({ trackServerUrl, loggerProvider, payloadBatcher, workerScript, }: {
26
+ constructor({ trackServerUrl, loggerProvider, payloadBatcher, workerScript, enableTransportCompression, }: {
24
27
  trackServerUrl?: string;
25
28
  loggerProvider: ILogger;
26
29
  payloadBatcher?: PayloadBatcher;
27
30
  workerScript?: string;
31
+ enableTransportCompression?: boolean;
28
32
  });
29
33
  sendEventsList(destinationData: SessionReplayDestination): void;
30
34
  /**
@@ -44,6 +48,20 @@ export declare class SessionReplayTrackDestination implements AmplitudeSessionRe
44
48
  addToQueue(...list: SessionReplayDestinationContext[]): void;
45
49
  schedule(timeout: number): void;
46
50
  flush(useRetry?: boolean): Promise<void>;
51
+ /**
52
+ * Coalesces queued contexts that share the same destination identity so the post-throttle
53
+ * release sends fewer requests. Identity covers everything that affects the request URL,
54
+ * routing, or per-request semantics — splitting on any difference keeps each merged POST
55
+ * indistinguishable from the source contexts it replaced.
56
+ *
57
+ * Greedy concat with a soft char-length cap (`MERGE_AFTER_THROTTLE_SOFT_CAP`) keeps merged
58
+ * payloads well under the 413 ceiling; on the rare oversized merge, the existing
59
+ * split-and-retry path still bisects safely.
60
+ *
61
+ * The merged context's `onComplete` fans out to every source context's callback so each
62
+ * underlying IDB sequence record is cleaned up exactly once on success.
63
+ */
64
+ private mergeQueueAfterThrottle;
47
65
  send(context: SessionReplayDestinationContext, useRetry?: boolean): Promise<void>;
48
66
  private sendViaWorker;
49
67
  private sendOnMainThread;
@@ -1 +1 @@
1
- {"version":3,"file":"track-destination.d.ts","sourceRoot":"","sources":["../../src/track-destination.ts"],"names":[],"mappings":"AAAA,OAAO,EAAiC,OAAO,EAAE,UAAU,EAAU,MAAM,2BAA2B,CAAC;AAUvG,OAAO,EACL,6BAA6B,IAAI,sCAAsC,EACvE,wBAAwB,EACxB,+BAA+B,EAChC,MAAM,0BAA0B,CAAC;AAmClC,MAAM,MAAM,cAAc,GAAG,CAAC,EAAE,OAAO,EAAE,MAAM,EAAE,EAAE;IAAE,OAAO,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,EAAE,CAAA;CAAE,KAAK;IAC3F,OAAO,EAAE,MAAM,CAAC;IAChB,MAAM,EAAE,OAAO,EAAE,CAAC;CACnB,CAAC;AAMF,qBAAa,6BAA8B,YAAW,sCAAsC;IAC1F,cAAc,EAAE,OAAO,CAAC;IACxB,UAAU,SAAM;IAChB,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,YAAY,SAAQ;IACpB,OAAO,CAAC,SAAS,CAA8C;IAC/D,cAAc,EAAE,cAAc,CAAC;IAC/B,KAAK,EAAE,+BAA+B,EAAE,CAAM;IAC9C,OAAO,CAAC,MAAM,CAAC,CAAS;IACxB,OAAO,CAAC,aAAa,CAAK;IAC1B,OAAO,CAAC,qBAAqB,CAAwF;IAIrH,OAAO,CAAC,iBAAiB,CAAK;IAC9B,OAAO,CAAC,cAAc,CAA8B;gBAExC,EACV,cAAc,EACd,cAAc,EACd,cAAc,EACd,YAAY,GACb,EAAE;QACD,cAAc,CAAC,EAAE,MAAM,CAAC;QACxB,cAAc,EAAE,OAAO,CAAC;QACxB,cAAc,CAAC,EAAE,cAAc,CAAC;QAChC,YAAY,CAAC,EAAE,MAAM,CAAC;KACvB;IA0DD,cAAc,CAAC,eAAe,EAAE,wBAAwB;IAQxD;;;;;;OAMG;IACH,UAAU,CAAC,EACT,MAAM,EACN,SAAS,EACT,QAAQ,EACR,MAAM,EACN,UAAU,GACX,EAAE;QACD,MAAM,EAAE,MAAM,EAAE,CAAC;QACjB,SAAS,EAAE,MAAM,GAAG,MAAM,CAAC;QAC3B,QAAQ,EAAE,MAAM,CAAC;QACjB,MAAM,EAAE,MAAM,CAAC;QACf,UAAU,CAAC,EAAE,MAAM,OAAO,UAAU,CAAC;KACtC;IAgDD,UAAU,CAAC,GAAG,IAAI,EAAE,+BAA+B,EAAE;IA2BrD,QAAQ,CAAC,OAAO,EAAE,MAAM;IAelB,KAAK,CAAC,QAAQ,UAAQ;IActB,IAAI,CAAC,OAAO,EAAE,+BAA+B,EAAE,QAAQ,UAAO;YAkCtD,aAAa;YAgCb,gBAAgB;IA+ExB,aAAa,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,+BAA+B,EAAE,YAAY,SAAK;IAwB/F,6BAA6B,CAAC,OAAO,EAAE,+BAA+B,EAAE,KAAK,EAAE,OAAO,GAAG,IAAI;IAgC7F,qBAAqB,CAAC,OAAO,EAAE,+BAA+B;IAQxD,mBAAmB,CAAC,OAAO,EAAE,+BAA+B;IAWlE,eAAe,CAAC,EACd,OAAO,EACP,GAAG,EACH,OAAO,GACR,EAAE;QACD,OAAO,EAAE,+BAA+B,CAAC;QACzC,GAAG,CAAC,EAAE,MAAM,CAAC;QACb,OAAO,CAAC,EAAE,MAAM,CAAC;KAClB;IASD;;;;;;;;;;OAUG;IACH,OAAO,CAAC,oBAAoB;IAyB5B,OAAO,CAAC,WAAW;CAyBpB"}
1
+ {"version":3,"file":"track-destination.d.ts","sourceRoot":"","sources":["../../src/track-destination.ts"],"names":[],"mappings":"AAAA,OAAO,EAAiC,OAAO,EAAE,UAAU,EAAU,MAAM,2BAA2B,CAAC;AAUvG,OAAO,EACL,6BAA6B,IAAI,sCAAsC,EACvE,wBAAwB,EACxB,+BAA+B,EAChC,MAAM,0BAA0B,CAAC;AAoClC,MAAM,MAAM,cAAc,GAAG,CAAC,EAAE,OAAO,EAAE,MAAM,EAAE,EAAE;IAAE,OAAO,EAAE,MAAM,CAAC;IAAC,MAAM,EAAE,MAAM,EAAE,CAAA;CAAE,KAAK;IAC3F,OAAO,EAAE,MAAM,CAAC;IAChB,MAAM,EAAE,OAAO,EAAE,CAAC;CACnB,CAAC;AAMF,qBAAa,6BAA8B,YAAW,sCAAsC;IAC1F,cAAc,EAAE,OAAO,CAAC;IACxB,UAAU,SAAM;IAChB,cAAc,CAAC,EAAE,MAAM,CAAC;IACxB,YAAY,SAAQ;IAIpB,OAAO,CAAC,0BAA0B,CAAU;IAC5C,OAAO,CAAC,SAAS,CAA8C;IAC/D,cAAc,EAAE,cAAc,CAAC;IAC/B,KAAK,EAAE,+BAA+B,EAAE,CAAM;IAC9C,OAAO,CAAC,MAAM,CAAC,CAAS;IACxB,OAAO,CAAC,aAAa,CAAK;IAC1B,OAAO,CAAC,qBAAqB,CAAwF;IAIrH,OAAO,CAAC,iBAAiB,CAAK;IAI9B,OAAO,CAAC,gBAAgB,CAAS;IAGjC,OAAO,CAAC,sBAAsB,CAAS;IACvC,OAAO,CAAC,cAAc,CAA8B;gBAExC,EACV,cAAc,EACd,cAAc,EACd,cAAc,EACd,YAAY,EACZ,0BAA0B,GAC3B,EAAE;QACD,cAAc,CAAC,EAAE,MAAM,CAAC;QACxB,cAAc,EAAE,OAAO,CAAC;QACxB,cAAc,CAAC,EAAE,cAAc,CAAC;QAChC,YAAY,CAAC,EAAE,MAAM,CAAC;QACtB,0BAA0B,CAAC,EAAE,OAAO,CAAC;KACtC;IA2DD,cAAc,CAAC,eAAe,EAAE,wBAAwB;IAQxD;;;;;;OAMG;IACH,UAAU,CAAC,EACT,MAAM,EACN,SAAS,EACT,QAAQ,EACR,MAAM,EACN,UAAU,GACX,EAAE;QACD,MAAM,EAAE,MAAM,EAAE,CAAC;QACjB,SAAS,EAAE,MAAM,GAAG,MAAM,CAAC;QAC3B,QAAQ,EAAE,MAAM,CAAC;QACjB,MAAM,EAAE,MAAM,CAAC;QACf,UAAU,CAAC,EAAE,MAAM,OAAO,UAAU,CAAC;KACtC;IAgDD,UAAU,CAAC,GAAG,IAAI,EAAE,+BAA+B,EAAE;IA2BrD,QAAQ,CAAC,OAAO,EAAE,MAAM;IAqBlB,KAAK,CAAC,QAAQ,UAAQ;IAmB5B;;;;;;;;;;;;OAYG;IACH,OAAO,CAAC,uBAAuB;IA8EzB,IAAI,CAAC,OAAO,EAAE,+BAA+B,EAAE,QAAQ,UAAO;YAkCtD,aAAa;YAiCb,gBAAgB;IA2FxB,aAAa,CAAC,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,+BAA+B,EAAE,YAAY,SAAK;IAwB/F,6BAA6B,CAAC,OAAO,EAAE,+BAA+B,EAAE,KAAK,EAAE,OAAO,GAAG,IAAI;IAuC7F,qBAAqB,CAAC,OAAO,EAAE,+BAA+B;IAQxD,mBAAmB,CAAC,OAAO,EAAE,+BAA+B;IAWlE,eAAe,CAAC,EACd,OAAO,EACP,GAAG,EACH,OAAO,GACR,EAAE;QACD,OAAO,EAAE,+BAA+B,CAAC;QACzC,GAAG,CAAC,EAAE,MAAM,CAAC;QACb,OAAO,CAAC,EAAE,MAAM,CAAC;KAClB;IASD;;;;;;;;;;OAUG;IACH,OAAO,CAAC,oBAAoB;IA0B5B,OAAO,CAAC,WAAW;CAyBpB"}