2020-03-10 01:19:24 +01:00
|
|
|
//
|
|
|
|
// NewsBlurAPICaller.swift
|
|
|
|
// Account
|
|
|
|
//
|
|
|
|
// Created by Anh-Quang Do on 3/9/20.
|
|
|
|
// Copyright (c) 2020 Ranchero Software, LLC. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Foundation
|
|
|
|
import RSWeb
|
|
|
|
|
|
|
|
enum NewsBlurError: LocalizedError {
|
|
|
|
case general(message: String)
|
|
|
|
|
|
|
|
var errorDescription: String? {
|
|
|
|
switch self {
|
|
|
|
case .general(let message):
|
|
|
|
return message
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
final class NewsBlurAPICaller: NSObject {
|
2020-03-10 04:26:11 +01:00
|
|
|
static let SessionIdCookie = "newsblur_sessionid"
|
2020-03-10 01:19:24 +01:00
|
|
|
|
|
|
|
private let baseURL = URL(string: "https://www.newsblur.com/")!
|
|
|
|
private var transport: Transport!
|
|
|
|
|
|
|
|
var credentials: Credentials?
|
|
|
|
weak var accountMetadata: AccountMetadata?
|
|
|
|
|
|
|
|
init(transport: Transport!) {
|
|
|
|
super.init()
|
|
|
|
self.transport = transport
|
|
|
|
}
|
|
|
|
|
|
|
|
func validateCredentials(completion: @escaping (Result<Credentials?, Error>) -> Void) {
|
|
|
|
let url = baseURL.appendingPathComponent("api/login")
|
|
|
|
let request = URLRequest(url: url, credentials: credentials)
|
|
|
|
|
|
|
|
transport.send(request: request, resultType: NewsBlurLoginResponse.self) { result in
|
|
|
|
switch result {
|
2020-03-10 04:26:11 +01:00
|
|
|
case .success(let response, let payload):
|
|
|
|
guard let url = response.url, let headerFields = response.allHeaderFields as? [String: String], payload?.code != -1 else {
|
2020-03-10 01:19:24 +01:00
|
|
|
let error = payload?.errors?.username ?? payload?.errors?.others
|
|
|
|
if let message = error?.first {
|
|
|
|
completion(.failure(NewsBlurError.general(message: message)))
|
|
|
|
} else {
|
|
|
|
completion(.failure(NewsBlurError.general(message: "Failed to log in")))
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
2020-03-10 04:26:11 +01:00
|
|
|
|
|
|
|
guard let username = self.credentials?.username else {
|
|
|
|
completion(.failure(NewsBlurError.general(message: "Failed to log in")))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let cookies = HTTPCookie.cookies(withResponseHeaderFields: headerFields, for: url)
|
|
|
|
for cookie in cookies where cookie.name == Self.SessionIdCookie {
|
|
|
|
let credentials = Credentials(type: .newsBlurSessionId, username: username, secret: cookie.value)
|
|
|
|
completion(.success(credentials))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
completion(.failure(NewsBlurError.general(message: "Failed to retrieve session")))
|
2020-03-10 01:19:24 +01:00
|
|
|
case .failure(let error):
|
|
|
|
completion(.failure(error))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func logout(completion: @escaping (Result<Void, Error>) -> Void) {
|
|
|
|
let url = baseURL.appendingPathComponent("api/logout")
|
|
|
|
let request = URLRequest(url: url, credentials: credentials)
|
|
|
|
|
|
|
|
transport.send(request: request) { result in
|
|
|
|
switch result {
|
|
|
|
case .success:
|
|
|
|
completion(.success(()))
|
|
|
|
case .failure(let error):
|
|
|
|
completion(.failure(error))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-03-10 04:26:48 +01:00
|
|
|
|
2020-03-13 23:18:47 +01:00
|
|
|
func retrieveSubscriptions(completion: @escaping (Result<[NewsBlurSubscription]?, Error>) -> Void) {
|
2020-03-10 04:26:48 +01:00
|
|
|
let url = baseURL
|
|
|
|
.appendingPathComponent("reader/feeds")
|
|
|
|
.appendingQueryItems([
|
|
|
|
URLQueryItem(name: "flat", value: "true"),
|
|
|
|
URLQueryItem(name: "update_counts", value: "false"),
|
|
|
|
])
|
|
|
|
|
|
|
|
guard let callURL = url else {
|
|
|
|
completion(.failure(TransportError.noURL))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let request = URLRequest(url: callURL, credentials: credentials)
|
|
|
|
transport.send(request: request, resultType: NewsBlurFeedsResponse.self) { result in
|
|
|
|
switch result {
|
2020-03-11 02:08:56 +01:00
|
|
|
case .success((_, let payload)):
|
2020-03-13 23:18:47 +01:00
|
|
|
completion(.success(payload?.subscriptions))
|
2020-03-11 02:08:56 +01:00
|
|
|
case .failure(let error):
|
|
|
|
completion(.failure(error))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-13 23:18:47 +01:00
|
|
|
func retrieveUnreadArticleHashes(completion: @escaping (Result<[NewsBlurArticleHash]?, Error>) -> Void) {
|
2020-03-11 02:08:56 +01:00
|
|
|
let url = baseURL
|
|
|
|
.appendingPathComponent("reader/unread_story_hashes")
|
|
|
|
.appendingQueryItems([
|
|
|
|
URLQueryItem(name: "include_timestamps", value: "true"),
|
|
|
|
])
|
|
|
|
|
|
|
|
guard let callURL = url else {
|
|
|
|
completion(.failure(TransportError.noURL))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let request = URLRequest(url: callURL, credentials: credentials)
|
|
|
|
transport.send(request: request, resultType: NewsBlurUnreadArticleHashesResponse.self, dateDecoding: .secondsSince1970) { result in
|
|
|
|
switch result {
|
|
|
|
case .success((_, let payload)):
|
2020-03-13 23:18:47 +01:00
|
|
|
completion(.success(payload?.subscriptions.values.flatMap { $0 }))
|
|
|
|
case .failure(let error):
|
|
|
|
completion(.failure(error))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func retrieveArticles(hashes: [NewsBlurArticleHash], completion: @escaping (Result<[NewsBlurArticle]?, Error>) -> Void) {
|
|
|
|
let url = baseURL
|
|
|
|
.appendingPathComponent("reader/river_stories")
|
|
|
|
.appendingQueryItem(.init(name: "include_hidden", value: "true"))?
|
|
|
|
.appendingQueryItems(hashes.map { URLQueryItem(name: "h", value: $0.hash) })
|
|
|
|
|
|
|
|
guard let callURL = url else {
|
|
|
|
completion(.failure(TransportError.noURL))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let request = URLRequest(url: callURL, credentials: credentials)
|
|
|
|
transport.send(request: request, resultType: NewsBlurArticlesResponse.self, dateDecoding: .formatted(NewsBlurDate.yyyyMMddHHmmss)) { result in
|
|
|
|
switch result {
|
|
|
|
case .success((_, let payload)):
|
|
|
|
completion(.success(payload?.articles))
|
2020-03-10 04:26:48 +01:00
|
|
|
case .failure(let error):
|
|
|
|
completion(.failure(error))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-03-10 01:19:24 +01:00
|
|
|
}
|