2023-01-10 08:04:25 +01:00
|
|
|
import Foundation
|
|
|
|
import OAuthSwift
|
|
|
|
|
|
|
|
public typealias Scope = String
|
|
|
|
public typealias Scopes = [Scope]
|
|
|
|
public typealias Token = String
|
|
|
|
|
|
|
|
public enum MastodonClientError: Swift.Error {
|
|
|
|
case oAuthCancelled
|
|
|
|
}
|
|
|
|
|
|
|
|
public protocol MastodonClientProtocol {
|
|
|
|
static func request(for baseURL: URL, target: TargetType, withBearerToken token: String?) throws -> URLRequest
|
|
|
|
}
|
|
|
|
|
|
|
|
public extension MastodonClientProtocol {
|
|
|
|
static func request(for baseURL: URL, target: TargetType, withBearerToken token: String? = nil) throws -> URLRequest {
|
|
|
|
|
|
|
|
var urlComponents = URLComponents(url: baseURL.appendingPathComponent(target.path), resolvingAgainstBaseURL: false)
|
|
|
|
urlComponents?.queryItems = target.queryItems?.map { URLQueryItem(name: $0.0, value: $0.1) }
|
|
|
|
|
|
|
|
guard let url = urlComponents?.url else { throw NetworkingError.cannotCreateUrlRequest }
|
|
|
|
|
|
|
|
var request = URLRequest(url: url)
|
|
|
|
|
|
|
|
target.headers?.forEach { header in
|
|
|
|
request.setValue(header.1, forHTTPHeaderField: header.0)
|
|
|
|
}
|
|
|
|
|
|
|
|
if let token = token {
|
|
|
|
request.setValue("Bearer \(token)", forHTTPHeaderField: "Authorization")
|
|
|
|
}
|
|
|
|
|
|
|
|
request.httpMethod = target.method.rawValue
|
|
|
|
request.httpBody = target.httpBody
|
|
|
|
|
|
|
|
return request
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public class MastodonClient: MastodonClientProtocol {
|
|
|
|
|
|
|
|
let urlSession: URLSession
|
|
|
|
let baseURL: URL
|
|
|
|
|
|
|
|
/// oAuth
|
|
|
|
var oauthClient: OAuth2Swift?
|
|
|
|
var oAuthHandle: OAuthSwiftRequestHandle?
|
|
|
|
var oAuthContinuation: CheckedContinuation<OAuthSwiftCredential, Swift.Error>?
|
|
|
|
|
|
|
|
public init(baseURL: URL, urlSession: URLSession = .shared) {
|
|
|
|
self.baseURL = baseURL
|
|
|
|
self.urlSession = urlSession
|
|
|
|
}
|
|
|
|
|
|
|
|
public func getAuthenticated(token: Token) -> MastodonClientAuthenticated {
|
|
|
|
MastodonClientAuthenticated(baseURL: baseURL, urlSession: urlSession, token: token)
|
|
|
|
}
|
|
|
|
|
|
|
|
deinit {
|
|
|
|
oAuthContinuation?.resume(throwing: MastodonClientError.oAuthCancelled)
|
|
|
|
oAuthHandle?.cancel()
|
|
|
|
}
|
2023-01-15 12:41:55 +01:00
|
|
|
|
|
|
|
public func downloadJson<T>(_ type: T.Type, request: URLRequest) async throws -> T where T: Decodable {
|
|
|
|
let (data, response) = try await urlSession.data(for: request)
|
|
|
|
guard (response as? HTTPURLResponse)?.status?.responseType == .success else {
|
|
|
|
throw NetworkError.notSuccessResponse(response)
|
|
|
|
}
|
|
|
|
|
|
|
|
return try JSONDecoder().decode(type, from: data)
|
|
|
|
}
|
2023-01-10 08:04:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public class MastodonClientAuthenticated: MastodonClientProtocol {
|
|
|
|
|
|
|
|
public let token: Token
|
|
|
|
public let baseURL: URL
|
|
|
|
public let urlSession: URLSession
|
|
|
|
|
|
|
|
init(baseURL: URL, urlSession: URLSession, token: Token) {
|
|
|
|
self.token = token
|
|
|
|
self.baseURL = baseURL
|
|
|
|
self.urlSession = urlSession
|
|
|
|
}
|
|
|
|
|
|
|
|
public func getHomeTimeline(
|
|
|
|
maxId: StatusId? = nil,
|
|
|
|
sinceId: StatusId? = nil,
|
|
|
|
minId: StatusId? = nil,
|
|
|
|
limit: Int? = nil) async throws -> [Status] {
|
|
|
|
|
|
|
|
let request = try Self.request(
|
|
|
|
for: baseURL,
|
|
|
|
target: Mastodon.Timelines.home(maxId, sinceId, minId, limit),
|
|
|
|
withBearerToken: token
|
|
|
|
)
|
2023-01-15 12:41:55 +01:00
|
|
|
|
|
|
|
return try await downloadJson([Status].self, request: request)
|
2023-01-10 08:04:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public func getPublicTimeline(isLocal: Bool = false,
|
|
|
|
maxId: StatusId? = nil,
|
|
|
|
sinceId: StatusId? = nil) async throws -> [Status] {
|
|
|
|
|
|
|
|
let request = try Self.request(
|
|
|
|
for: baseURL,
|
|
|
|
target: Mastodon.Timelines.pub(isLocal, maxId, sinceId),
|
|
|
|
withBearerToken: token
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2023-01-15 12:41:55 +01:00
|
|
|
return try await downloadJson([Status].self, request: request)
|
2023-01-10 08:04:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public func getTagTimeline(tag: String,
|
|
|
|
isLocal: Bool = false,
|
|
|
|
maxId: StatusId? = nil,
|
|
|
|
sinceId: StatusId? = nil) async throws -> [Status] {
|
|
|
|
|
|
|
|
let request = try Self.request(
|
|
|
|
for: baseURL,
|
|
|
|
target: Mastodon.Timelines.tag(tag, isLocal, maxId, sinceId),
|
|
|
|
withBearerToken: token
|
|
|
|
)
|
|
|
|
|
2023-01-15 12:41:55 +01:00
|
|
|
return try await downloadJson([Status].self, request: request)
|
2023-01-10 08:04:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public func saveMarkers(_ markers: [Mastodon.Markers.Timeline: StatusId]) async throws -> Markers {
|
|
|
|
let request = try Self.request(
|
|
|
|
for: baseURL,
|
|
|
|
target: Mastodon.Markers.set(markers),
|
|
|
|
withBearerToken: token
|
|
|
|
)
|
|
|
|
|
2023-01-15 12:41:55 +01:00
|
|
|
return try await downloadJson(Markers.self, request: request)
|
2023-01-10 08:04:25 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public func readMarkers(_ markers: Set<Mastodon.Markers.Timeline>) async throws -> Markers {
|
|
|
|
let request = try Self.request(
|
|
|
|
for: baseURL,
|
|
|
|
target: Mastodon.Markers.read(markers),
|
|
|
|
withBearerToken: token
|
|
|
|
)
|
|
|
|
|
2023-01-15 12:41:55 +01:00
|
|
|
return try await downloadJson(Markers.self, request: request)
|
|
|
|
}
|
|
|
|
|
|
|
|
public func downloadJson<T>(_ type: T.Type, request: URLRequest) async throws -> T where T: Decodable {
|
|
|
|
let (data, response) = try await urlSession.data(for: request)
|
|
|
|
guard (response as? HTTPURLResponse)?.status?.responseType == .success else {
|
|
|
|
throw NetworkError.notSuccessResponse(response)
|
|
|
|
}
|
|
|
|
|
|
|
|
return try JSONDecoder().decode(type, from: data)
|
2023-01-10 08:04:25 +01:00
|
|
|
}
|
|
|
|
}
|