mirror of
https://github.com/yattee/yattee.git
synced 2024-11-10 00:08:21 +00:00
133 lines
3.9 KiB
Swift
133 lines
3.9 KiB
Swift
import Foundation
|
|
import Siesta
|
|
import SwiftyJSON
|
|
|
|
extension TypedContentAccessors {
|
|
var json: JSON { typedContent(ifNone: JSON.null) }
|
|
}
|
|
|
|
let SwiftyJSONTransformer =
|
|
ResponseContentTransformer(transformErrors: true) { JSON($0.content as AnyObject) }
|
|
|
|
final class InvidiousAPI: Service {
|
|
static let shared = InvidiousAPI()
|
|
|
|
static let instance = "https://invidious.home.arekf.net"
|
|
|
|
static func proxyURLForAsset(_ url: String) -> URL? {
|
|
guard let instanceURLComponents = URLComponents(string: InvidiousAPI.instance),
|
|
var urlComponents = URLComponents(string: url) else { return nil }
|
|
|
|
urlComponents.scheme = instanceURLComponents.scheme
|
|
urlComponents.host = instanceURLComponents.host
|
|
|
|
return urlComponents.url
|
|
}
|
|
|
|
init() {
|
|
SiestaLog.Category.enabled = .all
|
|
|
|
super.init(baseURL: "\(InvidiousAPI.instance)/api/v1")
|
|
|
|
configure {
|
|
$0.pipeline[.parsing].add(SwiftyJSONTransformer, contentTypes: ["*/json"])
|
|
}
|
|
|
|
configure("/auth/**") {
|
|
$0.headers["Cookie"] = self.authHeader
|
|
}
|
|
|
|
configure("**", requestMethods: [.post]) {
|
|
$0.pipeline[.parsing].removeTransformers()
|
|
}
|
|
|
|
configureTransformer("/popular", requestMethods: [.get]) { (content: Entity<JSON>) -> [Video] in
|
|
content.json.arrayValue.map(Video.init)
|
|
}
|
|
|
|
configureTransformer("/trending", requestMethods: [.get]) { (content: Entity<JSON>) -> [Video] in
|
|
content.json.arrayValue.map(Video.init)
|
|
}
|
|
|
|
configureTransformer("/search", requestMethods: [.get]) { (content: Entity<JSON>) -> [Video] in
|
|
content.json.arrayValue.map(Video.init)
|
|
}
|
|
|
|
configureTransformer("/auth/playlists", requestMethods: [.get]) { (content: Entity<JSON>) -> [Playlist] in
|
|
content.json.arrayValue.map(Playlist.init)
|
|
}
|
|
|
|
configureTransformer("/auth/feed", requestMethods: [.get]) { (content: Entity<JSON>) -> [Video] in
|
|
if let feedVideos = content.json.dictionaryValue["videos"] {
|
|
return feedVideos.arrayValue.map { Video($0) }
|
|
}
|
|
|
|
return []
|
|
}
|
|
|
|
configureTransformer("/channels/*", requestMethods: [.get]) { (content: Entity<JSON>) -> [Video] in
|
|
if let channelVideos = content.json.dictionaryValue["latestVideos"] {
|
|
return channelVideos.arrayValue.map { Video($0) }
|
|
}
|
|
|
|
return []
|
|
}
|
|
|
|
configureTransformer("/videos/*", requestMethods: [.get]) { (content: Entity<JSON>) -> Video in
|
|
Video(content.json)
|
|
}
|
|
}
|
|
|
|
var authHeader: String? = "SID=\(Profile().sid)"
|
|
|
|
var popular: Resource {
|
|
resource("/popular")
|
|
}
|
|
|
|
func trending(category: TrendingCategory, country: Country) -> Resource {
|
|
resource("/trending")
|
|
.withParam("type", category.name)
|
|
.withParam("region", country.rawValue)
|
|
}
|
|
|
|
var subscriptions: Resource {
|
|
resource("/auth/feed")
|
|
}
|
|
|
|
func channelVideos(_ id: String) -> Resource {
|
|
resource("/channels/\(id)")
|
|
}
|
|
|
|
func video(_ id: String) -> Resource {
|
|
resource("/videos/\(id)")
|
|
}
|
|
|
|
var playlists: Resource {
|
|
resource("/auth/playlists")
|
|
}
|
|
|
|
func search(_ query: String) -> Resource {
|
|
resource("/search")
|
|
.withParam("q", searchQuery(query))
|
|
}
|
|
|
|
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
|
|
}
|
|
|
|
return searchQuery.addingPercentEncoding(withAllowedCharacters: .alphanumerics)!
|
|
}
|
|
}
|