Source: lib/ads/server_side_ad_manager.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @fileoverview
  8. * @suppress {missingRequire} TODO(b/152540451): this shouldn't be needed
  9. */
  10. goog.provide('shaka.ads.ServerSideAdManager');
  11. goog.require('goog.asserts');
  12. goog.require('shaka.ads.ServerSideAd');
  13. goog.require('shaka.log');
  14. goog.require('shaka.util.IReleasable');
  15. /**
  16. * A class responsible for server-side ad interactions.
  17. * @implements {shaka.util.IReleasable}
  18. */
  19. shaka.ads.ServerSideAdManager = class {
  20. /**
  21. * @param {HTMLElement} adContainer
  22. * @param {HTMLMediaElement} video
  23. * @param {string} locale
  24. * @param {function(!shaka.util.FakeEvent)} onEvent
  25. */
  26. constructor(adContainer, video, locale, onEvent) {
  27. /** @private {HTMLElement} */
  28. this.adContainer_ = adContainer;
  29. /** @private {HTMLMediaElement} */
  30. this.video_ = video;
  31. /** @private
  32. {?shaka.util.PublicPromise.<string>} */
  33. this.streamPromise_ = null;
  34. /** @private {number} */
  35. this.streamRequestStartTime_ = NaN;
  36. /** @private {function(!shaka.util.FakeEvent)} */
  37. this.onEvent_ = onEvent;
  38. /** @private {boolean} */
  39. this.isLiveContent_ = false;
  40. /**
  41. * Time to seek to after an ad if that ad was played as the result of
  42. * snapback.
  43. * @private {?number}
  44. */
  45. this.snapForwardTime_ = null;
  46. /** @private {shaka.ads.ServerSideAd} */
  47. this.ad_ = null;
  48. /** @private {?google.ima.dai.api.AdProgressData} */
  49. this.adProgressData_ = null;
  50. /** @private {string} */
  51. this.backupUrl_ = '';
  52. /** @private {!Array.<!shaka.extern.AdCuePoint>} */
  53. this.currentCuePoints_ = [];
  54. /** @private {shaka.util.EventManager} */
  55. this.eventManager_ = new shaka.util.EventManager();
  56. /** @private {google.ima.dai.api.UiSettings} */
  57. const uiSettings = new google.ima.dai.api.UiSettings();
  58. uiSettings.setLocale(locale);
  59. /** @private {google.ima.dai.api.StreamManager} */
  60. this.streamManager_ = new google.ima.dai.api.StreamManager(
  61. this.video_, this.adContainer_, uiSettings);
  62. this.onEvent_(new shaka.util.FakeEvent(
  63. shaka.ads.AdManager.IMA_STREAM_MANAGER_LOADED,
  64. (new Map()).set('imaStreamManager', this.streamManager_)));
  65. // Events
  66. this.eventManager_.listen(this.streamManager_,
  67. google.ima.dai.api.StreamEvent.Type.LOADED, (e) => {
  68. shaka.log.info('Ad SS Loaded');
  69. this.onLoaded_(
  70. /** @type {!google.ima.dai.api.StreamEvent} */ (e));
  71. });
  72. this.eventManager_.listen(this.streamManager_,
  73. google.ima.dai.api.StreamEvent.Type.ERROR, () => {
  74. shaka.log.info('Ad SS Error');
  75. this.onError_();
  76. });
  77. this.eventManager_.listen(this.streamManager_,
  78. google.ima.dai.api.StreamEvent.Type.AD_BREAK_STARTED, () => {
  79. shaka.log.info('Ad Break Started');
  80. });
  81. this.eventManager_.listen(this.streamManager_,
  82. google.ima.dai.api.StreamEvent.Type.STARTED, (e) => {
  83. shaka.log.info('Ad Started');
  84. this.onAdStart_(/** @type {!google.ima.dai.api.StreamEvent} */ (e));
  85. });
  86. this.eventManager_.listen(this.streamManager_,
  87. google.ima.dai.api.StreamEvent.Type.AD_BREAK_ENDED, () => {
  88. shaka.log.info('Ad Break Ended');
  89. this.onAdBreakEnded_();
  90. });
  91. this.eventManager_.listen(this.streamManager_,
  92. google.ima.dai.api.StreamEvent.Type.AD_PROGRESS, (e) => {
  93. this.onAdProgress_(
  94. /** @type {!google.ima.dai.api.StreamEvent} */ (e));
  95. });
  96. this.eventManager_.listen(this.streamManager_,
  97. google.ima.dai.api.StreamEvent.Type.FIRST_QUARTILE, () => {
  98. shaka.log.info('Ad event: First Quartile');
  99. this.onEvent_(
  100. new shaka.util.FakeEvent(shaka.ads.AdManager.AD_FIRST_QUARTILE));
  101. });
  102. this.eventManager_.listen(this.streamManager_,
  103. google.ima.dai.api.StreamEvent.Type.MIDPOINT, () => {
  104. shaka.log.info('Ad event: Midpoint');
  105. this.onEvent_(
  106. new shaka.util.FakeEvent(shaka.ads.AdManager.AD_MIDPOINT));
  107. });
  108. this.eventManager_.listen(this.streamManager_,
  109. google.ima.dai.api.StreamEvent.Type.THIRD_QUARTILE, () => {
  110. shaka.log.info('Ad event: Third Quartile');
  111. this.onEvent_(
  112. new shaka.util.FakeEvent(shaka.ads.AdManager.AD_THIRD_QUARTILE));
  113. });
  114. this.eventManager_.listen(this.streamManager_,
  115. google.ima.dai.api.StreamEvent.Type.COMPLETE, () => {
  116. shaka.log.info('Ad event: Complete');
  117. this.onEvent_(
  118. new shaka.util.FakeEvent(shaka.ads.AdManager.AD_COMPLETE));
  119. this.onEvent_(
  120. new shaka.util.FakeEvent(shaka.ads.AdManager.AD_STOPPED));
  121. this.adContainer_.removeAttribute('ad-active');
  122. this.ad_ = null;
  123. });
  124. this.eventManager_.listen(this.streamManager_,
  125. google.ima.dai.api.StreamEvent.Type.SKIPPED, () => {
  126. shaka.log.info('Ad event: Skipped');
  127. this.onEvent_(
  128. new shaka.util.FakeEvent(shaka.ads.AdManager.AD_SKIPPED));
  129. this.onEvent_(
  130. new shaka.util.FakeEvent(shaka.ads.AdManager.AD_STOPPED));
  131. });
  132. this.eventManager_.listen(this.streamManager_,
  133. google.ima.dai.api.StreamEvent.Type.CUEPOINTS_CHANGED, (e) => {
  134. shaka.log.info('Ad event: Cue points changed');
  135. this.onCuePointsChanged_(
  136. /** @type {!google.ima.dai.api.StreamEvent} */ (e));
  137. });
  138. }
  139. /**
  140. * @param {!google.ima.dai.api.StreamRequest} streamRequest
  141. * @param {string=} backupUrl
  142. * @return {!Promise.<string>}
  143. */
  144. streamRequest(streamRequest, backupUrl) {
  145. if (this.streamPromise_) {
  146. return Promise.reject(new shaka.util.Error(
  147. shaka.util.Error.Severity.RECOVERABLE,
  148. shaka.util.Error.Category.ADS,
  149. shaka.util.Error.Code.CURRENT_DAI_REQUEST_NOT_FINISHED));
  150. }
  151. if (streamRequest instanceof google.ima.dai.api.LiveStreamRequest) {
  152. this.isLiveContent_ = true;
  153. }
  154. this.streamPromise_ = new shaka.util.PublicPromise();
  155. this.streamManager_.requestStream(streamRequest);
  156. this.backupUrl_ = backupUrl || '';
  157. this.streamRequestStartTime_ = Date.now() / 1000;
  158. return this.streamPromise_;
  159. }
  160. /**
  161. * @param {Object} adTagParameters
  162. */
  163. replaceAdTagParameters(adTagParameters) {
  164. this.streamManager_.replaceAdTagParameters(adTagParameters);
  165. }
  166. /**
  167. * Resets the stream manager and removes any continuous polling.
  168. */
  169. stop() {
  170. // TODO:
  171. // For SS DAI streams, if a different asset gets unloaded as
  172. // part of the process
  173. // of loading a DAI asset, stream manager state gets reset and we
  174. // don't get any ad events.
  175. // We need to figure out if it makes sense to stop the SS
  176. // manager on unload, and, if it does, find
  177. // a way to do it safely.
  178. // this.streamManager_.reset();
  179. this.backupUrl_ = '';
  180. this.snapForwardTime_ = null;
  181. this.currentCuePoints_ = [];
  182. }
  183. /** @override */
  184. release() {
  185. this.stop();
  186. if (this.eventManager_) {
  187. this.eventManager_.release();
  188. }
  189. }
  190. /**
  191. * @param {string} type
  192. * @param {Uint8Array|string} data
  193. * Comes as string in DASH and as Uint8Array in HLS.
  194. * @param {number} timestamp (in seconds)
  195. */
  196. onTimedMetadata(type, data, timestamp) {
  197. this.streamManager_.processMetadata(type, data, timestamp);
  198. }
  199. /**
  200. * @param {shaka.extern.ID3Metadata} value
  201. */
  202. onCueMetadataChange(value) {
  203. // Native HLS over Safari/iOS/iPadOS
  204. // For live event streams, the stream needs some way of informing the SDK
  205. // that an ad break is coming up or ending. In the IMA DAI SDK, this is
  206. // done through timed metadata. Timed metadata is carried as part of the
  207. // DAI stream content and carries ad break timing information used by the
  208. // SDK to track ad breaks.
  209. if (value['key'] && value['data']) {
  210. const metadata = {};
  211. metadata[value['key']] = value['data'];
  212. this.streamManager_.onTimedMetadata(metadata);
  213. }
  214. }
  215. /**
  216. * @return {!Array.<!shaka.extern.AdCuePoint>}
  217. */
  218. getCuePoints() {
  219. return this.currentCuePoints_;
  220. }
  221. /**
  222. * If a seek jumped over the ad break, return to the start of the
  223. * ad break, then complete the seek after the ad played through.
  224. * @private
  225. */
  226. checkForSnapback_() {
  227. const currentTime = this.video_.currentTime;
  228. if (currentTime == 0) {
  229. return;
  230. }
  231. this.streamManager_.streamTimeForContentTime(currentTime);
  232. const previousCuePoint =
  233. this.streamManager_.previousCuePointForStreamTime(currentTime);
  234. // The cue point gets marked as 'played' as soon as the playhead hits it
  235. // (at the start of an ad), so when we come back to this method as a result
  236. // of seeking back to the user-selected time, the 'played' flag will be set.
  237. if (previousCuePoint && !previousCuePoint.played) {
  238. shaka.log.info('Seeking back to the start of the ad break at ' +
  239. previousCuePoint.start + ' and will return to ' + currentTime);
  240. this.snapForwardTime_ = currentTime;
  241. this.video_.currentTime = previousCuePoint.start;
  242. }
  243. }
  244. /**
  245. * @param {!google.ima.dai.api.StreamEvent} e
  246. * @private
  247. */
  248. onAdStart_(e) {
  249. goog.asserts.assert(this.streamManager_,
  250. 'Should have a stream manager at this point!');
  251. const imaAd = e.getAd();
  252. this.ad_ = new shaka.ads.ServerSideAd(imaAd, this.video_);
  253. // Ad object and ad progress data come from two different IMA events.
  254. // It's a race, and we don't know, which one will fire first - the
  255. // event that contains an ad object (AD_STARTED) or the one that
  256. // contains ad progress info (AD_PROGRESS).
  257. // If the progress event fired first, we must've saved the progress
  258. // info and can now add it to the ad object.
  259. if (this.adProgressData_) {
  260. this.ad_.setProgressData(this.adProgressData_);
  261. }
  262. this.onEvent_(new shaka.util.FakeEvent(shaka.ads.AdManager.AD_STARTED,
  263. (new Map()).set('ad', this.ad_)));
  264. this.adContainer_.setAttribute('ad-active', 'true');
  265. }
  266. /**
  267. * @private
  268. */
  269. onAdBreakEnded_() {
  270. this.adContainer_.removeAttribute('ad-active');
  271. const currentTime = this.video_.currentTime;
  272. // If the ad break was a result of snapping back (a user seeked over
  273. // an ad break and was returned to it), seek forward to the point,
  274. // originally chosen by the user.
  275. if (this.snapForwardTime_ && this.snapForwardTime_ > currentTime) {
  276. this.video_.currentTime = this.snapForwardTime_;
  277. this.snapForwardTime_ = null;
  278. }
  279. }
  280. /**
  281. * @param {!google.ima.dai.api.StreamEvent} e
  282. * @private
  283. */
  284. onLoaded_(e) {
  285. const now = Date.now() / 1000;
  286. const loadTime = now - this.streamRequestStartTime_;
  287. this.onEvent_(new shaka.util.FakeEvent(shaka.ads.AdManager.ADS_LOADED,
  288. (new Map()).set('loadTime', loadTime)));
  289. const streamData = e.getStreamData();
  290. const url = streamData.url;
  291. this.streamPromise_.resolve(url);
  292. this.streamPromise_ = null;
  293. if (!this.isLiveContent_) {
  294. this.eventManager_.listen(this.video_, 'seeked', () => {
  295. this.checkForSnapback_();
  296. });
  297. }
  298. }
  299. /**
  300. * @private
  301. */
  302. onError_() {
  303. if (!this.backupUrl_.length) {
  304. this.streamPromise_.reject('IMA Stream request returned an error ' +
  305. 'and there was no backup asset uri provided.');
  306. this.streamPromise_ = null;
  307. return;
  308. }
  309. shaka.log.warning('IMA stream request returned an error. ' +
  310. 'Falling back to the backup asset uri.');
  311. this.streamPromise_.resolve(this.backupUrl_);
  312. this.streamPromise_ = null;
  313. }
  314. /**
  315. * @param {!google.ima.dai.api.StreamEvent} e
  316. * @private
  317. */
  318. onAdProgress_(e) {
  319. const streamData = e.getStreamData();
  320. const adProgressData = streamData.adProgressData;
  321. this.adProgressData_ = adProgressData;
  322. if (this.ad_) {
  323. this.ad_.setProgressData(this.adProgressData_);
  324. }
  325. }
  326. /**
  327. * @param {!google.ima.dai.api.StreamEvent} e
  328. * @private
  329. */
  330. onCuePointsChanged_(e) {
  331. const streamData = e.getStreamData();
  332. /** @type {!Array.<!shaka.extern.AdCuePoint>} */
  333. const cuePoints = [];
  334. for (const point of streamData.cuepoints) {
  335. /** @type {shaka.extern.AdCuePoint} */
  336. const shakaCuePoint = {
  337. start: point.start,
  338. end: point.end,
  339. };
  340. cuePoints.push(shakaCuePoint);
  341. }
  342. this.currentCuePoints_ = cuePoints;
  343. this.onEvent_(new shaka.util.FakeEvent(
  344. shaka.ads.AdManager.CUEPOINTS_CHANGED,
  345. (new Map()).set('cuepoints', cuePoints)));
  346. }
  347. };