2021-03-18 12:42:26 +01:00
|
|
|
//
|
|
|
|
// MastodonAttachmentService+UploadState.swift
|
|
|
|
// Mastodon
|
|
|
|
//
|
|
|
|
// Created by MainasuK Cirno on 2021-3-18.
|
|
|
|
//
|
|
|
|
|
|
|
|
import os.log
|
|
|
|
import Foundation
|
2021-07-15 11:25:15 +02:00
|
|
|
import Combine
|
2021-03-18 12:42:26 +01:00
|
|
|
import GameplayKit
|
|
|
|
import MastodonSDK
|
|
|
|
|
|
|
|
extension MastodonAttachmentService {
|
|
|
|
class UploadState: GKState {
|
|
|
|
weak var service: MastodonAttachmentService?
|
|
|
|
|
|
|
|
init(service: MastodonAttachmentService) {
|
|
|
|
self.service = service
|
|
|
|
}
|
|
|
|
|
|
|
|
override func didEnter(from previousState: GKState?) {
|
|
|
|
os_log("%{public}s[%{public}ld], %{public}s: enter %s, previous: %s", ((#file as NSString).lastPathComponent), #line, #function, self.debugDescription, previousState.debugDescription)
|
|
|
|
service?.uploadStateMachineSubject.send(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension MastodonAttachmentService.UploadState {
|
|
|
|
|
|
|
|
class Initial: MastodonAttachmentService.UploadState {
|
|
|
|
override func isValidNextState(_ stateClass: AnyClass) -> Bool {
|
|
|
|
guard service?.authenticationBox != nil else { return false }
|
2021-03-22 11:40:32 +01:00
|
|
|
if stateClass == Initial.self {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2021-05-31 10:42:49 +02:00
|
|
|
if service?.file.value != nil {
|
2021-03-22 11:40:32 +01:00
|
|
|
return stateClass == Uploading.self
|
|
|
|
} else {
|
|
|
|
return stateClass == Fail.self
|
|
|
|
}
|
2021-03-18 12:42:26 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class Uploading: MastodonAttachmentService.UploadState {
|
2021-07-15 11:25:15 +02:00
|
|
|
var needsFallback = false
|
|
|
|
|
2021-03-18 12:42:26 +01:00
|
|
|
override func isValidNextState(_ stateClass: AnyClass) -> Bool {
|
2021-08-09 11:02:32 +02:00
|
|
|
return stateClass == Fail.self
|
|
|
|
|| stateClass == Finish.self
|
|
|
|
|| stateClass == Uploading.self
|
|
|
|
|| stateClass == Processing.self
|
2021-03-18 12:42:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
override func didEnter(from previousState: GKState?) {
|
|
|
|
super.didEnter(from: previousState)
|
|
|
|
|
|
|
|
guard let service = service, let stateMachine = stateMachine else { return }
|
|
|
|
guard let authenticationBox = service.authenticationBox else { return }
|
2021-05-31 10:42:49 +02:00
|
|
|
guard let file = service.file.value else { return }
|
2021-03-18 12:42:26 +01:00
|
|
|
|
|
|
|
let description = service.description.value
|
2021-06-14 23:09:50 +02:00
|
|
|
let query = Mastodon.API.Media.UploadMediaQuery(
|
2021-03-18 12:42:26 +01:00
|
|
|
file: file,
|
|
|
|
thumbnail: nil,
|
|
|
|
description: description,
|
|
|
|
focus: nil
|
|
|
|
)
|
2021-07-15 11:25:15 +02:00
|
|
|
|
|
|
|
// and needs clone the `query` if needs retry
|
2021-03-18 12:42:26 +01:00
|
|
|
service.context.apiService.uploadMedia(
|
|
|
|
domain: authenticationBox.domain,
|
|
|
|
query: query,
|
2021-07-15 11:25:15 +02:00
|
|
|
mastodonAuthenticationBox: authenticationBox,
|
|
|
|
needsFallback: needsFallback
|
2021-03-18 12:42:26 +01:00
|
|
|
)
|
|
|
|
.receive(on: DispatchQueue.main)
|
2021-07-15 11:25:15 +02:00
|
|
|
.sink { [weak self] completion in
|
|
|
|
guard let self = self else { return }
|
2021-03-18 12:42:26 +01:00
|
|
|
switch completion {
|
|
|
|
case .failure(let error):
|
2021-07-15 11:25:15 +02:00
|
|
|
if let apiError = error as? Mastodon.API.Error,
|
|
|
|
apiError.httpResponseStatus == .notFound,
|
|
|
|
self.needsFallback == false
|
|
|
|
{
|
|
|
|
self.needsFallback = true
|
|
|
|
stateMachine.enter(Uploading.self)
|
2021-07-20 10:40:04 +02:00
|
|
|
os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s: upload attachment fallback to V1", ((#file as NSString).lastPathComponent), #line, #function)
|
2021-07-15 11:25:15 +02:00
|
|
|
} else {
|
|
|
|
os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s: upload attachment fail: %s", ((#file as NSString).lastPathComponent), #line, #function, error.localizedDescription)
|
|
|
|
service.error.send(error)
|
|
|
|
stateMachine.enter(Fail.self)
|
|
|
|
}
|
2021-03-18 12:42:26 +01:00
|
|
|
case .finished:
|
|
|
|
os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s: upload attachment success", ((#file as NSString).lastPathComponent), #line, #function)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
} receiveValue: { response in
|
2021-07-15 11:25:15 +02:00
|
|
|
os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s: upload attachment %s success: %s", ((#file as NSString).lastPathComponent), #line, #function, response.value.id, response.value.url ?? "<nil>")
|
2021-03-18 12:42:26 +01:00
|
|
|
service.attachment.value = response.value
|
2021-08-09 11:02:32 +02:00
|
|
|
if response.statusCode == 202 {
|
|
|
|
// check if still processing
|
|
|
|
stateMachine.enter(Processing.self)
|
|
|
|
} else {
|
|
|
|
stateMachine.enter(Finish.self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
.store(in: &service.disposeBag)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class Processing: MastodonAttachmentService.UploadState {
|
|
|
|
|
|
|
|
static let retryLimit = 10
|
|
|
|
var retryCount = 0
|
|
|
|
|
|
|
|
override func isValidNextState(_ stateClass: AnyClass) -> Bool {
|
|
|
|
return stateClass == Fail.self || stateClass == Finish.self || stateClass == Processing.self
|
|
|
|
}
|
|
|
|
|
|
|
|
override func didEnter(from previousState: GKState?) {
|
|
|
|
super.didEnter(from: previousState)
|
|
|
|
|
|
|
|
guard let service = service, let stateMachine = stateMachine else { return }
|
|
|
|
guard let authenticationBox = service.authenticationBox else { return }
|
|
|
|
guard let attachment = service.attachment.value else { return }
|
|
|
|
|
|
|
|
retryCount += 1
|
|
|
|
guard retryCount < Processing.retryLimit else {
|
|
|
|
stateMachine.enter(Fail.self)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
service.context.apiService.getMedia(
|
|
|
|
attachmentID: attachment.id,
|
|
|
|
mastodonAuthenticationBox: authenticationBox
|
|
|
|
)
|
|
|
|
.retry(3)
|
|
|
|
.receive(on: DispatchQueue.main)
|
|
|
|
.sink { [weak self] completion in
|
|
|
|
guard let _ = self else { return }
|
|
|
|
switch completion {
|
|
|
|
case .failure(let error):
|
|
|
|
os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s: get attachment fail: %s", ((#file as NSString).lastPathComponent), #line, #function, error.localizedDescription)
|
|
|
|
service.error.send(error)
|
|
|
|
stateMachine.enter(Fail.self)
|
|
|
|
case .finished:
|
|
|
|
os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s: get attachment success", ((#file as NSString).lastPathComponent), #line, #function)
|
|
|
|
break
|
|
|
|
}
|
|
|
|
} receiveValue: { [weak self] response in
|
|
|
|
guard let self = self else { return }
|
|
|
|
guard let _ = response.value.url else {
|
|
|
|
DispatchQueue.main.asyncAfter(deadline: .now() + 2) { [weak self] in
|
|
|
|
os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s: processing, retry in 2s", ((#file as NSString).lastPathComponent), #line, #function)
|
|
|
|
self?.stateMachine?.enter(Processing.self)
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2021-03-18 12:42:26 +01:00
|
|
|
stateMachine.enter(Finish.self)
|
|
|
|
}
|
|
|
|
.store(in: &service.disposeBag)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class Fail: MastodonAttachmentService.UploadState {
|
|
|
|
override func isValidNextState(_ stateClass: AnyClass) -> Bool {
|
|
|
|
// allow discard publishing
|
|
|
|
return stateClass == Uploading.self || stateClass == Finish.self
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class Finish: MastodonAttachmentService.UploadState {
|
|
|
|
override func isValidNextState(_ stateClass: AnyClass) -> Bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|