diff --git a/app/src/main/java/com/github/libretube/util/DashHelper.kt b/app/src/main/java/com/github/libretube/util/DashHelper.kt index 33684d742..e543ff6ef 100644 --- a/app/src/main/java/com/github/libretube/util/DashHelper.kt +++ b/app/src/main/java/com/github/libretube/util/DashHelper.kt @@ -13,164 +13,162 @@ import javax.xml.transform.stream.StreamResult // Based off of https://github.com/TeamPiped/Piped/blob/master/src/utils/DashUtils.js -class DashHelper { - companion object { +object DashHelper { - private val builderFactory: DocumentBuilderFactory = DocumentBuilderFactory.newInstance() - private val transformerFactory: TransformerFactory = TransformerFactory.newInstance() + private val builderFactory: DocumentBuilderFactory = DocumentBuilderFactory.newInstance() + private val transformerFactory: TransformerFactory = TransformerFactory.newInstance() - private data class AdapSetInfo( - val mimeType: String, - val audioTrackId: String? = null, - val formats: MutableList = mutableListOf() - ) + private data class AdapSetInfo( + val mimeType: String, + val audioTrackId: String? = null, + val formats: MutableList = mutableListOf() + ) - fun createManifest(streams: Streams): String { - val builder: DocumentBuilder = builderFactory.newDocumentBuilder() + fun createManifest(streams: Streams): String { + val builder: DocumentBuilder = builderFactory.newDocumentBuilder() - val doc = builder.newDocument() - val mpd = doc.createElement("MPD") - mpd.setAttribute("xmlns", "urn:mpeg:dash:schema:mpd:2011") - mpd.setAttribute("profiles", "urn:mpeg:dash:profile:full:2011") - mpd.setAttribute("minBufferTime", "PT1.5S") - mpd.setAttribute("type", "static") - mpd.setAttribute("mediaPresentationDuration", "PT${streams.duration}S") + val doc = builder.newDocument() + val mpd = doc.createElement("MPD") + mpd.setAttribute("xmlns", "urn:mpeg:dash:schema:mpd:2011") + mpd.setAttribute("profiles", "urn:mpeg:dash:profile:full:2011") + mpd.setAttribute("minBufferTime", "PT1.5S") + mpd.setAttribute("type", "static") + mpd.setAttribute("mediaPresentationDuration", "PT${streams.duration}S") - val period = doc.createElement("Period") + val period = doc.createElement("Period") - val adapSetInfos = ArrayList() + val adapSetInfos = ArrayList() - for (stream in streams.videoStreams!!) { - // ignore dual format streams - if (!stream.videoOnly!!) { - continue - } + for (stream in streams.videoStreams!!) { + // ignore dual format streams + if (!stream.videoOnly!!) { + continue + } - val adapSetInfo = adapSetInfos.find { it.mimeType == stream.mimeType } - if (adapSetInfo != null) { - adapSetInfo.formats.add(stream) - continue - } - adapSetInfos.add( - AdapSetInfo( - stream.mimeType!!, - null, - mutableListOf(stream) - ) + val adapSetInfo = adapSetInfos.find { it.mimeType == stream.mimeType } + if (adapSetInfo != null) { + adapSetInfo.formats.add(stream) + continue + } + adapSetInfos.add( + AdapSetInfo( + stream.mimeType!!, + null, + mutableListOf(stream) ) - } - - for (stream in streams.audioStreams!!) { - val adapSetInfo = - adapSetInfos.find { it.mimeType == stream.mimeType && it.audioTrackId == stream.audioTrackId } - if (adapSetInfo != null) { - adapSetInfo.formats.add(stream) - continue - } - adapSetInfos.add( - AdapSetInfo( - stream.mimeType!!, - null, - mutableListOf(stream) - ) - ) - } - - for (adapSet in adapSetInfos) { - val adapSetElement = doc.createElement("AdaptationSet") - adapSetElement.setAttribute("mimeType", adapSet.mimeType) - adapSetElement.setAttribute("startWithSAP", "1") - adapSetElement.setAttribute("subsegmentAlignment", "true") - if (adapSet.audioTrackId != null) { - adapSetElement.setAttribute("lang", adapSet.audioTrackId.substring(0, 2)) - } - - val isVideo = adapSet.mimeType.contains("video") - - if (isVideo) { - adapSetElement.setAttribute("scanType", "progressive") - } - - for (stream in adapSet.formats) { - val rep = let { - if (isVideo) { - createVideoRepresentation(doc, stream) - } else { - createAudioRepresentation(doc, stream) - } - } - adapSetElement.appendChild(rep) - } - - period.appendChild(adapSetElement) - } - - mpd.appendChild(period) - - doc.appendChild(mpd) - - val domSource = DOMSource(doc) - val writer = StringWriter() - - val transformer = transformerFactory.newTransformer() - transformer.transform(domSource, StreamResult(writer)) - - return writer.toString() - } - - private fun createAudioRepresentation(doc: Document, stream: PipedStream): Element { - val representation = doc.createElement("Representation") - representation.setAttribute("bandwidth", stream.bitrate.toString()) - representation.setAttribute("codecs", stream.codec!!) - representation.setAttribute("mimeType", stream.mimeType!!) - - val audioChannelConfiguration = doc.createElement("AudioChannelConfiguration") - audioChannelConfiguration.setAttribute( - "schemeIdUri", - "urn:mpeg:dash:23003:3:audio_channel_configuration:2011" ) - audioChannelConfiguration.setAttribute("value", "2") - - val baseUrl = doc.createElement("BaseURL") - baseUrl.appendChild(doc.createTextNode(stream.url!!)) - - val segmentBase = doc.createElement("SegmentBase") - segmentBase.setAttribute("indexRange", "${stream.indexStart}-${stream.indexEnd}") - - val initialization = doc.createElement("Initialization") - initialization.setAttribute("range", "${stream.initStart}-${stream.initEnd}") - segmentBase.appendChild(initialization) - - representation.appendChild(audioChannelConfiguration) - representation.appendChild(baseUrl) - representation.appendChild(segmentBase) - - return representation } - private fun createVideoRepresentation(doc: Document, stream: PipedStream): Element { - val representation = doc.createElement("Representation") - representation.setAttribute("codecs", stream.codec!!) - representation.setAttribute("bandwidth", stream.bitrate.toString()) - representation.setAttribute("width", stream.width.toString()) - representation.setAttribute("height", stream.height.toString()) - representation.setAttribute("maxPlayoutRate", "1") - representation.setAttribute("frameRate", stream.fps.toString()) - - val baseUrl = doc.createElement("BaseURL") - baseUrl.appendChild(doc.createTextNode(stream.url!!)) - - val segmentBase = doc.createElement("SegmentBase") - segmentBase.setAttribute("indexRange", "${stream.indexStart}-${stream.indexEnd}") - - val initialization = doc.createElement("Initialization") - initialization.setAttribute("range", "${stream.initStart}-${stream.initEnd}") - segmentBase.appendChild(initialization) - - representation.appendChild(baseUrl) - representation.appendChild(segmentBase) - - return representation + for (stream in streams.audioStreams!!) { + val adapSetInfo = + adapSetInfos.find { it.mimeType == stream.mimeType && it.audioTrackId == stream.audioTrackId } + if (adapSetInfo != null) { + adapSetInfo.formats.add(stream) + continue + } + adapSetInfos.add( + AdapSetInfo( + stream.mimeType!!, + null, + mutableListOf(stream) + ) + ) } + + for (adapSet in adapSetInfos) { + val adapSetElement = doc.createElement("AdaptationSet") + adapSetElement.setAttribute("mimeType", adapSet.mimeType) + adapSetElement.setAttribute("startWithSAP", "1") + adapSetElement.setAttribute("subsegmentAlignment", "true") + if (adapSet.audioTrackId != null) { + adapSetElement.setAttribute("lang", adapSet.audioTrackId.substring(0, 2)) + } + + val isVideo = adapSet.mimeType.contains("video") + + if (isVideo) { + adapSetElement.setAttribute("scanType", "progressive") + } + + for (stream in adapSet.formats) { + val rep = let { + if (isVideo) { + createVideoRepresentation(doc, stream) + } else { + createAudioRepresentation(doc, stream) + } + } + adapSetElement.appendChild(rep) + } + + period.appendChild(adapSetElement) + } + + mpd.appendChild(period) + + doc.appendChild(mpd) + + val domSource = DOMSource(doc) + val writer = StringWriter() + + val transformer = transformerFactory.newTransformer() + transformer.transform(domSource, StreamResult(writer)) + + return writer.toString() + } + + private fun createAudioRepresentation(doc: Document, stream: PipedStream): Element { + val representation = doc.createElement("Representation") + representation.setAttribute("bandwidth", stream.bitrate.toString()) + representation.setAttribute("codecs", stream.codec!!) + representation.setAttribute("mimeType", stream.mimeType!!) + + val audioChannelConfiguration = doc.createElement("AudioChannelConfiguration") + audioChannelConfiguration.setAttribute( + "schemeIdUri", + "urn:mpeg:dash:23003:3:audio_channel_configuration:2011" + ) + audioChannelConfiguration.setAttribute("value", "2") + + val baseUrl = doc.createElement("BaseURL") + baseUrl.appendChild(doc.createTextNode(stream.url!!)) + + val segmentBase = doc.createElement("SegmentBase") + segmentBase.setAttribute("indexRange", "${stream.indexStart}-${stream.indexEnd}") + + val initialization = doc.createElement("Initialization") + initialization.setAttribute("range", "${stream.initStart}-${stream.initEnd}") + segmentBase.appendChild(initialization) + + representation.appendChild(audioChannelConfiguration) + representation.appendChild(baseUrl) + representation.appendChild(segmentBase) + + return representation + } + + private fun createVideoRepresentation(doc: Document, stream: PipedStream): Element { + val representation = doc.createElement("Representation") + representation.setAttribute("codecs", stream.codec!!) + representation.setAttribute("bandwidth", stream.bitrate.toString()) + representation.setAttribute("width", stream.width.toString()) + representation.setAttribute("height", stream.height.toString()) + representation.setAttribute("maxPlayoutRate", "1") + representation.setAttribute("frameRate", stream.fps.toString()) + + val baseUrl = doc.createElement("BaseURL") + baseUrl.appendChild(doc.createTextNode(stream.url!!)) + + val segmentBase = doc.createElement("SegmentBase") + segmentBase.setAttribute("indexRange", "${stream.indexStart}-${stream.indexEnd}") + + val initialization = doc.createElement("Initialization") + initialization.setAttribute("range", "${stream.initStart}-${stream.initEnd}") + segmentBase.appendChild(initialization) + + representation.appendChild(baseUrl) + representation.appendChild(segmentBase) + + return representation } }