Source: lib/offline/indexeddb/v2_storage_cell.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.offline.indexeddb.V2StorageCell');
  7. goog.require('shaka.offline.indexeddb.BaseStorageCell');
  8. goog.require('shaka.util.PeriodCombiner');
  9. /**
  10. * The V2StorageCell is for all stores that follow the shaka.externs V2 and V3
  11. * offline types. V2 was introduced in Shaka Player v2.3.0 and quickly
  12. * replaced with V3 in Shaka Player v2.3.2. V3 was then deprecated in v3.0.
  13. *
  14. * Upgrading from V1 to V2 initially broke the database in a way that prevented
  15. * adding new records. The problem was with the upgrade process, not with the
  16. * database format. Once database upgrades were removed, we increased the
  17. * database version to V3 and marked V2 as read-only. Therefore, V2 and V3
  18. * databases can both be read by this cell.
  19. *
  20. * The manifest and segment stores didn't change in database V4, but a separate
  21. * table for session IDs was added. So this cell also covers DB V4.
  22. *
  23. * @implements {shaka.extern.StorageCell}
  24. */
  25. shaka.offline.indexeddb.V2StorageCell = class
  26. extends shaka.offline.indexeddb.BaseStorageCell {
  27. /**
  28. * @override
  29. * @param {shaka.extern.ManifestDBV2} old
  30. * @return {!Promise.<shaka.extern.ManifestDB>}
  31. */
  32. async convertManifest(old) {
  33. const streamsPerPeriod = [];
  34. for (let i = 0; i < old.periods.length; ++i) {
  35. // The last period ends at the end of the presentation.
  36. const periodEnd = i == old.periods.length - 1 ?
  37. old.duration : old.periods[i + 1].startTime;
  38. const duration = periodEnd - old.periods[i].startTime;
  39. const streams = this.convertPeriod_(old.periods[i], duration);
  40. streamsPerPeriod.push(streams);
  41. }
  42. const streams = await shaka.util.PeriodCombiner.combineDbStreams(
  43. streamsPerPeriod);
  44. return {
  45. appMetadata: old.appMetadata,
  46. creationTime: 0,
  47. drmInfo: old.drmInfo,
  48. duration: old.duration,
  49. // JSON serialization turns Infinity into null, so turn it back now.
  50. expiration: old.expiration == null ? Infinity : old.expiration,
  51. originalManifestUri: old.originalManifestUri,
  52. sessionIds: old.sessionIds,
  53. size: old.size,
  54. streams,
  55. sequenceMode: false,
  56. };
  57. }
  58. /**
  59. * @param {shaka.extern.PeriodDBV2} period
  60. * @param {number} periodDuration
  61. * @return {!Array.<shaka.extern.StreamDB>}
  62. * @private
  63. */
  64. convertPeriod_(period, periodDuration) {
  65. const streams = [];
  66. for (const stream of period.streams) {
  67. // The v4 version of the database as written by v2.5.0 - v2.5.9 might have
  68. // been corrupted slightly. A bug caused the stream metadata from all
  69. // periods to be written to each period. This was corrected in v2.5.10.
  70. // To fix this, we can identify the extra streams by their lack of
  71. // variantIds and skip them.
  72. if (stream.variantIds.length == 0) {
  73. continue;
  74. }
  75. streams.push(this.convertStream_(
  76. stream, period.startTime, period.startTime + periodDuration));
  77. }
  78. return streams;
  79. }
  80. /**
  81. * @param {shaka.extern.StreamDBV2} old
  82. * @param {number} periodStart
  83. * @param {number} periodEnd
  84. * @return {shaka.extern.StreamDB}
  85. * @private
  86. */
  87. convertStream_(old, periodStart, periodEnd) {
  88. return {
  89. id: old.id,
  90. originalId: old.originalId,
  91. groupId: null,
  92. primary: old.primary,
  93. type: old.contentType,
  94. mimeType: old.mimeType,
  95. codecs: old.codecs,
  96. frameRate: old.frameRate,
  97. pixelAspectRatio: old.pixelAspectRatio,
  98. hdr: undefined,
  99. colorGamut: undefined,
  100. videoLayout: undefined,
  101. kind: old.kind,
  102. language: old.language,
  103. originalLanguage: old.language || null,
  104. label: old.label,
  105. width: old.width,
  106. height: old.height,
  107. encrypted: old.encrypted,
  108. keyIds: new Set([old.keyId]),
  109. segments: old.segments.map((segment) =>
  110. this.convertSegment_(
  111. segment, old.initSegmentKey, periodStart, periodEnd,
  112. old.presentationTimeOffset)),
  113. variantIds: old.variantIds,
  114. roles: [],
  115. forced: false,
  116. audioSamplingRate: null,
  117. channelsCount: null,
  118. spatialAudio: false,
  119. closedCaptions: null,
  120. tilesLayout: undefined,
  121. external: false,
  122. fastSwitching: false,
  123. isAudioMuxedInVideo: false,
  124. };
  125. }
  126. /**
  127. * @param {shaka.extern.SegmentDBV2} old
  128. * @param {?number} initSegmentKey
  129. * @param {number} periodStart
  130. * @param {number} periodEnd
  131. * @param {number} presentationTimeOffset
  132. * @return {shaka.extern.SegmentDB}
  133. * @private
  134. */
  135. convertSegment_(
  136. old, initSegmentKey, periodStart, periodEnd, presentationTimeOffset) {
  137. const timestampOffset = periodStart - presentationTimeOffset;
  138. return {
  139. startTime: periodStart + old.startTime,
  140. endTime: periodStart + old.endTime,
  141. initSegmentKey,
  142. appendWindowStart: periodStart,
  143. appendWindowEnd: periodEnd,
  144. timestampOffset,
  145. dataKey: old.dataKey,
  146. tilesLayout: '',
  147. mimeType: null,
  148. codecs: null,
  149. thumbnailSprite: null,
  150. };
  151. }
  152. };