Source: lib/util/state_history.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.util.StateHistory');
  7. goog.require('goog.asserts');
  8. goog.require('shaka.log');
  9. /**
  10. * This class is used to track the time spent in arbitrary states. When told of
  11. * a state, it will assume that state was active until a new state is provided.
  12. * When provided with identical states back-to-back, the existing entry will be
  13. * updated.
  14. *
  15. * @final
  16. */
  17. shaka.util.StateHistory = class {
  18. /** */
  19. constructor() {
  20. /**
  21. * The state that we think is still the current change. It is "open" for
  22. * updating.
  23. *
  24. * @private {?shaka.extern.StateChange}
  25. */
  26. this.open_ = null;
  27. /**
  28. * The stats that are "closed" for updating. The "open" state becomes closed
  29. * once we move to a new state.
  30. *
  31. * @private {!Array.<shaka.extern.StateChange>}
  32. */
  33. this.closed_ = [];
  34. }
  35. /**
  36. * @param {string} state
  37. * @return {boolean} True if this changed the state
  38. */
  39. update(state) {
  40. // |open_| will only be |null| when we first call |update|.
  41. if (this.open_ == null) {
  42. this.start_(state);
  43. return true;
  44. } else {
  45. return this.update_(state);
  46. }
  47. }
  48. /**
  49. * Go through all entries in the history and count how much time was spend in
  50. * the given state.
  51. *
  52. * @param {string} state
  53. * @return {number}
  54. */
  55. getTimeSpentIn(state) {
  56. let sum = 0;
  57. if (this.open_ && this.open_.state == state) {
  58. sum += this.open_.duration;
  59. }
  60. for (const entry of this.closed_) {
  61. sum += entry.state == state ? entry.duration : 0;
  62. }
  63. return sum;
  64. }
  65. /**
  66. * Get a copy of each state change entry in the history. A copy of each entry
  67. * is created to break the reference to the internal data.
  68. *
  69. * @return {!Array.<shaka.extern.StateChange>}
  70. */
  71. getCopy() {
  72. const clone = (entry) => {
  73. return {
  74. timestamp: entry.timestamp,
  75. state: entry.state,
  76. duration: entry.duration,
  77. };
  78. };
  79. const copy = [];
  80. for (const entry of this.closed_) {
  81. copy.push(clone(entry));
  82. }
  83. if (this.open_) {
  84. copy.push(clone(this.open_));
  85. }
  86. return copy;
  87. }
  88. /**
  89. * @param {string} state
  90. * @private
  91. */
  92. start_(state) {
  93. goog.asserts.assert(
  94. this.open_ == null,
  95. 'There must be no open entry in order when we start');
  96. shaka.log.v1('Changing Player state to', state);
  97. this.open_ = {
  98. timestamp: this.getNowInSeconds_(),
  99. state: state,
  100. duration: 0,
  101. };
  102. }
  103. /**
  104. * @param {string} state
  105. * @return {boolean} True if this changed the state
  106. * @private
  107. */
  108. update_(state) {
  109. goog.asserts.assert(
  110. this.open_,
  111. 'There must be an open entry in order to update it');
  112. const currentTimeSeconds = this.getNowInSeconds_();
  113. // Always update the duration so that it can always be as accurate as
  114. // possible.
  115. this.open_.duration = currentTimeSeconds - this.open_.timestamp;
  116. // If the state has not changed, there is no need to add a new entry.
  117. if (this.open_.state == state) {
  118. return false;
  119. }
  120. // We have changed states, so "close" the open state.
  121. shaka.log.v1('Changing Player state to', state);
  122. this.closed_.push(this.open_);
  123. this.open_ = {
  124. timestamp: currentTimeSeconds,
  125. state: state,
  126. duration: 0,
  127. };
  128. return true;
  129. }
  130. /**
  131. * Get the system time in seconds.
  132. *
  133. * @return {number}
  134. * @private
  135. */
  136. getNowInSeconds_() {
  137. return Date.now() / 1000;
  138. }
  139. };