2018-09-11 14:52:21 +00:00
|
|
|
//
|
|
|
|
// Client.swift
|
|
|
|
// Pachyderm
|
|
|
|
//
|
|
|
|
// Created by Shadowfacts on 9/8/18.
|
|
|
|
// Copyright © 2018 Shadowfacts. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
|
|
|
|
/**
|
|
|
|
The base Mastodon API client.
|
|
|
|
*/
|
|
|
|
public class Client {
|
|
|
|
|
|
|
|
public typealias Callback<Result: Decodable> = (Response<Result>) -> Void
|
|
|
|
|
|
|
|
let baseURL: URL
|
|
|
|
let session: URLSession
|
|
|
|
|
|
|
|
public var accessToken: String?
|
|
|
|
|
|
|
|
public var appID: String?
|
|
|
|
public var clientID: String?
|
|
|
|
public var clientSecret: String?
|
|
|
|
|
|
|
|
public var timeoutInterval: TimeInterval = 60
|
|
|
|
|
|
|
|
lazy var decoder: JSONDecoder = {
|
|
|
|
let decoder = JSONDecoder()
|
|
|
|
let formatter = DateFormatter()
|
|
|
|
formatter.dateFormat = "yyyy-MM-dd'T'HH:mm:ss.SZ"
|
|
|
|
formatter.timeZone = TimeZone(abbreviation: "UTC")
|
|
|
|
formatter.locale = Locale(identifier: "en_US_POSIX")
|
|
|
|
decoder.dateDecodingStrategy = .formatted(formatter)
|
|
|
|
return decoder
|
|
|
|
}()
|
|
|
|
|
|
|
|
public init(baseURL: URL, accessToken: String? = nil, session: URLSession = .shared) {
|
|
|
|
self.baseURL = baseURL
|
|
|
|
self.accessToken = accessToken
|
|
|
|
self.session = session
|
|
|
|
}
|
|
|
|
|
2018-09-17 23:22:37 +00:00
|
|
|
public func run<Result>(_ request: Request<Result>, completion: @escaping Callback<Result>) {
|
2018-09-11 14:52:21 +00:00
|
|
|
guard let request = createURLRequest(request: request) else {
|
|
|
|
completion(.failure(Error.invalidRequest))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let task = session.dataTask(with: request) { data, response, error in
|
|
|
|
if let error = error {
|
|
|
|
completion(.failure(error))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
guard let data = data,
|
|
|
|
let response = response as? HTTPURLResponse else {
|
|
|
|
completion(.failure(Error.invalidResponse))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
guard response.statusCode == 200 else {
|
|
|
|
let mastodonError = try? self.decoder.decode(MastodonError.self, from: data)
|
|
|
|
let error = mastodonError.flatMap { Error.mastodonError($0.description) } ?? Error.unknownError
|
|
|
|
completion(.failure(error))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
guard let result = try? self.decoder.decode(Result.self, from: data) else {
|
|
|
|
completion(.failure(Error.invalidModel))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if var result = result as? ClientModel {
|
|
|
|
result.client = self
|
|
|
|
} else if var result = result as? [ClientModel] {
|
|
|
|
result.client = self
|
|
|
|
}
|
|
|
|
let pagination = response.allHeaderFields["Link"].flatMap { $0 as? String }.flatMap(Pagination.init)
|
|
|
|
|
|
|
|
completion(.success(result, pagination))
|
|
|
|
}
|
|
|
|
task.resume()
|
|
|
|
}
|
|
|
|
|
|
|
|
func createURLRequest<Result>(request: Request<Result>) -> URLRequest? {
|
|
|
|
guard var components = URLComponents(url: baseURL, resolvingAgainstBaseURL: true) else { return nil }
|
|
|
|
components.path = request.path
|
|
|
|
components.queryItems = request.queryParameters.queryItems
|
|
|
|
guard let url = components.url else { return nil }
|
|
|
|
var urlRequest = URLRequest(url: url, timeoutInterval: timeoutInterval)
|
|
|
|
urlRequest.httpMethod = request.method.name
|
|
|
|
urlRequest.httpBody = request.body.data
|
|
|
|
urlRequest.setValue(request.body.mimeType, forHTTPHeaderField: "Content-Type")
|
|
|
|
if let accessToken = accessToken {
|
|
|
|
urlRequest.setValue("Bearer \(accessToken)", forHTTPHeaderField: "Authorization")
|
|
|
|
}
|
|
|
|
return urlRequest
|
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Authorization
|
|
|
|
public func registerApp(name: String, redirectURI: String, scopes: [Scope], website: URL? = nil, completion: @escaping Callback<RegisteredApplication>) {
|
|
|
|
let request = Request<RegisteredApplication>(method: .post, path: "/api/v1/apps", body: .parameters([
|
|
|
|
"client_name" => name,
|
|
|
|
"redirect_uris" => redirectURI,
|
|
|
|
"scopes" => scopes.scopeString,
|
|
|
|
"website" => website?.absoluteString
|
|
|
|
]))
|
|
|
|
run(request) { result in
|
|
|
|
defer { completion(result) }
|
|
|
|
guard case let .success(application, _) = result else { return }
|
|
|
|
|
|
|
|
self.appID = application.id
|
|
|
|
self.clientID = application.clientID
|
|
|
|
self.clientSecret = application.clientSecret
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public func getAccessToken(authorizationCode: String, redirectURI: String, completion: @escaping Callback<LoginSettings>) {
|
|
|
|
let request = Request<LoginSettings>(method: .post, path: "/oauth/token", body: .parameters([
|
|
|
|
"client_id" => clientID,
|
|
|
|
"client_secret" => clientSecret,
|
|
|
|
"grant_type" => "authorization_code",
|
|
|
|
"code" => authorizationCode,
|
|
|
|
"redirect_uri" => redirectURI
|
|
|
|
]))
|
|
|
|
run(request) { result in
|
|
|
|
defer { completion(result) }
|
|
|
|
guard case let .success(loginSettings, _) = result else { return }
|
|
|
|
|
|
|
|
self.accessToken = loginSettings.accessToken
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Self
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getSelfAccount() -> Request<Account> {
|
|
|
|
return Request<Account>(method: .get, path: "/api/v1/accounts/verify_credentials")
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getFavourites() -> Request<[Status]> {
|
|
|
|
return Request<[Status]>(method: .get, path: "/api/v1/favourites")
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getRelationships(accounts: [Account]? = nil) -> Request<[Relationship]> {
|
|
|
|
return Request<[Relationship]>(method: .get, path: "/api/v1/accounts/relationships", queryParameters: "id" => accounts?.map { $0.id })
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getInstance() -> Request<Instance> {
|
|
|
|
return Request<Instance>(method: .get, path: "/api/v1/instance")
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getCustomEmoji() -> Request<[Emoji]> {
|
|
|
|
return Request<[Emoji]>(method: .get, path: "/api/v1/custom_emojis")
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Accounts
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getAccount(id: String) -> Request<Account> {
|
|
|
|
return Request<Account>(method: .get, path: "/api/v1/accounts/\(id)")
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 00:58:05 +00:00
|
|
|
public func searchForAccount(query: String, limit: Int? = nil, following: Bool? = nil) -> Request<[Account]> {
|
|
|
|
return Request<[Account]>(method: .get, path: "/api/v1/accounts/search", queryParameters: [
|
2018-09-11 14:52:21 +00:00
|
|
|
"q" => query,
|
|
|
|
"limit" => limit,
|
|
|
|
"following" => following
|
|
|
|
])
|
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Blocks
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getBlocks() -> Request<[Account]> {
|
|
|
|
return Request<[Account]>(method: .get, path: "/api/v1/blocks")
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getDomainBlocks() -> Request<[String]> {
|
|
|
|
return Request<[String]>(method: .get, path: "api/v1/domain_blocks")
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 00:58:05 +00:00
|
|
|
public func block(domain: String) -> Request<Empty> {
|
|
|
|
return Request<Empty>(method: .post, path: "/api/v1/domain_blocks", body: .parameters([
|
2018-09-11 14:52:21 +00:00
|
|
|
"domain" => domain
|
|
|
|
]))
|
|
|
|
}
|
|
|
|
|
2018-09-18 00:58:05 +00:00
|
|
|
public func unblock(domain: String) -> Request<Empty> {
|
|
|
|
return Request<Empty>(method: .delete, path: "/api/v1/domain_blocks", body: .parameters([
|
2018-09-11 14:52:21 +00:00
|
|
|
"domain" => domain
|
|
|
|
]))
|
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Filters
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getFilters() -> Request<[Filter]> {
|
|
|
|
return Request<[Filter]>(method: .get, path: "/api/v1/filters")
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 00:58:05 +00:00
|
|
|
public func createFilter(phrase: String, context: [Filter.Context], irreversible: Bool? = nil, wholeWord: Bool? = nil, expiresAt: Date? = nil) -> Request<Filter> {
|
|
|
|
return Request<Filter>(method: .post, path: "/api/v1/filters", body: .parameters([
|
2018-09-11 14:52:21 +00:00
|
|
|
"phrase" => phrase,
|
|
|
|
"irreversible" => irreversible,
|
|
|
|
"whole_word" => wholeWord,
|
|
|
|
"expires_at" => expiresAt
|
|
|
|
] + "context" => context.contextStrings))
|
|
|
|
}
|
|
|
|
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getFilter(id: String) -> Request<Filter> {
|
|
|
|
return Request<Filter>(method: .get, path: "/api/v1/filters/\(id)")
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Follows
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getFollowRequests(range: RequestRange = .default) -> Request<[Account]> {
|
2018-09-11 14:52:21 +00:00
|
|
|
var request = Request<[Account]>(method: .get, path: "/api/v1/follow_requests")
|
|
|
|
request.range = range
|
2018-09-18 00:58:05 +00:00
|
|
|
return request
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getFollowSuggestions() -> Request<[Account]> {
|
|
|
|
return Request<[Account]>(method: .get, path: "/api/v1/suggestions")
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 00:58:05 +00:00
|
|
|
public func followRemote(acct: String) -> Request<Account> {
|
|
|
|
return Request<Account>(method: .post, path: "/api/v1/follows", body: .parameters(["uri" => acct]))
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Lists
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getLists() -> Request<[List]> {
|
|
|
|
return Request<[List]>(method: .get, path: "/api/v1/lists")
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getList(id: String) -> Request<List> {
|
|
|
|
return Request<List>(method: .get, path: "/api/v1/lists/\(id)")
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 00:58:05 +00:00
|
|
|
public func createList(title: String) -> Request<List> {
|
|
|
|
return Request<List>(method: .post, path: "/api/v1/lists", body: .parameters(["title" => title]))
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Media
|
2018-09-18 00:58:05 +00:00
|
|
|
public func upload(attachment: FormAttachment, description: String? = nil, focus: (Float, Float)? = nil) -> Request<Attachment> {
|
|
|
|
return Request<Attachment>(method: .post, path: "/api/v1/media", body: .formData([
|
2018-09-11 14:52:21 +00:00
|
|
|
"description" => description,
|
|
|
|
"focus" => focus
|
|
|
|
], attachment))
|
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Mutes
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getMutes(range: RequestRange) -> Request<[Account]> {
|
2018-09-11 14:52:21 +00:00
|
|
|
var request = Request<[Account]>(method: .get, path: "/api/v1/mutes")
|
|
|
|
request.range = range
|
2018-09-18 00:58:05 +00:00
|
|
|
return request
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Notifications
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getNotifications(range: RequestRange = .default) -> Request<[Notification]> {
|
2018-09-11 14:52:21 +00:00
|
|
|
var request = Request<[Notification]>(method: .get, path: "/api/v1/notifications")
|
|
|
|
request.range = range
|
2018-09-18 00:58:05 +00:00
|
|
|
return request
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 00:58:05 +00:00
|
|
|
public func clearNotifications() -> Request<Empty> {
|
|
|
|
return Request<Empty>(method: .post, path: "/api/v1/notifications/clear")
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Reports
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getReports() -> Request<[Report]> {
|
|
|
|
return Request<[Report]>(method: .get, path: "/api/v1/reports")
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
2018-09-18 00:58:05 +00:00
|
|
|
public func report(account: Account, statuses: [Status], comment: String) -> Request<Report> {
|
|
|
|
return Request<Report>(method: .post, path: "/api/v1/reports", body: .parameters([
|
2018-09-11 14:52:21 +00:00
|
|
|
"account_id" => account.id,
|
|
|
|
"comment" => comment
|
|
|
|
] + "status_ids" => statuses.map { $0.id }))
|
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Search
|
2018-09-18 00:58:05 +00:00
|
|
|
public func search(query: String, resolve: Bool? = nil) -> Request<SearchResults> {
|
|
|
|
return Request<SearchResults>(method: .get, path: "/api/v2/search", queryParameters: [
|
2018-09-11 14:52:21 +00:00
|
|
|
"q" => query,
|
|
|
|
"resolve" => resolve
|
|
|
|
])
|
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Statuses
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getStatus(id: String) -> Request<Status> {
|
|
|
|
return Request<Status>(method: .get, path: "/api/v1/statuses/\(id)")
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public func createStatus(text: String,
|
|
|
|
inReplyTo: Status? = nil,
|
|
|
|
media: [Attachment]? = nil,
|
|
|
|
sensitive: Bool? = nil,
|
|
|
|
spoilerText: String? = nil,
|
|
|
|
visiblity: Status.Visibility? = nil,
|
2018-09-18 00:58:05 +00:00
|
|
|
language: String? = nil) -> Request<Status> {
|
|
|
|
return Request<Status>(method: .post, path: "/api/v1/statuses", body: .parameters([
|
2018-09-11 14:52:21 +00:00
|
|
|
"status" => text,
|
|
|
|
"in_reply_to_id" => inReplyTo?.id,
|
|
|
|
"sensitive" => sensitive,
|
|
|
|
"spoiler_text" => spoilerText,
|
|
|
|
"visibility" => visiblity?.rawValue,
|
|
|
|
"language" => language
|
2018-09-11 17:45:48 +00:00
|
|
|
] + "media_ids" => media?.map { $0.id }))
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// MARK: - Timelines
|
2018-09-18 00:58:05 +00:00
|
|
|
public func getStatuses(timeline: Timeline, range: RequestRange = .default) -> Request<[Status]> {
|
|
|
|
return timeline.request(range: range)
|
2018-09-11 14:52:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
extension Client {
|
|
|
|
public enum Error: Swift.Error {
|
|
|
|
case unknownError
|
|
|
|
case invalidRequest
|
|
|
|
case invalidResponse
|
|
|
|
case invalidModel
|
|
|
|
case mastodonError(String)
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|