mirror of
https://github.com/yattee/yattee.git
synced 2025-01-06 21:07:12 +00:00
30cdaf88e1
We don't test every single stream anymore. If we get an 200, 206 or 403 we immediately stop testing streams. Unknown streams are also skipped. This speeds up starting playback, since we don'T have to wait for the network anymore.
228 lines
10 KiB
Swift
228 lines
10 KiB
Swift
import AVFoundation
|
|
import Foundation
|
|
import Siesta
|
|
import SwiftUI
|
|
|
|
extension PlayerModel {
|
|
var isLoadingAvailableStreams: Bool {
|
|
streamSelection.isNil || availableStreams.isEmpty
|
|
}
|
|
|
|
var isLoadingStream: Bool {
|
|
!stream.isNil && stream != streamSelection
|
|
}
|
|
|
|
var availableStreamsSorted: [Stream] {
|
|
availableStreams.sorted(by: streamsSorter)
|
|
}
|
|
|
|
func loadAvailableStreams(_ video: Video, onCompletion: @escaping (ResponseInfo) -> Void = { _ in }) {
|
|
captions = nil
|
|
availableStreams = []
|
|
|
|
guard let playerInstance else { return }
|
|
|
|
guard let api = playerAPI(video) else { return }
|
|
logger.info("loading streams from \(playerInstance.description)")
|
|
fetchStreams(api.video(video.videoID), instance: playerInstance, video: video, onCompletion: onCompletion)
|
|
}
|
|
|
|
private func fetchStreams(
|
|
_ resource: Resource,
|
|
instance: Instance,
|
|
video: Video,
|
|
onCompletion: @escaping (ResponseInfo) -> Void = { _ in }
|
|
) {
|
|
resource
|
|
.load()
|
|
.onSuccess { response in
|
|
if let video: Video = response.typedContent() {
|
|
VideosCacheModel.shared.storeVideo(video)
|
|
guard video.videoID == self.currentVideo?.videoID else {
|
|
self.logger.info("ignoring loaded streams from \(instance.description) as current video has changed")
|
|
return
|
|
}
|
|
self.streamsWithInstance(instance: instance, streams: video.streams) { processedStreams in
|
|
self.availableStreams = processedStreams
|
|
}
|
|
} else {
|
|
self.logger.critical("no streams available from \(instance.description)")
|
|
}
|
|
}
|
|
.onCompletion(onCompletion)
|
|
.onFailure { [weak self] responseError in
|
|
self?.navigation.presentAlert(title: "Could not load streams", message: responseError.userMessage)
|
|
self?.videoBeingOpened = nil
|
|
}
|
|
}
|
|
|
|
func streamsWithInstance(instance: Instance, streams: [Stream], completion: @escaping ([Stream]) -> Void) {
|
|
// Queue for stream processing
|
|
let streamProcessingQueue = DispatchQueue(label: "stream.yattee.streamProcessing.Queue")
|
|
// Queue for accessing the processedStreams array
|
|
let processedStreamsQueue = DispatchQueue(label: "stream.yattee.processedStreams.Queue")
|
|
// DispatchGroup for managing multiple tasks
|
|
let streamProcessingGroup = DispatchGroup()
|
|
|
|
var processedStreams = [Stream]()
|
|
let instance = instance
|
|
|
|
var hasForbiddenAsset = false
|
|
var hasAllowedAsset = false
|
|
|
|
for stream in streams {
|
|
streamProcessingQueue.async(group: streamProcessingGroup) {
|
|
let forbiddenAssetTestGroup = DispatchGroup()
|
|
if !hasAllowedAsset, !hasForbiddenAsset, !instance.proxiesVideos, stream.format != Stream.Format.unknown {
|
|
let (nonHLSAssets, hlsURLs) = self.getAssets(from: [stream])
|
|
if let firstStream = nonHLSAssets.first {
|
|
let asset = firstStream.0
|
|
let url = firstStream.1
|
|
let requestRange = firstStream.2
|
|
|
|
if instance.app == .invidious {
|
|
self.testAsset(url: url, range: requestRange, isHLS: false, forbiddenAssetTestGroup: forbiddenAssetTestGroup) { status in
|
|
switch status {
|
|
case HTTPStatus.Forbidden:
|
|
hasForbiddenAsset = true
|
|
case HTTPStatus.PartialContent:
|
|
hasAllowedAsset = true
|
|
case HTTPStatus.OK:
|
|
hasAllowedAsset = true
|
|
default:
|
|
break
|
|
}
|
|
}
|
|
} else if instance.app == .piped {
|
|
self.testPipedAssets(asset: asset!, requestRange: requestRange, isHLS: false, forbiddenAssetTestGroup: forbiddenAssetTestGroup) { status in
|
|
switch status {
|
|
case HTTPStatus.Forbidden:
|
|
hasForbiddenAsset = true
|
|
case HTTPStatus.PartialContent:
|
|
hasAllowedAsset = true
|
|
case HTTPStatus.OK:
|
|
hasAllowedAsset = true
|
|
default:
|
|
break
|
|
}
|
|
}
|
|
}
|
|
} else if let firstHLS = hlsURLs.first {
|
|
let asset = AVURLAsset(url: firstHLS)
|
|
if instance.app == .piped {
|
|
self.testPipedAssets(asset: asset, requestRange: nil, isHLS: true, forbiddenAssetTestGroup: forbiddenAssetTestGroup) { status in
|
|
switch status {
|
|
case HTTPStatus.Forbidden:
|
|
hasForbiddenAsset = true
|
|
case HTTPStatus.PartialContent:
|
|
hasAllowedAsset = true
|
|
case HTTPStatus.OK:
|
|
hasAllowedAsset = true
|
|
default:
|
|
break
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
forbiddenAssetTestGroup.wait()
|
|
|
|
// Post-processing code
|
|
if instance.app == .invidious, hasForbiddenAsset || instance.proxiesVideos {
|
|
if let audio = stream.audioAsset {
|
|
stream.audioAsset = InvidiousAPI.proxiedAsset(instance: instance, asset: audio)
|
|
}
|
|
if let video = stream.videoAsset {
|
|
stream.videoAsset = InvidiousAPI.proxiedAsset(instance: instance, asset: video)
|
|
}
|
|
} else if instance.app == .piped, !instance.proxiesVideos, !hasForbiddenAsset {
|
|
if let hlsURL = stream.hlsURL {
|
|
PipedAPI.nonProxiedAsset(url: hlsURL) { possibleNonProxiedURL in
|
|
if let nonProxiedURL = possibleNonProxiedURL {
|
|
stream.hlsURL = nonProxiedURL.url
|
|
}
|
|
}
|
|
} else {
|
|
if let audio = stream.audioAsset {
|
|
PipedAPI.nonProxiedAsset(asset: audio) { nonProxiedAudioAsset in
|
|
stream.audioAsset = nonProxiedAudioAsset
|
|
}
|
|
}
|
|
if let video = stream.videoAsset {
|
|
PipedAPI.nonProxiedAsset(asset: video) { nonProxiedVideoAsset in
|
|
stream.videoAsset = nonProxiedVideoAsset
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Append to processedStreams within the processedStreamsQueue
|
|
processedStreamsQueue.sync {
|
|
processedStreams.append(stream)
|
|
}
|
|
}
|
|
}
|
|
|
|
streamProcessingGroup.notify(queue: .main) {
|
|
// Access and pass processedStreams within the processedStreamsQueue block
|
|
processedStreamsQueue.sync {
|
|
completion(processedStreams)
|
|
}
|
|
}
|
|
}
|
|
|
|
private func getAssets(from streams: [Stream]) -> (nonHLSAssets: [(AVURLAsset?, URL, String?)], hlsURLs: [URL]) {
|
|
var nonHLSAssets = [(AVURLAsset?, URL, String?)]()
|
|
var hlsURLs = [URL]()
|
|
|
|
for stream in streams {
|
|
if stream.isHLS {
|
|
if let url = stream.hlsURL?.url {
|
|
hlsURLs.append(url)
|
|
}
|
|
} else {
|
|
if let asset = stream.audioAsset {
|
|
nonHLSAssets.append((asset, asset.url, stream.requestRange))
|
|
}
|
|
if let asset = stream.videoAsset {
|
|
nonHLSAssets.append((asset, asset.url, stream.requestRange))
|
|
}
|
|
}
|
|
}
|
|
|
|
return (nonHLSAssets, hlsURLs)
|
|
}
|
|
|
|
private func testAsset(url: URL, range: String?, isHLS: Bool, forbiddenAssetTestGroup: DispatchGroup, completion: @escaping (Int) -> Void) {
|
|
// In case the range is nil, generate a random one.
|
|
let randomEnd = Int.random(in: 200 ... 800)
|
|
let requestRange = range ?? "0-\(randomEnd)"
|
|
|
|
forbiddenAssetTestGroup.enter()
|
|
URLTester.testURLResponse(url: url, range: requestRange, isHLS: isHLS) { statusCode in
|
|
completion(statusCode)
|
|
forbiddenAssetTestGroup.leave()
|
|
}
|
|
}
|
|
|
|
private func testPipedAssets(asset: AVURLAsset, requestRange: String?, isHLS: Bool, forbiddenAssetTestGroup: DispatchGroup, completion: @escaping (Int) -> Void) {
|
|
PipedAPI.nonProxiedAsset(asset: asset) { possibleNonProxiedAsset in
|
|
if let nonProxiedAsset = possibleNonProxiedAsset {
|
|
self.testAsset(url: nonProxiedAsset.url, range: requestRange, isHLS: isHLS, forbiddenAssetTestGroup: forbiddenAssetTestGroup, completion: completion)
|
|
} else {
|
|
completion(0)
|
|
}
|
|
}
|
|
}
|
|
|
|
func streamsSorter(lhs: Stream, rhs: Stream) -> Bool {
|
|
// Use optional chaining to simplify nil handling
|
|
guard let lhsRes = lhs.resolution?.height, let rhsRes = rhs.resolution?.height else {
|
|
return lhs.kind < rhs.kind
|
|
}
|
|
|
|
// Compare either kind or resolution based on conditions
|
|
return lhs.kind == rhs.kind ? (lhsRes > rhsRes) : (lhs.kind < rhs.kind)
|
|
}
|
|
}
|