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