mirror of
https://github.com/SideStore/SideStore.git
synced 2026-02-09 06:43:25 +01:00
* [Shared] Revises ALTLocalizedError protocol * Refactors errors to conform to revised ALTLocalizedError protocol * [Missing Commit] Remaining changes for ALTLocalizedError * [AltServer] Refactors errors to conform to revised ALTLocalizedError protocol * [Missing Commit] Declares ALTLocalizedTitleErrorKey + ALTLocalizedDescriptionKey * Updates Objective-C errors to match revised ALTLocalizedError * [Missing Commit] Unnecessary ALTLocalizedDescription logic * [Shared] Refactors NSError.withLocalizedFailure to properly support ALTLocalizedError * [Shared] Supports adding localized titles to errors via NSError.withLocalizedTitle() * Revises ErrorResponse logic to support arbitrary errors and user info values * [Missed Commit] Renames CodableServerError to CodableError * Merges ConnectionError into OperationError * [Missed Commit] Doesn’t check ALTWrappedError’s userInfo for localizedDescription * [Missed] Fixes incorrect errorDomain for ALTErrorEnums * [Missed] Removes nonexistent ALTWrappedError.h * Includes source file and line number in OperationError.unknown failureReason * Adds localizedTitle to AppManager operation errors * Fixes adding localizedTitle + localizedFailure to ALTWrappedError * Updates ToastView to use error’s localizedTitle as title * [Shared] Adds NSError.formattedDetailedDescription(with:) Returns formatted NSAttributedString containing all user info values intended for displaying to the user. * [Shared] Updates Error.localizedErrorCode to say “code” instead of “error” * Conforms ALTLocalizedError to CustomStringConvertible * Adds “View More Details” option to Error Log context menu to view detailed error description * [Shared] Fixes NSError.formattedDetailedDescription appearing black in dark mode * [AltServer] Updates error alert to match revised error logic Uses error’s localizedTitle as alert title. * [AltServer] Adds “View More Details” button to error alert to view detailed error info * [AltServer] Renames InstallError to OperationError and conforms to ALTErrorEnum * [Shared] Removes CodableError support for Date user info values Not currently used, and we don’t want to accidentally parse a non-Date as a Date in the meantime. * [Shared] Includes dynamic UserInfoValueProvider values in NSError.formattedDetailedDescription() * [Shared] Includes source file + line in NSError.formattedDetailedDescription() Automatically captures source file + line when throwing ALTErrorEnums. * [Shared] Captures source file + line for unknown errors * Removes sourceFunction from OperationError * Adds localizedTitle to AuthenticationViewController errors * [Shared] Moves nested ALTWrappedError logic to ALTWrappedError initializer * [AltServer] Removes now-redundant localized failure from JIT errors All JIT errors now have a localizedTitle which effectively says the same thing. * Makes OperationError.Code start at 1000 “Connection errors” subsection starts at 1200. * [Shared] Updates Error domains to revised [Source].[ErrorType] format * Updates ALTWrappedError.localizedDescription to prioritize using wrapped NSLocalizedDescription as failure reason * Makes ALTAppleAPIError codes start at 3000 * [AltServer] Adds relevant localizedFailures to ALTDeviceManager.installApplication() errors * Revises OperationError failureReasons and recovery suggestions All failure reasons now read correctly when preceded by a failure reason and “because”. * Revises ALTServerError error messages All failure reasons now read correctly when preceded by a failure reason and “because”. * Most failure reasons now read correctly when preceded by a failure reason and “because”. * ALTServerErrorUnderlyingError forwards all user info provider calls to underlying error. * Revises error messages for ALTAppleAPIErrorIncorrectCredentials * [Missed] Removes NSError+AltStore.swift from AltStore target * [Shared] Updates AltServerErrorDomain to revised [Source].[ErrorType] format * [Shared] Removes “code” from Error.localizedErrorCode * [Shared] Makes ALTServerError codes (appear to) start at 2000 We can’t change the actual error codes without breaking backwards compatibility, so instead we just add 2000 whenever we display ALTServerError codes to the user. * Moves VerificationError.errorFailure to VerifyAppOperation * Supports custom failure reason for OperationError.unknown * [Shared] Changes AltServerErrorDomain to “AltServer.ServerError” * [Shared] Converts ALTWrappedError to Objective-C class NSError subclasses must be written in ObjC for Swift.Error <-> NSError bridging to work correctly. * Fixes decoding CodableError nested user info values
380 lines
16 KiB
Swift
380 lines
16 KiB
Swift
//
|
|
// ErrorLogViewController.swift
|
|
// AltStore
|
|
//
|
|
// Created by Riley Testut on 9/6/22.
|
|
// Copyright © 2022 Riley Testut. All rights reserved.
|
|
//
|
|
|
|
import UIKit
|
|
import SafariServices
|
|
|
|
import AltStoreCore
|
|
import Roxas
|
|
|
|
import Nuke
|
|
|
|
import QuickLook
|
|
|
|
final class ErrorLogViewController: UITableViewController
|
|
{
|
|
private lazy var dataSource = self.makeDataSource()
|
|
private var expandedErrorIDs = Set<NSManagedObjectID>()
|
|
|
|
private var isScrolling = false {
|
|
didSet {
|
|
guard self.isScrolling != oldValue else { return }
|
|
self.updateButtonInteractivity()
|
|
}
|
|
}
|
|
|
|
private lazy var timeFormatter: DateFormatter = {
|
|
let dateFormatter = DateFormatter()
|
|
dateFormatter.dateStyle = .none
|
|
dateFormatter.timeStyle = .short
|
|
return dateFormatter
|
|
}()
|
|
|
|
override var preferredStatusBarStyle: UIStatusBarStyle {
|
|
return .lightContent
|
|
}
|
|
|
|
override func viewDidLoad()
|
|
{
|
|
super.viewDidLoad()
|
|
|
|
self.tableView.dataSource = self.dataSource
|
|
self.tableView.prefetchDataSource = self.dataSource
|
|
}
|
|
|
|
override func prepare(for segue: UIStoryboardSegue, sender: Any?)
|
|
{
|
|
guard let loggedError = sender as? LoggedError, segue.identifier == "showErrorDetails" else { return }
|
|
|
|
let navigationController = segue.destination as! UINavigationController
|
|
|
|
let errorDetailsViewController = navigationController.viewControllers.first as! ErrorDetailsViewController
|
|
errorDetailsViewController.loggedError = loggedError
|
|
}
|
|
|
|
@IBAction private func unwindFromErrorDetails(_ segue: UIStoryboardSegue)
|
|
{
|
|
}
|
|
}
|
|
|
|
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
|
|
|
|
let fetchedResultsController = NSFetchedResultsController(fetchRequest: fetchRequest, managedObjectContext: DatabaseManager.shared.viewContext, sectionNameKeyPath: #keyPath(LoggedError.localizedDateString), cacheName: nil)
|
|
|
|
let dataSource = RSTFetchedResultsTableViewPrefetchingDataSource<LoggedError, UIImage>(fetchedResultsController: fetchedResultsController)
|
|
dataSource.proxy = self
|
|
dataSource.rowAnimation = .fade
|
|
dataSource.cellConfigurationHandler = { [weak self] (cell, loggedError, indexPath) in
|
|
guard let self else { return }
|
|
|
|
let cell = cell as! ErrorLogTableViewCell
|
|
cell.dateLabel.text = self.timeFormatter.string(from: loggedError.date)
|
|
cell.errorFailureLabel.text = loggedError.localizedFailure ?? NSLocalizedString("Operation Failed", comment: "")
|
|
cell.errorCodeLabel.text = loggedError.error.localizedErrorCode
|
|
|
|
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)
|
|
|
|
cell.appIconImageView.image = nil
|
|
cell.appIconImageView.isIndicatingActivity = true
|
|
cell.appIconImageView.layer.borderColor = UIColor.gray.cgColor
|
|
|
|
let displayScale = (self.traitCollection.displayScale == 0.0) ? 1.0 : self.traitCollection.displayScale // 0.0 == "unspecified"
|
|
cell.appIconImageView.layer.borderWidth = 1.0 / displayScale
|
|
|
|
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)
|
|
},
|
|
UIAction(title: NSLocalizedString("View More Details", comment: ""), image: UIImage(systemName: "ellipsis.circle")) { [weak self] _ in
|
|
self?.viewMoreDetails(for: loggedError)
|
|
},
|
|
])
|
|
|
|
cell.menuButton.menu = menu
|
|
cell.menuButton.showsMenuAsPrimaryAction = self.isScrolling ? false : true
|
|
cell.selectionStyle = .none
|
|
} else {
|
|
cell.menuButton.isUserInteractionEnabled = false
|
|
}
|
|
|
|
// 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: ". ")
|
|
|
|
// Group all paragraphs together into single accessibility element (otherwise, each paragraph is independently selectable).
|
|
cell.errorDescriptionTextView.accessibilityLabel = cell.errorDescriptionTextView.text
|
|
}
|
|
dataSource.prefetchHandler = { (loggedError, indexPath, completion) in
|
|
RSTAsyncBlockOperation { (operation) in
|
|
loggedError.managedObjectContext?.perform {
|
|
if let installedApp = loggedError.installedApp
|
|
{
|
|
installedApp.loadIcon { (result) in
|
|
switch result
|
|
{
|
|
case .failure(let error): completion(nil, error)
|
|
case .success(let image): completion(image, nil)
|
|
}
|
|
}
|
|
}
|
|
else if let storeApp = loggedError.storeApp
|
|
{
|
|
ImagePipeline.shared.loadImage(with: storeApp.iconURL, progress: nil) { (response, error) in
|
|
guard !operation.isCancelled else { return operation.finish() }
|
|
|
|
if let image = response?.image
|
|
{
|
|
completion(image, nil)
|
|
}
|
|
else
|
|
{
|
|
completion(nil, error)
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
completion(nil, nil)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
dataSource.prefetchCompletionHandler = { (cell, image, indexPath, error) in
|
|
let cell = cell as! ErrorLogTableViewCell
|
|
cell.appIconImageView.image = image
|
|
cell.appIconImageView.isIndicatingActivity = false
|
|
}
|
|
|
|
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
|
|
|
|
return dataSource
|
|
}
|
|
}
|
|
|
|
private extension ErrorLogViewController
|
|
{
|
|
@IBAction func toggleCollapsingCell(_ sender: UIButton)
|
|
{
|
|
let point = self.tableView.convert(sender.center, from: sender.superview)
|
|
guard let indexPath = self.tableView.indexPathForRow(at: point), let cell = self.tableView.cellForRow(at: indexPath) as? ErrorLogTableViewCell else { return }
|
|
|
|
let loggedError = self.dataSource.item(at: indexPath)
|
|
|
|
if cell.errorDescriptionTextView.isCollapsed
|
|
{
|
|
self.expandedErrorIDs.remove(loggedError.objectID)
|
|
}
|
|
else
|
|
{
|
|
self.expandedErrorIDs.insert(loggedError.objectID)
|
|
}
|
|
|
|
self.tableView.performBatchUpdates {
|
|
cell.layoutIfNeeded()
|
|
}
|
|
}
|
|
|
|
@IBAction func showMinimuxerLogs(_ sender: UIBarButtonItem)
|
|
{
|
|
// Show minimuxer.log
|
|
let previewController = QLPreviewController()
|
|
previewController.dataSource = self
|
|
let navigationController = UINavigationController(rootViewController: previewController)
|
|
present(navigationController, animated: true, completion: nil)
|
|
}
|
|
|
|
@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()
|
|
})
|
|
self.present(alertController, animated: true)
|
|
}
|
|
|
|
func clearLoggedErrors()
|
|
{
|
|
DatabaseManager.shared.purgeLoggedErrors { result in
|
|
do
|
|
{
|
|
try result.get()
|
|
}
|
|
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)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
func copyErrorMessage(for loggedError: LoggedError)
|
|
{
|
|
let nsError = loggedError.error as NSError
|
|
let errorMessage = [nsError.localizedDescription, nsError.localizedRecoverySuggestion].compactMap { $0 }.joined(separator: "\n\n")
|
|
|
|
UIPasteboard.general.string = errorMessage
|
|
}
|
|
|
|
func copyErrorCode(for loggedError: LoggedError)
|
|
{
|
|
let errorCode = loggedError.error.localizedErrorCode
|
|
UIPasteboard.general.string = errorCode
|
|
}
|
|
|
|
func searchFAQ(for loggedError: LoggedError)
|
|
{
|
|
let baseURL = URL(string: "https://faq.altstore.io/getting-started/troubleshooting-guide")!
|
|
var components = URLComponents(url: baseURL, resolvingAgainstBaseURL: false)!
|
|
|
|
let query = [loggedError.domain, "\(loggedError.error.displayCode)"].joined(separator: "+")
|
|
components.queryItems = [URLQueryItem(name: "q", value: query)]
|
|
|
|
let safariViewController = SFSafariViewController(url: components.url ?? baseURL)
|
|
safariViewController.preferredControlTintColor = .altPrimary
|
|
self.present(safariViewController, animated: true)
|
|
}
|
|
|
|
func viewMoreDetails(for loggedError: LoggedError) {
|
|
self.performSegue(withIdentifier: "showErrorDetails", sender: loggedError)
|
|
}
|
|
}
|
|
|
|
extension ErrorLogViewController
|
|
{
|
|
override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath)
|
|
{
|
|
guard #unavailable(iOS 14) else { return }
|
|
let loggedError = self.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)
|
|
})
|
|
self.present(alertController, animated: true)
|
|
}
|
|
|
|
override func tableView(_ 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
|
|
do
|
|
{
|
|
let loggedError = context.object(with: loggedError.objectID) as! LoggedError
|
|
context.delete(loggedError)
|
|
|
|
try context.save()
|
|
completion(true)
|
|
}
|
|
catch
|
|
{
|
|
print("[ALTLog] Failed to delete LoggedError \(loggedError.objectID):", error)
|
|
completion(false)
|
|
}
|
|
}
|
|
}
|
|
|
|
let configuration = UISwipeActionsConfiguration(actions: [deleteAction])
|
|
configuration.performsFirstActionWithFullSwipe = false
|
|
return configuration
|
|
}
|
|
|
|
override func tableView(_ tableView: UITableView, titleForHeaderInSection section: Int) -> String?
|
|
{
|
|
let indexPath = IndexPath(row: 0, section: section)
|
|
let loggedError = self.dataSource.item(at: indexPath)
|
|
|
|
if Calendar.current.isDateInToday(loggedError.date)
|
|
{
|
|
return NSLocalizedString("Today", comment: "")
|
|
}
|
|
else
|
|
{
|
|
return loggedError.localizedDateString
|
|
}
|
|
}
|
|
}
|
|
|
|
extension ErrorLogViewController: QLPreviewControllerDataSource {
|
|
func numberOfPreviewItems(in controller: QLPreviewController) -> Int {
|
|
return 1
|
|
}
|
|
|
|
func previewController(_ controller: QLPreviewController, previewItemAt index: Int) -> QLPreviewItem {
|
|
let fileURL = FileManager.default.documentsDirectory.appendingPathComponent("minimuxer.log")
|
|
return fileURL as QLPreviewItem
|
|
}
|
|
}
|
|
|
|
extension ErrorLogViewController
|
|
{
|
|
override func scrollViewWillBeginDragging(_ scrollView: UIScrollView)
|
|
{
|
|
self.isScrolling = true
|
|
}
|
|
|
|
override func scrollViewDidEndDecelerating(_ scrollView: UIScrollView)
|
|
{
|
|
self.isScrolling = false
|
|
}
|
|
|
|
override func scrollViewDidEndDragging(_ scrollView: UIScrollView, willDecelerate decelerate: Bool)
|
|
{
|
|
guard !decelerate else { return }
|
|
self.isScrolling = false
|
|
}
|
|
|
|
private func updateButtonInteractivity()
|
|
{
|
|
guard #available(iOS 14, *) else { return }
|
|
|
|
for case let cell as ErrorLogTableViewCell in self.tableView.visibleCells
|
|
{
|
|
cell.menuButton.showsMenuAsPrimaryAction = self.isScrolling ? false : true
|
|
}
|
|
}
|
|
}
|