1
0
mirror of https://github.com/yattee/yattee.git synced 2024-12-14 06:10:32 +05:30
yattee/Model/Applications/PipedAPI.swift

556 lines
19 KiB
Swift
Raw Normal View History

2021-10-17 04:18:58 +05:30
import AVFoundation
import Foundation
import Siesta
import SwiftyJSON
2021-10-21 03:51:50 +05:30
final class PipedAPI: Service, ObservableObject, VideosAPI {
static var authorizedEndpoints = ["subscriptions", "subscribe", "unsubscribe", "user/playlists"]
2021-10-21 03:51:50 +05:30
@Published var account: Account!
2021-10-17 04:18:58 +05:30
2021-10-21 03:51:50 +05:30
init(account: Account? = nil) {
2021-10-17 04:18:58 +05:30
super.init()
guard account != nil else {
return
}
setAccount(account!)
}
2021-10-21 03:51:50 +05:30
func setAccount(_ account: Account) {
2021-10-17 04:18:58 +05:30
self.account = account
configure()
}
func configure() {
invalidateConfiguration()
2021-10-17 04:18:58 +05:30
configure {
$0.pipeline[.parsing].add(SwiftyJSONTransformer, contentTypes: ["*/json"])
}
configure(whenURLMatches: { url in self.needsAuthorization(url) }) {
$0.headers["Authorization"] = self.account.token
}
2021-10-21 03:51:50 +05:30
configureTransformer(pathPattern("channel/*")) { (content: Entity<JSON>) -> Channel? in
2021-12-17 22:09:26 +05:30
self.extractChannel(from: content.json)
2021-10-17 04:18:58 +05:30
}
2021-10-21 03:51:50 +05:30
2021-10-23 04:34:03 +05:30
configureTransformer(pathPattern("playlists/*")) { (content: Entity<JSON>) -> ChannelPlaylist? in
2021-12-17 22:09:26 +05:30
self.extractChannelPlaylist(from: content.json)
2021-10-23 04:34:03 +05:30
}
configureTransformer(pathPattern("user/playlists/create")) { (_: Entity<JSON>) in }
configureTransformer(pathPattern("user/playlists/delete")) { (_: Entity<JSON>) in }
configureTransformer(pathPattern("user/playlists/add")) { (_: Entity<JSON>) in }
configureTransformer(pathPattern("user/playlists/remove")) { (_: Entity<JSON>) in }
2021-10-21 03:51:50 +05:30
configureTransformer(pathPattern("streams/*")) { (content: Entity<JSON>) -> Video? in
2021-12-17 22:09:26 +05:30
self.extractVideo(from: content.json)
2021-10-21 03:51:50 +05:30
}
configureTransformer(pathPattern("trending")) { (content: Entity<JSON>) -> [Video] in
2021-12-17 22:09:26 +05:30
self.extractVideos(from: content.json)
2021-10-21 03:51:50 +05:30
}
configureTransformer(pathPattern("search")) { (content: Entity<JSON>) -> SearchPage in
let nextPage = content.json.dictionaryValue["nextpage"]?.stringValue
return SearchPage(
results: self.extractContentItems(from: content.json.dictionaryValue["items"]!),
nextPage: nextPage,
last: nextPage == "null"
)
2021-10-21 03:51:50 +05:30
}
configureTransformer(pathPattern("suggestions")) { (content: Entity<JSON>) -> [String] in
content.json.arrayValue.map(String.init)
}
configureTransformer(pathPattern("subscriptions")) { (content: Entity<JSON>) -> [Channel] in
2021-12-17 22:09:26 +05:30
content.json.arrayValue.map { self.extractChannel(from: $0)! }
}
configureTransformer(pathPattern("feed")) { (content: Entity<JSON>) -> [Video] in
2021-12-17 22:09:26 +05:30
content.json.arrayValue.map { self.extractVideo(from: $0)! }
}
2021-12-05 01:05:41 +05:30
configureTransformer(pathPattern("comments/*")) { (content: Entity<JSON>) -> CommentsPage in
let details = content.json.dictionaryValue
2021-12-17 22:09:26 +05:30
let comments = details["comments"]?.arrayValue.map { self.extractComment(from: $0)! } ?? []
let nextPage = details["nextpage"]?.stringValue
let disabled = details["disabled"]?.boolValue ?? false
2021-12-05 01:05:41 +05:30
return CommentsPage(comments: comments, nextPage: nextPage, disabled: disabled)
2021-12-05 01:05:41 +05:30
}
configureTransformer(pathPattern("user/playlists")) { (content: Entity<JSON>) -> [Playlist] in
content.json.arrayValue.map { self.extractUserPlaylist(from: $0)! }
}
if account.token.isNil {
updateToken()
}
}
func needsAuthorization(_ url: URL) -> Bool {
2021-12-17 22:09:26 +05:30
Self.authorizedEndpoints.contains { url.absoluteString.contains($0) }
}
2021-12-05 01:05:41 +05:30
func updateToken() {
guard !account.anonymous else {
return
}
account.token = nil
2021-12-05 01:05:41 +05:30
login.request(
.post,
json: ["username": account.username, "password": account.password]
)
.onSuccess { response in
self.account.token = response.json.dictionaryValue["token"]?.string ?? ""
self.configure()
}
}
var login: Resource {
resource(baseURL: account.url, path: "login")
2021-10-21 03:51:50 +05:30
}
func channel(_ id: String) -> Resource {
resource(baseURL: account.url, path: "channel/\(id)")
}
2021-11-02 03:26:18 +05:30
func channelVideos(_ id: String) -> Resource {
channel(id)
}
2021-10-23 04:34:03 +05:30
func channelPlaylist(_ id: String) -> Resource? {
resource(baseURL: account.url, path: "playlists/\(id)")
}
func trending(country: Country, category _: TrendingCategory? = nil) -> Resource {
2021-10-28 02:41:38 +05:30
resource(baseURL: account.instance.apiURL, path: "trending")
.withParam("region", country.rawValue)
}
func search(_ query: SearchQuery, page: String?) -> Resource {
let path = page.isNil ? "search" : "nextpage/search"
let resource = resource(baseURL: account.instance.apiURL, path: path)
.withParam("q", query.query)
.withParam("filter", "all")
if page.isNil {
return resource
}
return resource.withParam("nextpage", page)
}
func searchSuggestions(query: String) -> Resource {
2021-10-28 02:41:38 +05:30
resource(baseURL: account.instance.apiURL, path: "suggestions")
.withParam("query", query.lowercased())
}
func video(_ id: Video.ID) -> Resource {
2021-10-28 02:41:38 +05:30
resource(baseURL: account.instance.apiURL, path: "streams/\(id)")
}
var signedIn: Bool {
2022-04-04 03:48:49 +05:30
guard let account = account else {
return false
}
return !account.anonymous && !(account.token?.isEmpty ?? true)
}
var subscriptions: Resource? {
resource(baseURL: account.instance.apiURL, path: "subscriptions")
}
var feed: Resource? {
resource(baseURL: account.instance.apiURL, path: "feed")
.withParam("authToken", account.token)
}
var home: Resource? { nil }
var popular: Resource? { nil }
var playlists: Resource? {
resource(baseURL: account.instance.apiURL, path: "user/playlists")
}
func subscribe(_ channelID: String, onCompletion: @escaping () -> Void = {}) {
resource(baseURL: account.instance.apiURL, path: "subscribe")
.request(.post, json: ["channelId": channelID])
.onCompletion { _ in onCompletion() }
}
func unsubscribe(_ channelID: String, onCompletion: @escaping () -> Void = {}) {
resource(baseURL: account.instance.apiURL, path: "unsubscribe")
.request(.post, json: ["channelId": channelID])
.onCompletion { _ in onCompletion() }
}
func playlist(_ id: String) -> Resource? {
channelPlaylist(id)
}
func playlistVideo(_: String, _: String) -> Resource? { nil }
func playlistVideos(_: String) -> Resource? { nil }
func addVideoToPlaylist(
_ videoID: String,
_ playlistID: String,
onFailure: @escaping (RequestError) -> Void = { _ in },
onSuccess: @escaping () -> Void = {}
) {
let resource = resource(baseURL: account.instance.apiURL, path: "user/playlists/add")
let body = ["videoId": videoID, "playlistId": playlistID]
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 = resource(baseURL: account.instance.apiURL, path: "user/playlists/remove")
let body: [String: Any] = ["index": Int(index)!, "playlistId": playlistID]
resource
.request(.post, json: body)
.onSuccess { _ in onSuccess() }
.onFailure(onFailure)
}
func playlistForm(
_ name: String,
_: String,
playlist: Playlist?,
onFailure: @escaping (RequestError) -> Void,
onSuccess: @escaping (Playlist?) -> Void
) {
let body = ["name": name]
let resource = playlist.isNil ? resource(baseURL: account.instance.apiURL, path: "user/playlists/create") : nil
resource?
.request(.post, json: body)
.onSuccess { response in
if let modifiedPlaylist: Playlist = response.typedContent() {
onSuccess(modifiedPlaylist)
} else {
onSuccess(nil)
}
}
.onFailure(onFailure)
}
func deletePlaylist(
_ playlist: Playlist,
onFailure: @escaping (RequestError) -> Void,
onSuccess: @escaping () -> Void
) {
let resource = resource(baseURL: account.instance.apiURL, path: "user/playlists/delete")
let body = ["playlistId": playlist.id]
resource
.request(.post, json: body)
.onSuccess { _ in onSuccess() }
.onFailure(onFailure)
}
2021-12-05 01:05:41 +05:30
func comments(_ id: Video.ID, page: String?) -> Resource? {
let path = page.isNil ? "comments/\(id)" : "nextpage/comments/\(id)"
let resource = resource(baseURL: account.url, path: path)
if page.isNil {
return resource
}
return resource.withParam("nextpage", page)
}
private func pathPattern(_ path: String) -> String {
"**\(path)"
}
2021-12-17 22:09:26 +05:30
private func extractContentItem(from content: JSON) -> ContentItem? {
let details = content.dictionaryValue
let url: String! = details["url"]?.string
let contentType: ContentItem.ContentType
if !url.isNil {
if url.contains("/playlist") {
contentType = .playlist
} else if url.contains("/channel") {
contentType = .channel
} else {
contentType = .video
}
} else {
contentType = .video
}
switch contentType {
case .video:
2021-12-17 22:09:26 +05:30
if let video = extractVideo(from: content) {
return ContentItem(video: video)
}
case .playlist:
2021-12-17 22:09:26 +05:30
if let playlist = extractChannelPlaylist(from: content) {
2021-10-23 04:34:03 +05:30
return ContentItem(playlist: playlist)
}
case .channel:
2021-12-17 22:09:26 +05:30
if let channel = extractChannel(from: content) {
return ContentItem(channel: channel)
}
2022-03-27 16:19:57 +05:30
default:
return nil
}
return nil
}
2021-12-17 22:09:26 +05:30
private func extractContentItems(from content: JSON) -> [ContentItem] {
content.arrayValue.compactMap { extractContentItem(from: $0) }
}
2021-12-17 22:09:26 +05:30
private func extractChannel(from content: JSON) -> Channel? {
let attributes = content.dictionaryValue
guard let id = attributes["id"]?.stringValue ??
2021-10-23 04:34:03 +05:30
(attributes["url"] ?? attributes["uploaderUrl"])?.stringValue.components(separatedBy: "/").last
else {
return nil
}
let subscriptionsCount = attributes["subscriberCount"]?.intValue ?? attributes["subscribers"]?.intValue
var videos = [Video]()
if let relatedStreams = attributes["relatedStreams"] {
2021-12-17 22:09:26 +05:30
videos = extractVideos(from: relatedStreams)
}
2021-12-17 22:09:26 +05:30
let name = attributes["name"]?.stringValue ?? attributes["uploaderName"]?.stringValue ?? attributes["uploader"]?.stringValue ?? ""
let thumbnailURL = attributes["avatarUrl"]?.url ?? attributes["uploaderAvatar"]?.url ?? attributes["avatar"]?.url ?? attributes["thumbnail"]?.url
return Channel(
id: id,
2021-12-17 22:09:26 +05:30
name: name,
thumbnailURL: thumbnailURL,
subscriptionsCount: subscriptionsCount,
videos: videos
2021-10-21 03:51:50 +05:30
)
}
2021-12-17 22:09:26 +05:30
func extractChannelPlaylist(from json: JSON) -> ChannelPlaylist? {
2021-10-23 04:34:03 +05:30
let details = json.dictionaryValue
let id = details["url"]?.stringValue.components(separatedBy: "?list=").last ?? UUID().uuidString
let thumbnailURL = details["thumbnail"]?.url ?? details["thumbnailUrl"]?.url
var videos = [Video]()
if let relatedStreams = details["relatedStreams"] {
2021-12-17 22:09:26 +05:30
videos = extractVideos(from: relatedStreams)
2021-10-23 04:34:03 +05:30
}
return ChannelPlaylist(
id: id,
2022-05-22 21:23:12 +05:30
title: details["name"]?.stringValue ?? "",
2021-10-23 04:34:03 +05:30
thumbnailURL: thumbnailURL,
2021-10-23 17:21:02 +05:30
channel: extractChannel(from: json)!,
2021-10-23 04:34:03 +05:30
videos: videos,
videosCount: details["videos"]?.int
)
}
2021-12-17 22:09:26 +05:30
private func extractVideo(from content: JSON) -> Video? {
2021-10-21 03:51:50 +05:30
let details = content.dictionaryValue
let url = details["url"]?.string
if !url.isNil {
guard url!.contains("/watch") else {
return nil
}
}
let channelId = details["uploaderUrl"]!.stringValue.components(separatedBy: "/").last!
let thumbnails: [Thumbnail] = Thumbnail.Quality.allCases.compactMap {
2021-12-17 22:09:26 +05:30
if let url = buildThumbnailURL(from: content, quality: $0) {
2021-10-21 03:51:50 +05:30
return Thumbnail(url: url, quality: $0)
}
return nil
}
let author = details["uploaderName"]?.stringValue ?? details["uploader"]!.stringValue
2021-12-17 22:09:26 +05:30
let authorThumbnailURL = details["avatarUrl"]?.url ?? details["uploaderAvatar"]?.url ?? details["avatar"]?.url
let subscriptionsCount = details["uploaderSubscriberCount"]?.int
2021-12-17 22:09:26 +05:30
let uploaded = details["uploaded"]?.doubleValue
2022-06-14 21:53:15 +05:30
var published = (uploaded.isNil || uploaded == -1) ? nil : (uploaded! / 1000).formattedAsRelativeTime()
if published.isNil {
published = (details["uploadedDate"] ?? details["uploadDate"])?.stringValue ?? ""
}
2021-12-27 00:44:45 +05:30
let live = details["livestream"]?.boolValue ?? (details["duration"]?.intValue == -1)
2021-10-21 03:51:50 +05:30
return Video(
2021-12-17 22:09:26 +05:30
videoID: extractID(from: content),
2021-10-21 03:51:50 +05:30
title: details["title"]!.stringValue,
author: author,
length: details["duration"]!.doubleValue,
published: published!,
2021-10-21 03:51:50 +05:30
views: details["views"]!.intValue,
2021-12-17 22:09:26 +05:30
description: extractDescription(from: content),
channel: Channel(id: channelId, name: author, thumbnailURL: authorThumbnailURL, subscriptionsCount: subscriptionsCount),
2021-10-21 03:51:50 +05:30
thumbnails: thumbnails,
2021-12-27 00:44:45 +05:30
live: live,
2021-10-21 03:51:50 +05:30
likes: details["likes"]?.int,
dislikes: details["dislikes"]?.int,
2021-11-03 04:32:02 +05:30
streams: extractStreams(from: content),
related: extractRelated(from: content)
2021-10-21 03:51:50 +05:30
)
}
2021-12-17 22:09:26 +05:30
private func extractID(from content: JSON) -> Video.ID {
2021-10-21 03:51:50 +05:30
content.dictionaryValue["url"]?.stringValue.components(separatedBy: "?v=").last ??
2021-10-23 17:21:02 +05:30
extractThumbnailURL(from: content)!.relativeString.components(separatedBy: "/")[4]
2021-10-17 04:18:58 +05:30
}
2021-12-17 22:09:26 +05:30
private func extractThumbnailURL(from content: JSON) -> URL? {
2021-10-21 03:51:50 +05:30
content.dictionaryValue["thumbnail"]?.url! ?? content.dictionaryValue["thumbnailUrl"]!.url!
}
2021-12-17 22:09:26 +05:30
private func buildThumbnailURL(from content: JSON, quality: Thumbnail.Quality) -> URL? {
2021-10-23 17:21:02 +05:30
let thumbnailURL = extractThumbnailURL(from: content)
2021-10-21 03:51:50 +05:30
guard !thumbnailURL.isNil else {
return nil
}
return URL(string: thumbnailURL!
.absoluteString
.replacingOccurrences(of: "hqdefault", with: quality.filename)
.replacingOccurrences(of: "maxresdefault", with: quality.filename)
)!
}
private func extractUserPlaylist(from json: JSON) -> Playlist? {
let id = json["id"].stringValue
let title = json["name"].stringValue
let visibility = Playlist.Visibility.private
return Playlist(id: id, title: title, visibility: visibility)
}
2021-12-17 22:09:26 +05:30
private func extractDescription(from content: JSON) -> String? {
2021-10-21 03:51:50 +05:30
guard var description = content.dictionaryValue["description"]?.string else {
return nil
}
description = description.replacingOccurrences(
of: "<br/>|<br />|<br>",
with: "\n",
options: .regularExpression,
range: nil
)
description = description.replacingOccurrences(
of: "<[^>]+>",
with: "",
options: .regularExpression,
range: nil
)
return description
}
2021-12-17 22:09:26 +05:30
private func extractVideos(from content: JSON) -> [Video] {
2021-10-23 17:21:02 +05:30
content.arrayValue.compactMap(extractVideo(from:))
2021-10-21 03:51:50 +05:30
}
2021-12-17 22:09:26 +05:30
private func extractStreams(from content: JSON) -> [Stream] {
2021-10-17 04:18:58 +05:30
var streams = [Stream]()
2021-10-21 03:51:50 +05:30
if let hlsURL = content.dictionaryValue["hls"]?.url {
2021-10-17 04:18:58 +05:30
streams.append(Stream(hlsURL: hlsURL))
}
2022-02-17 01:53:11 +05:30
let audioStreams = content
.dictionaryValue["audioStreams"]?
.arrayValue
.filter { $0.dictionaryValue["format"]?.stringValue == "M4A" }
.sorted {
$0.dictionaryValue["bitrate"]?.intValue ?? 0 > $1.dictionaryValue["bitrate"]?.intValue ?? 0
} ?? []
guard let audioStream = audioStreams.first else {
2021-10-17 04:18:58 +05:30
return streams
}
2022-02-17 01:53:11 +05:30
let videoStreams = content.dictionaryValue["videoStreams"]?.arrayValue ?? []
2021-10-17 04:18:58 +05:30
videoStreams.forEach { videoStream in
2022-07-11 19:00:32 +05:30
guard let audioAssetUrl = audioStream.dictionaryValue["url"]?.url,
let videoAssetUrl = videoStream.dictionaryValue["url"]?.url
else {
return
}
let audioAsset = AVURLAsset(url: audioAssetUrl)
let videoAsset = AVURLAsset(url: videoAssetUrl)
2021-10-17 04:18:58 +05:30
let videoOnly = videoStream.dictionaryValue["videoOnly"]?.boolValue ?? true
let resolution = Stream.Resolution.from(resolution: videoStream.dictionaryValue["quality"]!.stringValue)
2022-02-17 01:53:11 +05:30
let videoFormat = videoStream.dictionaryValue["format"]?.stringValue
2021-10-17 04:18:58 +05:30
if videoOnly {
streams.append(
2022-02-17 01:53:11 +05:30
Stream(audioAsset: audioAsset, videoAsset: videoAsset, resolution: resolution, kind: .adaptive, videoFormat: videoFormat)
2021-10-17 04:18:58 +05:30
)
} else {
streams.append(
SingleAssetStream(avAsset: videoAsset, resolution: resolution, kind: .stream)
)
}
}
return streams
}
2021-12-17 22:09:26 +05:30
private func extractRelated(from content: JSON) -> [Video] {
2021-11-03 04:32:02 +05:30
content
.dictionaryValue["relatedStreams"]?
.arrayValue
.compactMap(extractVideo(from:)) ?? []
}
2021-12-17 22:09:26 +05:30
private func extractComment(from content: JSON) -> Comment? {
2021-12-05 01:05:41 +05:30
let details = content.dictionaryValue
let author = details["author"]?.stringValue ?? ""
let commentorUrl = details["commentorUrl"]?.stringValue
let channelId = commentorUrl?.components(separatedBy: "/")[2] ?? ""
return Comment(
id: details["commentId"]?.stringValue ?? UUID().uuidString,
author: author,
authorAvatarURL: details["thumbnail"]?.stringValue ?? "",
time: details["commentedTime"]?.stringValue ?? "",
pinned: details["pinned"]?.boolValue ?? false,
hearted: details["hearted"]?.boolValue ?? false,
likeCount: details["likeCount"]?.intValue ?? 0,
text: details["commentText"]?.stringValue ?? "",
repliesPage: details["repliesPage"]?.stringValue,
channel: Channel(id: channelId, name: author)
)
}
2021-10-17 04:18:58 +05:30
}