metatext/ServiceLayer/Sources/ServiceLayer/Services/NavigationService.swift
2021-05-09 22:10:14 -07:00

222 lines
7.7 KiB
Swift

// Copyright © 2020 Metabolist. All rights reserved.
import Combine
import DB
import Foundation
import Mastodon
import MastodonAPI
public enum Navigation {
case url(URL)
case collection(CollectionService)
case profile(ProfileService)
case notification(NotificationService)
case searchScope(SearchScope)
case webfingerStart
case webfingerEnd
}
public struct NavigationService {
private let environment: AppEnvironment
private let mastodonAPIClient: MastodonAPIClient
private let contentDatabase: ContentDatabase
private let status: Status?
init(environment: AppEnvironment,
mastodonAPIClient: MastodonAPIClient,
contentDatabase: ContentDatabase,
status: Status? = nil) {
self.environment = environment
self.mastodonAPIClient = mastodonAPIClient
self.contentDatabase = contentDatabase
self.status = status
}
}
public extension NavigationService {
func item(url: URL) -> AnyPublisher<Navigation, Never> {
if let tag = tag(url: url) {
return Just(
.collection(
TimelineService(
timeline: .tag(tag),
environment: environment,
mastodonAPIClient: mastodonAPIClient,
contentDatabase: contentDatabase)))
.eraseToAnyPublisher()
} else if let accountId = accountId(url: url) {
return Just(.profile(profileService(id: accountId))).eraseToAnyPublisher()
} else if mastodonAPIClient.instanceURL.host == url.host, let statusId = url.statusId {
return Just(.collection(contextService(id: statusId))).eraseToAnyPublisher()
}
if url.shouldWebfinger {
return webfinger(url: url)
} else {
return Just(.url(url)).eraseToAnyPublisher()
}
}
func contextService(id: Status.Id) -> ContextService {
ContextService(id: id, environment: environment,
mastodonAPIClient: mastodonAPIClient,
contentDatabase: contentDatabase)
}
func profileService(id: Account.Id) -> ProfileService {
ProfileService(id: id,
environment: environment,
mastodonAPIClient: mastodonAPIClient,
contentDatabase: contentDatabase)
}
func profileService(account: Account, relationship: Relationship? = nil) -> ProfileService {
ProfileService(account: account,
relationship: relationship,
environment: environment,
mastodonAPIClient: mastodonAPIClient,
contentDatabase: contentDatabase)
}
func statusService(status: Status) -> StatusService {
StatusService(environment: environment,
status: status,
mastodonAPIClient: mastodonAPIClient,
contentDatabase: contentDatabase)
}
func accountService(account: Account) -> AccountService {
AccountService(account: account,
environment: environment,
mastodonAPIClient: mastodonAPIClient,
contentDatabase: contentDatabase)
}
func loadMoreService(loadMore: LoadMore) -> LoadMoreService {
LoadMoreService(loadMore: loadMore, mastodonAPIClient: mastodonAPIClient, contentDatabase: contentDatabase)
}
func notificationService(notification: MastodonNotification) -> NotificationService {
NotificationService(
notification: notification,
environment: environment,
mastodonAPIClient: mastodonAPIClient,
contentDatabase: contentDatabase)
}
func conversationService(conversation: Conversation) -> ConversationService {
ConversationService(
conversation: conversation,
environment: environment,
mastodonAPIClient: mastodonAPIClient,
contentDatabase: contentDatabase)
}
func announcementService(announcement: Announcement) -> AnnouncementService {
AnnouncementService(announcement: announcement,
environment: environment,
mastodonAPIClient: mastodonAPIClient,
contentDatabase: contentDatabase)
}
func timelineService(timeline: Timeline) -> TimelineService {
TimelineService(timeline: timeline,
environment: environment,
mastodonAPIClient: mastodonAPIClient,
contentDatabase: contentDatabase)
}
}
private extension NavigationService {
func tag(url: URL) -> String? {
if status?.tags.first(where: { $0.url.url.path.lowercased() == url.path.lowercased() }) != nil {
return url.lastPathComponent
} else if
mastodonAPIClient.instanceURL.host == url.host {
return url.tag
}
return nil
}
func accountId(url: URL) -> String? {
if let mentionId = status?.mentions.first(where: {
$0.url.url.path.lowercased() == url.path.lowercased()
})?.id {
return mentionId
} else if
mastodonAPIClient.instanceURL.host == url.host {
return url.accountId
}
return nil
}
func webfinger(url: URL) -> AnyPublisher<Navigation, Never> {
let navigationSubject = PassthroughSubject<Navigation, Never>()
let request = mastodonAPIClient.request(ResultsEndpoint.search(.init(query: url.absoluteString)))
.handleEvents(
receiveSubscription: { _ in navigationSubject.send(.webfingerStart) },
receiveCompletion: { _ in navigationSubject.send(.webfingerEnd) })
.map { results -> Navigation in
if let tag = results.hashtags.first {
return .collection(
TimelineService(
timeline: .tag(tag.name),
environment: environment,
mastodonAPIClient: mastodonAPIClient,
contentDatabase: contentDatabase))
} else if let account = results.accounts.first {
return .profile(profileService(account: account))
} else if let status = results.statuses.first {
return .collection(contextService(id: status.id))
} else {
return .url(url)
}
}
.replaceError(with: .url(url))
return navigationSubject.merge(with: request).eraseToAnyPublisher()
}
}
private extension URL {
var isAccountURL: Bool {
(pathComponents.count == 2 && pathComponents[1].starts(with: "@"))
|| (pathComponents.count == 3 && pathComponents[0...1] == ["/", "users"])
}
var accountId: Account.Id? {
if let accountId = pathComponents.last, pathComponents == ["/", "web", "accounts", accountId] {
return accountId
}
return nil
}
var statusId: Status.Id? {
guard let statusId = pathComponents.last else { return nil }
if pathComponents.count == 3, pathComponents[1].starts(with: "@") {
return statusId
} else if pathComponents == ["/", "web", "statuses", statusId] {
return statusId
}
return nil
}
var tag: String? {
if let tag = pathComponents.last, pathComponents == ["/", "tags", tag] {
return tag
}
return nil
}
var shouldWebfinger: Bool {
isAccountURL || accountId != nil || statusId != nil || tag != nil
}
}