yattee/Model/Applications/InvidiousAPI.swift

499 lines
16 KiB
Swift
Raw Normal View History

import AVKit
2021-07-07 22:39:18 +00:00
import Defaults
2021-06-28 10:43:07 +00:00
import Foundation
import Siesta
import SwiftyJSON
2021-10-20 22:21:50 +00:00
final class InvidiousAPI: Service, ObservableObject, VideosAPI {
2021-09-25 08:18:22 +00:00
static let basePath = "/api/v1"
2021-06-28 10:43:07 +00:00
2021-10-20 22:21:50 +00:00
@Published var account: Account!
2021-06-28 10:43:07 +00:00
2021-10-16 22:48:58 +00:00
@Published var validInstance = true
2021-09-29 11:45:00 +00:00
@Published var signedIn = false
2021-06-28 10:43:07 +00:00
2021-10-20 22:21:50 +00:00
init(account: Account? = nil) {
2021-10-16 22:48:58 +00:00
super.init()
guard !account.isNil else {
2021-10-17 21:49:56 +00:00
self.account = .init(name: "Empty")
2021-10-16 22:48:58 +00:00
return
}
setAccount(account!)
}
2021-10-20 22:21:50 +00:00
func setAccount(_ account: Account) {
2021-09-25 08:18:22 +00:00
self.account = account
signedIn = false
2021-09-25 08:18:22 +00:00
2021-12-19 16:56:47 +00:00
validInstance = account.anonymous
2021-09-25 08:18:22 +00:00
configure()
2021-12-19 16:56:47 +00:00
if !account.anonymous {
validate()
}
2021-09-25 08:18:22 +00:00
}
func validate() {
validateInstance()
validateSID()
}
func validateInstance() {
guard !validInstance else {
return
}
2021-10-20 22:21:50 +00:00
home?
2021-09-25 08:18:22 +00:00
.load()
.onSuccess { _ in
self.validInstance = true
}
.onFailure { _ in
self.validInstance = false
}
}
func validateSID() {
guard !signedIn else {
return
}
2021-10-20 22:21:50 +00:00
feed?
2021-09-25 08:18:22 +00:00
.load()
.onSuccess { _ in
self.signedIn = true
}
.onFailure { _ in
self.signedIn = false
}
}
func configure() {
2021-06-28 10:43:07 +00:00
configure {
if !self.account.username.isEmpty {
2021-10-16 22:48:58 +00:00
$0.headers["Cookie"] = self.cookieHeader
}
2021-06-28 10:43:07 +00:00
$0.pipeline[.parsing].add(SwiftyJSONTransformer, contentTypes: ["*/json"])
}
configure("**", requestMethods: [.post]) {
$0.pipeline[.parsing].removeTransformers()
}
2021-09-25 08:18:22 +00:00
configureTransformer(pathPattern("popular"), requestMethods: [.get]) { (content: Entity<JSON>) -> [Video] in
2021-12-17 16:39:26 +00:00
content.json.arrayValue.map(self.extractVideo)
2021-06-28 10:43:07 +00:00
}
2021-09-25 08:18:22 +00:00
configureTransformer(pathPattern("trending"), requestMethods: [.get]) { (content: Entity<JSON>) -> [Video] in
2021-12-17 16:39:26 +00:00
content.json.arrayValue.map(self.extractVideo)
2021-06-28 10:43:07 +00:00
}
configureTransformer(pathPattern("search"), requestMethods: [.get]) { (content: Entity<JSON>) -> SearchPage in
2022-03-28 19:26:38 +00:00
let results = content.json.arrayValue.compactMap { json -> ContentItem? in
let type = json.dictionaryValue["type"]?.stringValue
if type == "channel" {
return ContentItem(channel: self.extractChannel(from: json))
} else if type == "playlist" {
return ContentItem(playlist: self.extractChannelPlaylist(from: json))
2022-03-28 19:26:38 +00:00
} else if type == "video" {
return ContentItem(video: self.extractVideo(from: json))
}
2022-03-28 19:26:38 +00:00
return nil
}
return SearchPage(results: results, last: results.isEmpty)
2021-06-28 10:43:07 +00:00
}
2021-09-25 08:18:22 +00:00
configureTransformer(pathPattern("search/suggestions"), requestMethods: [.get]) { (content: Entity<JSON>) -> [String] in
2021-09-13 20:41:16 +00:00
if let suggestions = content.json.dictionaryValue["suggestions"] {
return suggestions.arrayValue.map(String.init)
}
return []
}
2021-09-25 08:18:22 +00:00
configureTransformer(pathPattern("auth/playlists"), requestMethods: [.get]) { (content: Entity<JSON>) -> [Playlist] in
2021-12-17 16:39:26 +00:00
content.json.arrayValue.map(self.extractPlaylist)
2021-06-28 10:43:07 +00:00
}
2021-09-25 08:18:22 +00:00
configureTransformer(pathPattern("auth/playlists/*"), requestMethods: [.get]) { (content: Entity<JSON>) -> Playlist in
2021-12-17 16:39:26 +00:00
self.extractPlaylist(from: content.json)
2021-08-29 21:36:18 +00:00
}
2021-09-25 08:18:22 +00:00
configureTransformer(pathPattern("auth/playlists"), requestMethods: [.post, .patch]) { (content: Entity<Data>) -> Playlist in
2021-07-08 15:14:54 +00:00
// hacky, to verify if possible to get it in easier way
Playlist(JSON(parseJSON: String(data: content.content, encoding: .utf8)!))
}
2021-09-25 08:18:22 +00:00
configureTransformer(pathPattern("auth/feed"), requestMethods: [.get]) { (content: Entity<JSON>) -> [Video] in
2021-06-28 10:43:07 +00:00
if let feedVideos = content.json.dictionaryValue["videos"] {
2021-12-17 16:39:26 +00:00
return feedVideos.arrayValue.map(self.extractVideo)
2021-06-28 10:43:07 +00:00
}
return []
}
2021-09-25 08:18:22 +00:00
configureTransformer(pathPattern("auth/subscriptions"), requestMethods: [.get]) { (content: Entity<JSON>) -> [Channel] in
2021-12-17 16:39:26 +00:00
content.json.arrayValue.map(self.extractChannel)
2021-08-25 22:12:59 +00:00
}
2021-09-25 08:18:22 +00:00
configureTransformer(pathPattern("channels/*"), requestMethods: [.get]) { (content: Entity<JSON>) -> Channel in
2021-12-17 16:39:26 +00:00
self.extractChannel(from: content.json)
2021-06-28 10:43:07 +00:00
}
2021-09-25 08:18:22 +00:00
configureTransformer(pathPattern("channels/*/latest"), requestMethods: [.get]) { (content: Entity<JSON>) -> [Video] in
2021-12-17 16:39:26 +00:00
content.json.arrayValue.map(self.extractVideo)
2021-09-18 20:36:42 +00:00
}
2021-10-22 23:04:03 +00:00
configureTransformer(pathPattern("playlists/*"), requestMethods: [.get]) { (content: Entity<JSON>) -> ChannelPlaylist in
2021-12-17 16:39:26 +00:00
self.extractChannelPlaylist(from: content.json)
2021-10-22 23:04:03 +00:00
}
2021-09-25 08:18:22 +00:00
configureTransformer(pathPattern("videos/*"), requestMethods: [.get]) { (content: Entity<JSON>) -> Video in
2021-12-17 16:39:26 +00:00
self.extractVideo(from: content.json)
2021-06-28 10:43:07 +00:00
}
}
private func pathPattern(_ path: String) -> String {
2021-09-25 08:18:22 +00:00
"**\(InvidiousAPI.basePath)/\(path)"
}
private func basePathAppending(_ path: String) -> String {
2021-09-25 08:18:22 +00:00
"\(InvidiousAPI.basePath)/\(path)"
}
private var cookieHeader: String {
"SID=\(account.username)"
2021-09-25 08:18:22 +00:00
}
2021-06-28 10:43:07 +00:00
2021-10-20 22:21:50 +00:00
var popular: Resource? {
2021-09-25 08:18:22 +00:00
resource(baseURL: account.url, path: "\(InvidiousAPI.basePath)/popular")
2021-06-28 10:43:07 +00:00
}
2021-10-20 22:21:50 +00:00
func trending(country: Country, category: TrendingCategory?) -> Resource {
2021-09-25 08:18:22 +00:00
resource(baseURL: account.url, path: "\(InvidiousAPI.basePath)/trending")
2021-11-04 22:01:27 +00:00
.withParam("type", category?.name)
2021-06-28 10:43:07 +00:00
.withParam("region", country.rawValue)
}
2021-10-20 22:21:50 +00:00
var home: Resource? {
2021-09-25 08:18:22 +00:00
resource(baseURL: account.url, path: "/feed/subscriptions")
2021-09-19 17:31:21 +00:00
}
2021-10-20 22:21:50 +00:00
var feed: Resource? {
2021-09-25 08:18:22 +00:00
resource(baseURL: account.url, path: "\(InvidiousAPI.basePath)/auth/feed")
}
2021-10-20 22:21:50 +00:00
var subscriptions: Resource? {
2021-09-25 08:18:22 +00:00
resource(baseURL: account.url, path: basePathAppending("auth/subscriptions"))
2021-08-25 22:12:59 +00:00
}
func subscribe(_ channelID: String, onCompletion: @escaping () -> Void = {}) {
resource(baseURL: account.url, path: basePathAppending("auth/subscriptions"))
.child(channelID)
.request(.post)
.onCompletion { _ in onCompletion() }
}
func unsubscribe(_ channelID: String, onCompletion: @escaping () -> Void) {
resource(baseURL: account.url, path: basePathAppending("auth/subscriptions"))
.child(channelID)
.request(.delete)
.onCompletion { _ in onCompletion() }
2021-08-25 22:12:59 +00:00
}
func channel(_ id: String) -> Resource {
2021-09-25 08:18:22 +00:00
resource(baseURL: account.url, path: basePathAppending("channels/\(id)"))
2021-06-28 10:43:07 +00:00
}
2021-09-18 20:36:42 +00:00
func channelVideos(_ id: String) -> Resource {
2021-09-25 08:18:22 +00:00
resource(baseURL: account.url, path: basePathAppending("channels/\(id)/latest"))
2021-09-18 20:36:42 +00:00
}
2021-06-28 10:43:07 +00:00
func video(_ id: String) -> Resource {
2021-09-25 08:18:22 +00:00
resource(baseURL: account.url, path: basePathAppending("videos/\(id)"))
2021-06-28 10:43:07 +00:00
}
2021-10-20 22:21:50 +00:00
var playlists: Resource? {
if account.isNil || account.anonymous {
return nil
}
return resource(baseURL: account.url, path: basePathAppending("auth/playlists"))
2021-06-28 10:43:07 +00:00
}
2021-10-20 22:21:50 +00:00
func playlist(_ id: String) -> Resource? {
2021-09-25 08:18:22 +00:00
resource(baseURL: account.url, path: basePathAppending("auth/playlists/\(id)"))
2021-07-08 17:18:36 +00:00
}
2021-10-20 22:21:50 +00:00
func playlistVideos(_ id: String) -> Resource? {
playlist(id)?.child("videos")
}
2021-10-20 22:21:50 +00:00
func playlistVideo(_ playlistID: String, _ videoID: String) -> Resource? {
playlist(playlistID)?.child("videos").child(videoID)
}
func addVideoToPlaylist(
_ videoID: String,
_ playlistID: String,
onFailure: @escaping (RequestError) -> Void = { _ in },
onSuccess: @escaping () -> Void = {}
) {
let resource = playlistVideos(playlistID)
let body = ["videoId": videoID]
resource?
.request(.post, json: body)
.onSuccess { _ in onSuccess() }
.onFailure(onFailure)
}
func removeVideoFromPlaylist(
_ index: String,
_ playlistID: String,
onFailure: @escaping (RequestError) -> Void,
onSuccess: @escaping () -> Void
) {
let resource = playlistVideo(playlistID, index)
resource?
.request(.delete)
.onSuccess { _ in onSuccess() }
.onFailure(onFailure)
}
func playlistForm(
_ name: String,
_ visibility: String,
playlist: Playlist?,
onFailure: @escaping (RequestError) -> Void,
onSuccess: @escaping (Playlist?) -> Void
) {
let body = ["title": name, "privacy": visibility]
let resource = !playlist.isNil ? self.playlist(playlist!.id) : playlists
resource?
.request(!playlist.isNil ? .patch : .post, json: body)
.onSuccess { response in
if let modifiedPlaylist: Playlist = response.typedContent() {
onSuccess(modifiedPlaylist)
}
}
.onFailure(onFailure)
}
func deletePlaylist(
_ playlist: Playlist,
onFailure: @escaping (RequestError) -> Void,
onSuccess: @escaping () -> Void
) {
self.playlist(playlist.id)?
.request(.delete)
.onSuccess { _ in onSuccess() }
.onFailure(onFailure)
}
2021-10-22 23:04:03 +00:00
func channelPlaylist(_ id: String) -> Resource? {
resource(baseURL: account.url, path: basePathAppending("playlists/\(id)"))
}
func search(_ query: SearchQuery, page: String?) -> Resource {
2021-09-25 08:18:22 +00:00
var resource = resource(baseURL: account.url, path: basePathAppending("search"))
2021-07-07 22:39:18 +00:00
.withParam("q", searchQuery(query.query))
.withParam("sort_by", query.sortBy.parameter)
.withParam("type", "all")
2021-07-07 22:39:18 +00:00
if let date = query.date, date != .any {
resource = resource.withParam("date", date.rawValue)
2021-07-07 22:39:18 +00:00
}
if let duration = query.duration, duration != .any {
resource = resource.withParam("duration", duration.rawValue)
2021-07-07 22:39:18 +00:00
}
if let page = page {
resource = resource.withParam("page", page)
}
2021-07-07 22:39:18 +00:00
return resource
2021-06-28 10:43:07 +00:00
}
2021-09-13 20:41:16 +00:00
func searchSuggestions(query: String) -> Resource {
2021-09-25 08:18:22 +00:00
resource(baseURL: account.url, path: basePathAppending("search/suggestions"))
2021-09-13 20:41:16 +00:00
.withParam("q", query.lowercased())
}
2021-12-04 19:35:41 +00:00
func comments(_: Video.ID, page _: String?) -> Resource? { nil }
2021-06-28 10:43:07 +00:00
private func searchQuery(_ query: String) -> String {
var searchQuery = query
let url = URLComponents(string: query)
if url != nil,
url!.host == "youtu.be"
{
searchQuery = url!.path.replacingOccurrences(of: "/", with: "")
}
let queryItem = url?.queryItems?.first { item in item.name == "v" }
if let id = queryItem?.value {
searchQuery = id
}
2021-07-08 15:14:54 +00:00
return searchQuery
2021-06-28 10:43:07 +00:00
}
2021-10-22 15:00:09 +00:00
static func proxiedAsset(instance: Instance, asset: AVURLAsset) -> AVURLAsset? {
2021-10-27 21:11:38 +00:00
guard let instanceURLComponents = URLComponents(string: instance.apiURL),
2021-10-22 15:00:09 +00:00
var urlComponents = URLComponents(url: asset.url, resolvingAgainstBaseURL: false) else { return nil }
urlComponents.scheme = instanceURLComponents.scheme
urlComponents.host = instanceURLComponents.host
2021-10-22 15:00:09 +00:00
guard let url = urlComponents.url else {
return nil
}
return AVURLAsset(url: url)
}
2021-12-17 16:39:26 +00:00
func extractVideo(from json: JSON) -> Video {
let indexID: String?
var id: Video.ID
var publishedAt: Date?
if let publishedInterval = json["published"].double {
publishedAt = Date(timeIntervalSince1970: publishedInterval)
}
let videoID = json["videoId"].stringValue
if let index = json["indexId"].string {
indexID = index
id = videoID + index
} else {
indexID = nil
id = videoID
}
return Video(
id: id,
videoID: videoID,
title: json["title"].stringValue,
author: json["author"].stringValue,
length: json["lengthSeconds"].doubleValue,
published: json["publishedText"].stringValue,
views: json["viewCount"].intValue,
description: json["description"].stringValue,
genre: json["genre"].stringValue,
channel: extractChannel(from: json),
thumbnails: extractThumbnails(from: json),
indexID: indexID,
live: json["liveNow"].boolValue,
upcoming: json["isUpcoming"].boolValue,
publishedAt: publishedAt,
likes: json["likeCount"].int,
dislikes: json["dislikeCount"].int,
keywords: json["keywords"].arrayValue.map { $0.stringValue },
2021-11-02 23:02:02 +00:00
streams: extractStreams(from: json),
related: extractRelated(from: json)
)
}
2021-12-17 16:39:26 +00:00
func extractChannel(from json: JSON) -> Channel {
var thumbnailURL = json["authorThumbnails"].arrayValue.last?.dictionaryValue["url"]?.stringValue ?? ""
// append https protocol to unproxied thumbnail URL if it's missing
if thumbnailURL.count > 2,
String(thumbnailURL[..<thumbnailURL.index(thumbnailURL.startIndex, offsetBy: 2)]) == "//"
{
thumbnailURL = "https:\(thumbnailURL)"
}
return Channel(
id: json["authorId"].stringValue,
name: json["author"].stringValue,
thumbnailURL: URL(string: thumbnailURL),
subscriptionsCount: json["subCount"].int,
subscriptionsText: json["subCountText"].string,
2021-12-17 16:39:26 +00:00
videos: json.dictionaryValue["latestVideos"]?.arrayValue.map(extractVideo) ?? []
)
}
2021-12-17 16:39:26 +00:00
func extractChannelPlaylist(from json: JSON) -> ChannelPlaylist {
2021-10-22 23:04:03 +00:00
let details = json.dictionaryValue
return ChannelPlaylist(
id: details["playlistId"]!.stringValue,
title: details["title"]!.stringValue,
thumbnailURL: details["playlistThumbnail"]?.url,
channel: extractChannel(from: json),
2021-12-17 16:39:26 +00:00
videos: details["videos"]?.arrayValue.compactMap(extractVideo) ?? []
2021-10-22 23:04:03 +00:00
)
}
2021-12-17 16:39:26 +00:00
private func extractThumbnails(from details: JSON) -> [Thumbnail] {
details["videoThumbnails"].arrayValue.map { json in
Thumbnail(url: json["url"].url!, quality: .init(rawValue: json["quality"].string!)!)
}
}
2021-12-17 16:39:26 +00:00
private func extractStreams(from json: JSON) -> [Stream] {
2021-11-02 23:02:02 +00:00
extractFormatStreams(from: json["formatStreams"].arrayValue) +
extractAdaptiveFormats(from: json["adaptiveFormats"].arrayValue)
}
2021-12-17 16:39:26 +00:00
private func extractFormatStreams(from streams: [JSON]) -> [Stream] {
streams.map {
SingleAssetStream(
avAsset: AVURLAsset(url: $0["url"].url!),
resolution: Stream.Resolution.from(resolution: $0["resolution"].stringValue),
kind: .stream,
encoding: $0["encoding"].stringValue
)
}
}
2021-12-17 16:39:26 +00:00
private func extractAdaptiveFormats(from streams: [JSON]) -> [Stream] {
let audioAssetURL = streams.first { $0["type"].stringValue.starts(with: "audio/mp4") }
guard audioAssetURL != nil else {
return []
}
let videoAssetsURLs = streams.filter { $0["type"].stringValue.starts(with: "video/mp4") && $0["encoding"].stringValue == "h264" }
return videoAssetsURLs.map {
Stream(
audioAsset: AVURLAsset(url: audioAssetURL!["url"].url!),
videoAsset: AVURLAsset(url: $0["url"].url!),
resolution: Stream.Resolution.from(resolution: $0["resolution"].stringValue),
kind: .adaptive,
encoding: $0["encoding"].stringValue
)
}
}
2021-11-02 23:02:02 +00:00
2021-12-17 16:39:26 +00:00
private func extractRelated(from content: JSON) -> [Video] {
2021-11-02 23:02:02 +00:00
content
.dictionaryValue["recommendedVideos"]?
.arrayValue
.compactMap(extractVideo(from:)) ?? []
}
2021-12-17 16:39:26 +00:00
private func extractPlaylist(from content: JSON) -> Playlist {
.init(
id: content["playlistId"].stringValue,
title: content["title"].stringValue,
visibility: content["isListed"].boolValue ? .public : .private,
updated: content["updated"].doubleValue,
videos: content["videos"].arrayValue.map { extractVideo(from: $0) }
)
}
2021-06-28 10:43:07 +00:00
}