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