Source: lib/polyfill/abort_controller.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.polyfill.AbortController');
  7. goog.require('shaka.polyfill');
  8. goog.require('shaka.util.FakeEvent');
  9. goog.require('shaka.util.FakeEventTarget');
  10. /**
  11. * @summary A polyfill for systems that do not implement AbortController.
  12. * This is used both with the fetch API for HTTP requests and inside the HLS
  13. * parser.
  14. * @export
  15. * @extends AbortController
  16. */
  17. shaka.polyfill.AbortController = class {
  18. /**
  19. * Install the polyfill if needed.
  20. * @export
  21. */
  22. static install() {
  23. if (window.AbortController) {
  24. // Not needed.
  25. return;
  26. }
  27. window.AbortController = shaka.polyfill.AbortController;
  28. window.AbortSignal = shaka.polyfill.AbortController.AbortSignal;
  29. }
  30. /** */
  31. constructor() {
  32. /** @private {!shaka.polyfill.AbortController.AbortSignal} */
  33. this.signal_ = new shaka.polyfill.AbortController.AbortSignal();
  34. }
  35. /**
  36. * @override
  37. * @suppress {const|duplicate} Since the original is defined as "const", we
  38. * need this suppression to override it.
  39. */
  40. get signal() {
  41. return this.signal_;
  42. }
  43. /**
  44. * @param {*=} reason
  45. * @override
  46. */
  47. abort(reason) {
  48. this.signal_.doAbort_(reason);
  49. }
  50. };
  51. /**
  52. * @summary A polyfill for AbortSignal, part of the AbortController API.
  53. * @implements {AbortSignal}
  54. */
  55. shaka.polyfill.AbortController.AbortSignal =
  56. class extends shaka.util.FakeEventTarget {
  57. /** */
  58. constructor() {
  59. super();
  60. /** @private {boolean} */
  61. this.aborted_ = false;
  62. /** @private {*} */
  63. this.reason_ = undefined;
  64. /** @type {?function(!Event)} */
  65. this.onabort = null;
  66. }
  67. /** @override */
  68. get aborted() {
  69. return this.aborted_;
  70. }
  71. /** @return {*} */
  72. get reason() {
  73. return this.reason_;
  74. }
  75. /** @return {void} */
  76. throwIfAborted() {
  77. if (this.aborted_) {
  78. throw this.reason_;
  79. }
  80. }
  81. /**
  82. * @param {*} reason
  83. * @private
  84. */
  85. doAbort_(reason) {
  86. if (this.aborted_) {
  87. return;
  88. }
  89. this.aborted_ = true;
  90. this.reason_ = reason;
  91. if (this.reason_ === undefined) {
  92. // This is equivalent to a native implementation.
  93. this.reason_ = new DOMException(
  94. 'signal is aborted without reason', 'AbortError');
  95. }
  96. // According to MDN:
  97. // "Event type - A generic Event with no added properties."
  98. const event = new shaka.util.FakeEvent('abort');
  99. if (this.onabort) {
  100. this.onabort(event);
  101. }
  102. this.dispatchEvent(event);
  103. }
  104. /**
  105. * @param {*=} reason
  106. * @return {!AbortSignal}
  107. */
  108. static abort(reason) {
  109. const signal = new shaka.polyfill.AbortController.AbortSignal();
  110. signal.doAbort_(reason);
  111. return signal;
  112. }
  113. /**
  114. * @param {number} timeMs
  115. * @return {!AbortSignal}
  116. */
  117. static timeout(timeMs) {
  118. const signal = new shaka.polyfill.AbortController.AbortSignal();
  119. window.setTimeout(() => {
  120. // This is equivalent to a native implementation.
  121. signal.doAbort_(new DOMException('signal timed out', 'TimeoutError'));
  122. }, timeMs);
  123. return signal;
  124. }
  125. };
  126. shaka.polyfill.register(shaka.polyfill.AbortController.install);