Files
SideStore/Sources/SideStoreAppKit/Settings/Error Log/ErrorLogViewController.swift

284 lines
13 KiB
Swift
Raw Normal View History

//
// ErrorLogViewController.swift
// AltStore
//
// Created by Riley Testut on 9/6/22.
// Copyright © 2022 Riley Testut. All rights reserved.
//
import SafariServices
2023-03-01 00:48:36 -05:00
import UIKit
2023-03-01 00:48:36 -05:00
import SideStoreCore
2023-03-01 14:36:52 -05:00
import RoxasUIKit
import SideKit
import Nuke
import QuickLook
2023-03-01 00:48:36 -05:00
final class ErrorLogViewController: UITableViewController {
private lazy var dataSource = self.makeDataSource()
private var expandedErrorIDs = Set<NSManagedObjectID>()
2023-03-01 00:48:36 -05:00
private lazy var timeFormatter: DateFormatter = {
let dateFormatter = DateFormatter()
dateFormatter.dateStyle = .none
dateFormatter.timeStyle = .short
return dateFormatter
}()
2023-03-01 00:48:36 -05:00
override var preferredStatusBarStyle: UIStatusBarStyle {
2023-03-01 00:48:36 -05:00
.lightContent
}
2023-03-01 00:48:36 -05:00
override func viewDidLoad() {
super.viewDidLoad()
2023-03-01 00:48:36 -05:00
tableView.dataSource = dataSource
tableView.prefetchDataSource = dataSource
}
}
2023-03-01 00:48:36 -05:00
private extension ErrorLogViewController {
func makeDataSource() -> RSTFetchedResultsTableViewPrefetchingDataSource<LoggedError, UIImage> {
let fetchRequest = LoggedError.fetchRequest() as NSFetchRequest<LoggedError>
fetchRequest.sortDescriptors = [NSSortDescriptor(keyPath: \LoggedError.date, ascending: false)]
fetchRequest.returnsObjectsAsFaults = false
2023-03-01 00:48:36 -05:00
let fetchedResultsController = NSFetchedResultsController(fetchRequest: fetchRequest, managedObjectContext: DatabaseManager.shared.viewContext, sectionNameKeyPath: #keyPath(LoggedError.localizedDateString), cacheName: nil)
2023-03-01 00:48:36 -05:00
let dataSource = RSTFetchedResultsTableViewPrefetchingDataSource<LoggedError, UIImage>(fetchedResultsController: fetchedResultsController)
dataSource.proxy = self
dataSource.rowAnimation = .fade
2023-03-01 00:48:36 -05:00
dataSource.cellConfigurationHandler = { [weak self] cell, loggedError, _ in
guard let self else { return }
2023-03-01 00:48:36 -05:00
let cell = cell as! ErrorLogTableViewCell
cell.dateLabel.text = self.timeFormatter.string(from: loggedError.date)
cell.errorFailureLabel.text = loggedError.localizedFailure ?? NSLocalizedString("Operation Failed", comment: "")
2023-03-01 00:48:36 -05:00
switch loggedError.domain {
case AltServerErrorDomain: cell.errorCodeLabel?.text = String(format: NSLocalizedString("AltServer Error %@", comment: ""), NSNumber(value: loggedError.code))
case OperationError.domain: cell.errorCodeLabel?.text = String(format: NSLocalizedString("AltStore Error %@", comment: ""), NSNumber(value: loggedError.code))
default: cell.errorCodeLabel?.text = loggedError.error.localizedErrorCode
}
2023-03-01 00:48:36 -05:00
let nsError = loggedError.error as NSError
let errorDescription = [nsError.localizedDescription, nsError.localizedRecoverySuggestion].compactMap { $0 }.joined(separator: "\n\n")
cell.errorDescriptionTextView.text = errorDescription
cell.errorDescriptionTextView.maximumNumberOfLines = 5
cell.errorDescriptionTextView.isCollapsed = !self.expandedErrorIDs.contains(loggedError.objectID)
cell.errorDescriptionTextView.moreButton.addTarget(self, action: #selector(ErrorLogViewController.toggleCollapsingCell(_:)), for: .primaryActionTriggered)
2023-03-01 00:48:36 -05:00
cell.appIconImageView.image = nil
cell.appIconImageView.isIndicatingActivity = true
cell.appIconImageView.layer.borderColor = UIColor.gray.cgColor
2023-03-01 00:48:36 -05:00
let displayScale = (self.traitCollection.displayScale == 0.0) ? 1.0 : self.traitCollection.displayScale // 0.0 == "unspecified"
cell.appIconImageView.layer.borderWidth = 1.0 / displayScale
2023-03-01 00:48:36 -05:00
if #available(iOS 14, *) {
let menu = UIMenu(title: "", children: [
UIAction(title: NSLocalizedString("Copy Error Message", comment: ""), image: UIImage(systemName: "doc.on.doc")) { [weak self] _ in
self?.copyErrorMessage(for: loggedError)
},
UIAction(title: NSLocalizedString("Copy Error Code", comment: ""), image: UIImage(systemName: "doc.on.doc")) { [weak self] _ in
self?.copyErrorCode(for: loggedError)
},
UIAction(title: NSLocalizedString("Search FAQ", comment: ""), image: UIImage(systemName: "magnifyingglass")) { [weak self] _ in
self?.searchFAQ(for: loggedError)
}
])
cell.menuButton.menu = menu
}
2023-03-01 00:48:36 -05:00
// Include errorDescriptionTextView's text in cell summary.
cell.accessibilityLabel = [cell.errorFailureLabel.text, cell.dateLabel.text, cell.errorCodeLabel.text, cell.errorDescriptionTextView.text].compactMap { $0 }.joined(separator: ". ")
2023-03-01 00:48:36 -05:00
// Group all paragraphs together into single accessibility element (otherwise, each paragraph is independently selectable).
cell.errorDescriptionTextView.accessibilityLabel = cell.errorDescriptionTextView.text
}
2023-03-01 00:48:36 -05:00
dataSource.prefetchHandler = { loggedError, _, completion in
RSTAsyncBlockOperation { operation in
loggedError.managedObjectContext?.perform {
2023-03-01 00:48:36 -05:00
if let installedApp = loggedError.installedApp {
installedApp.loadIcon { result in
switch result {
case let .failure(error): completion(nil, error)
case let .success(image): completion(image, nil)
}
}
2023-03-01 00:48:36 -05:00
} else if let storeApp = loggedError.storeApp {
ImagePipeline.shared.loadImage(with: storeApp.iconURL, progress: nil) { response, error in
guard !operation.isCancelled else { return operation.finish() }
2023-03-01 00:48:36 -05:00
if let image = response?.image {
completion(image, nil)
2023-03-01 00:48:36 -05:00
} else {
completion(nil, error)
}
}
2023-03-01 00:48:36 -05:00
} else {
completion(nil, nil)
}
}
}
}
2023-03-01 00:48:36 -05:00
dataSource.prefetchCompletionHandler = { cell, image, _, _ in
let cell = cell as! ErrorLogTableViewCell
cell.appIconImageView.image = image
cell.appIconImageView.isIndicatingActivity = false
}
2023-03-01 00:48:36 -05:00
let placeholderView = RSTPlaceholderView()
placeholderView.textLabel.text = NSLocalizedString("No Errors", comment: "")
placeholderView.detailTextLabel.text = NSLocalizedString("Errors that occur when sideloading or refreshing apps will appear here.", comment: "")
dataSource.placeholderView = placeholderView
2023-03-01 00:48:36 -05:00
return dataSource
}
}
2023-03-01 00:48:36 -05:00
private extension ErrorLogViewController {
@IBAction func toggleCollapsingCell(_ sender: UIButton) {
let point = tableView.convert(sender.center, from: sender.superview)
guard let indexPath = tableView.indexPathForRow(at: point), let cell = tableView.cellForRow(at: indexPath) as? ErrorLogTableViewCell else { return }
let loggedError = dataSource.item(at: indexPath)
if cell.errorDescriptionTextView.isCollapsed {
expandedErrorIDs.remove(loggedError.objectID)
} else {
expandedErrorIDs.insert(loggedError.objectID)
}
2023-03-01 00:48:36 -05:00
tableView.performBatchUpdates {
cell.layoutIfNeeded()
}
}
2023-03-01 00:48:36 -05:00
@IBAction func showMinimuxerLogs(_: UIBarButtonItem) {
// Show minimuxer.log
let previewController = QLPreviewController()
previewController.dataSource = self
let navigationController = UINavigationController(rootViewController: previewController)
present(navigationController, animated: true, completion: nil)
}
2023-03-01 00:48:36 -05:00
@IBAction func clearLoggedErrors(_ sender: UIBarButtonItem) {
let alertController = UIAlertController(title: NSLocalizedString("Are you sure you want to clear the error log?", comment: ""), message: nil, preferredStyle: .actionSheet)
alertController.popoverPresentationController?.barButtonItem = sender
alertController.addAction(.cancel)
alertController.addAction(UIAlertAction(title: NSLocalizedString("Clear Error Log", comment: ""), style: .destructive) { _ in
self.clearLoggedErrors()
})
2023-03-01 00:48:36 -05:00
present(alertController, animated: true)
}
2023-03-01 00:48:36 -05:00
func clearLoggedErrors() {
DatabaseManager.shared.purgeLoggedErrors { result in
2023-03-01 00:48:36 -05:00
do {
try result.get()
2023-03-01 00:48:36 -05:00
} catch {
DispatchQueue.main.async {
let alertController = UIAlertController(title: NSLocalizedString("Failed to Clear Error Log", comment: ""), message: error.localizedDescription, preferredStyle: .alert)
alertController.addAction(.ok)
self.present(alertController, animated: true)
}
}
}
}
2023-03-01 00:48:36 -05:00
func copyErrorMessage(for loggedError: LoggedError) {
let nsError = loggedError.error as NSError
let errorMessage = [nsError.localizedDescription, nsError.localizedRecoverySuggestion].compactMap { $0 }.joined(separator: "\n\n")
2023-03-01 00:48:36 -05:00
UIPasteboard.general.string = errorMessage
}
2023-03-01 00:48:36 -05:00
func copyErrorCode(for loggedError: LoggedError) {
let errorCode = loggedError.error.localizedErrorCode
UIPasteboard.general.string = errorCode
}
2023-03-01 00:48:36 -05:00
func searchFAQ(for loggedError: LoggedError) {
let baseURL = URL(string: "https://faq.altstore.io/getting-started/troubleshooting-guide")!
var components = URLComponents(url: baseURL, resolvingAgainstBaseURL: false)!
2023-03-01 00:48:36 -05:00
let query = [loggedError.domain, "\(loggedError.code)"].joined(separator: "+")
components.queryItems = [URLQueryItem(name: "q", value: query)]
2023-03-01 00:48:36 -05:00
let safariViewController = SFSafariViewController(url: components.url ?? baseURL)
safariViewController.preferredControlTintColor = .altPrimary
2023-03-01 00:48:36 -05:00
present(safariViewController, animated: true)
}
}
2023-03-01 00:48:36 -05:00
extension ErrorLogViewController {
override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
let loggedError = dataSource.item(at: indexPath)
let alertController = UIAlertController(title: nil, message: nil, preferredStyle: .actionSheet)
alertController.addAction(UIAlertAction(title: UIAlertAction.cancel.title, style: UIAlertAction.cancel.style) { _ in
tableView.deselectRow(at: indexPath, animated: true)
})
alertController.addAction(UIAlertAction(title: NSLocalizedString("Copy Error Message", comment: ""), style: .default) { [weak self] _ in
self?.copyErrorMessage(for: loggedError)
tableView.deselectRow(at: indexPath, animated: true)
})
alertController.addAction(UIAlertAction(title: NSLocalizedString("Copy Error Code", comment: ""), style: .default) { [weak self] _ in
self?.copyErrorCode(for: loggedError)
tableView.deselectRow(at: indexPath, animated: true)
})
alertController.addAction(UIAlertAction(title: NSLocalizedString("Search FAQ", comment: ""), style: .default) { [weak self] _ in
self?.searchFAQ(for: loggedError)
tableView.deselectRow(at: indexPath, animated: true)
})
2023-03-01 00:48:36 -05:00
present(alertController, animated: true)
}
2023-03-01 00:48:36 -05:00
override func tableView(_: UITableView, trailingSwipeActionsConfigurationForRowAt indexPath: IndexPath) -> UISwipeActionsConfiguration? {
let deleteAction = UIContextualAction(style: .destructive, title: NSLocalizedString("Delete", comment: "")) { _, _, completion in
let loggedError = self.dataSource.item(at: indexPath)
DatabaseManager.shared.persistentContainer.performBackgroundTask { context in
2023-03-01 00:48:36 -05:00
do {
let loggedError = context.object(with: loggedError.objectID) as! LoggedError
context.delete(loggedError)
2023-03-01 00:48:36 -05:00
try context.save()
completion(true)
2023-03-01 00:48:36 -05:00
} catch {
print("[ALTLog] Failed to delete LoggedError \(loggedError.objectID):", error)
completion(false)
}
}
}
2023-03-01 00:48:36 -05:00
let configuration = UISwipeActionsConfiguration(actions: [deleteAction])
configuration.performsFirstActionWithFullSwipe = false
return configuration
}
2023-03-01 00:48:36 -05:00
override func tableView(_: UITableView, titleForHeaderInSection section: Int) -> String? {
let indexPath = IndexPath(row: 0, section: section)
2023-03-01 00:48:36 -05:00
let loggedError = dataSource.item(at: indexPath)
if Calendar.current.isDateInToday(loggedError.date) {
return NSLocalizedString("Today", comment: "")
2023-03-01 00:48:36 -05:00
} else {
return loggedError.localizedDateString
}
}
}
extension ErrorLogViewController: QLPreviewControllerDataSource {
2023-03-01 00:48:36 -05:00
func numberOfPreviewItems(in _: QLPreviewController) -> Int {
1
}
2023-03-01 00:48:36 -05:00
func previewController(_: QLPreviewController, previewItemAt _: Int) -> QLPreviewItem {
let fileURL = FileManager.default.documentsDirectory.appendingPathComponent("minimuxer.log")
return fileURL as QLPreviewItem
}
}