wtf.js 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. 'use strict';
  2. /**
  3. * @license Angular v<unknown>
  4. * (c) 2010-2022 Google LLC. https://angular.io/
  5. * License: MIT
  6. */
  7. /**
  8. * @fileoverview
  9. * @suppress {missingRequire}
  10. */
  11. (function (global) {
  12. // Detect and setup WTF.
  13. let wtfTrace = null;
  14. let wtfEvents = null;
  15. const wtfEnabled = (function () {
  16. const wtf = global['wtf'];
  17. if (wtf) {
  18. wtfTrace = wtf.trace;
  19. if (wtfTrace) {
  20. wtfEvents = wtfTrace.events;
  21. return true;
  22. }
  23. }
  24. return false;
  25. })();
  26. class WtfZoneSpec {
  27. constructor() {
  28. this.name = 'WTF';
  29. }
  30. onFork(parentZoneDelegate, currentZone, targetZone, zoneSpec) {
  31. const retValue = parentZoneDelegate.fork(targetZone, zoneSpec);
  32. WtfZoneSpec.forkInstance(zonePathName(targetZone), retValue.name);
  33. return retValue;
  34. }
  35. onInvoke(parentZoneDelegate, currentZone, targetZone, delegate, applyThis, applyArgs, source) {
  36. const src = source || 'unknown';
  37. let scope = WtfZoneSpec.invokeScope[src];
  38. if (!scope) {
  39. scope = WtfZoneSpec.invokeScope[src] =
  40. wtfEvents.createScope(`Zone:invoke:${source}(ascii zone)`);
  41. }
  42. return wtfTrace.leaveScope(scope(zonePathName(targetZone)), parentZoneDelegate.invoke(targetZone, delegate, applyThis, applyArgs, source));
  43. }
  44. onHandleError(parentZoneDelegate, currentZone, targetZone, error) {
  45. return parentZoneDelegate.handleError(targetZone, error);
  46. }
  47. onScheduleTask(parentZoneDelegate, currentZone, targetZone, task) {
  48. const key = task.type + ':' + task.source;
  49. let instance = WtfZoneSpec.scheduleInstance[key];
  50. if (!instance) {
  51. instance = WtfZoneSpec.scheduleInstance[key] =
  52. wtfEvents.createInstance(`Zone:schedule:${key}(ascii zone, any data)`);
  53. }
  54. const retValue = parentZoneDelegate.scheduleTask(targetZone, task);
  55. instance(zonePathName(targetZone), shallowObj(task.data, 2));
  56. return retValue;
  57. }
  58. onInvokeTask(parentZoneDelegate, currentZone, targetZone, task, applyThis, applyArgs) {
  59. const source = task.source;
  60. let scope = WtfZoneSpec.invokeTaskScope[source];
  61. if (!scope) {
  62. scope = WtfZoneSpec.invokeTaskScope[source] =
  63. wtfEvents.createScope(`Zone:invokeTask:${source}(ascii zone)`);
  64. }
  65. return wtfTrace.leaveScope(scope(zonePathName(targetZone)), parentZoneDelegate.invokeTask(targetZone, task, applyThis, applyArgs));
  66. }
  67. onCancelTask(parentZoneDelegate, currentZone, targetZone, task) {
  68. const key = task.source;
  69. let instance = WtfZoneSpec.cancelInstance[key];
  70. if (!instance) {
  71. instance = WtfZoneSpec.cancelInstance[key] =
  72. wtfEvents.createInstance(`Zone:cancel:${key}(ascii zone, any options)`);
  73. }
  74. const retValue = parentZoneDelegate.cancelTask(targetZone, task);
  75. instance(zonePathName(targetZone), shallowObj(task.data, 2));
  76. return retValue;
  77. }
  78. }
  79. WtfZoneSpec.forkInstance = wtfEnabled ? wtfEvents.createInstance('Zone:fork(ascii zone, ascii newZone)') : null;
  80. WtfZoneSpec.scheduleInstance = {};
  81. WtfZoneSpec.cancelInstance = {};
  82. WtfZoneSpec.invokeScope = {};
  83. WtfZoneSpec.invokeTaskScope = {};
  84. function shallowObj(obj, depth) {
  85. if (!obj || !depth)
  86. return null;
  87. const out = {};
  88. for (const key in obj) {
  89. if (obj.hasOwnProperty(key)) {
  90. // explicit : any due to https://github.com/microsoft/TypeScript/issues/33191
  91. let value = obj[key];
  92. switch (typeof value) {
  93. case 'object':
  94. const name = value && value.constructor && value.constructor.name;
  95. value = name == Object.name ? shallowObj(value, depth - 1) : name;
  96. break;
  97. case 'function':
  98. value = value.name || undefined;
  99. break;
  100. }
  101. out[key] = value;
  102. }
  103. }
  104. return out;
  105. }
  106. function zonePathName(zone) {
  107. let name = zone.name;
  108. let localZone = zone.parent;
  109. while (localZone != null) {
  110. name = localZone.name + '::' + name;
  111. localZone = localZone.parent;
  112. }
  113. return name;
  114. }
  115. Zone['wtfZoneSpec'] = !wtfEnabled ? null : new WtfZoneSpec();
  116. })(typeof window === 'object' && window || typeof self === 'object' && self || global);