mirror of https://github.com/oxen-io/session-ios
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
140 lines
5.8 KiB
Swift
140 lines
5.8 KiB
Swift
// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
|
|
//
|
|
// stringlint:disable
|
|
|
|
import Foundation
|
|
import SessionUtilitiesKit
|
|
import SignalUtilitiesKit
|
|
import SignalCoreKit
|
|
import SessionMessagingKit
|
|
|
|
public class AppEnvironment {
|
|
|
|
enum ExtensionType {
|
|
case share
|
|
case notification
|
|
|
|
var name: String {
|
|
switch self {
|
|
case .share: return "ShareExtension"
|
|
case .notification: return "NotificationExtension"
|
|
}
|
|
}
|
|
}
|
|
|
|
private static var _shared: AppEnvironment = AppEnvironment()
|
|
|
|
public class var shared: AppEnvironment {
|
|
get { return _shared }
|
|
set {
|
|
guard SNUtilitiesKit.isRunningTests else {
|
|
owsFailDebug("Can only switch environments in tests.")
|
|
return
|
|
}
|
|
|
|
_shared = newValue
|
|
}
|
|
}
|
|
|
|
public var callManager: SessionCallManager
|
|
public var notificationPresenter: NotificationPresenter
|
|
public var pushRegistrationManager: PushRegistrationManager
|
|
public var fileLogger: DDFileLogger
|
|
|
|
// Stored properties cannot be marked as `@available`, only classes and functions.
|
|
// Instead, store a private `Any` and wrap it with a public `@available` getter
|
|
private var _userNotificationActionHandler: Any?
|
|
|
|
public var userNotificationActionHandler: UserNotificationActionHandler {
|
|
return _userNotificationActionHandler as! UserNotificationActionHandler
|
|
}
|
|
|
|
private init() {
|
|
self.callManager = SessionCallManager()
|
|
self.notificationPresenter = NotificationPresenter()
|
|
self.pushRegistrationManager = PushRegistrationManager()
|
|
self._userNotificationActionHandler = UserNotificationActionHandler()
|
|
self.fileLogger = DDFileLogger()
|
|
|
|
SwiftSingletons.register(self)
|
|
}
|
|
|
|
public func setup() {
|
|
// Hang certain singletons on Environment too.
|
|
SessionEnvironment.shared?.callManager.mutate {
|
|
$0 = callManager
|
|
}
|
|
SessionEnvironment.shared?.notificationsManager.mutate {
|
|
$0 = notificationPresenter
|
|
}
|
|
setupLogFiles()
|
|
}
|
|
|
|
private func setupLogFiles() {
|
|
fileLogger.rollingFrequency = kDayInterval // Refresh everyday
|
|
fileLogger.logFileManager.maximumNumberOfLogFiles = 3 // Save 3 days' log files
|
|
DDLog.add(fileLogger)
|
|
|
|
// The extensions write their logs to the app shared directory but the main app writes
|
|
// to a local directory (so they can be exported via XCode) - the below code reads any
|
|
// logs from the shared directly and attempts to add them to the main app logs to make
|
|
// debugging user issues in extensions easier
|
|
DispatchQueue.global(qos: .background).async { [fileLogger] in
|
|
let extensionInfo: [(dir: String, type: ExtensionType)] = [
|
|
("\(OWSFileSystem.appSharedDataDirectoryPath())/Logs/NotificationExtension", .notification),
|
|
("\(OWSFileSystem.appSharedDataDirectoryPath())/Logs/ShareExtension", .share)
|
|
]
|
|
let extensionLogs: [(path: String, type: ExtensionType)] = extensionInfo.flatMap { dir, type -> [(path: String, type: ExtensionType)] in
|
|
guard let files: [String] = try? FileManager.default.contentsOfDirectory(atPath: dir) else { return [] }
|
|
|
|
return files.map { ("\(dir)/\($0)", type) }
|
|
}
|
|
// Log to ensure the log file exists
|
|
OWSLogger.info("")
|
|
DDLog.flushLog()
|
|
|
|
do {
|
|
guard
|
|
let currentLogFileInfo: DDLogFileInfo = fileLogger.currentLogFileInfo,
|
|
let fileHandle: FileHandle = FileHandle(forWritingAtPath: currentLogFileInfo.filePath)
|
|
else { throw StorageError.objectNotFound }
|
|
|
|
// Ensure we close the file handle
|
|
defer { fileHandle.closeFile() }
|
|
|
|
// Move to the end of the file to insert the logs
|
|
if #available(iOS 13.4, *) { try fileHandle.seekToEnd() }
|
|
else { fileHandle.seekToEndOfFile() }
|
|
|
|
try extensionLogs
|
|
.grouped(by: \.type)
|
|
.forEach { type, value in
|
|
guard
|
|
let typeNameStartData: Data = "🧩 \(type.name) -- Start\n".data(using: .utf8),
|
|
let typeNameEndData: Data = "🧩 \(type.name) -- End\n".data(using: .utf8)
|
|
else { throw StorageError.invalidData }
|
|
|
|
// Write the type start separator
|
|
if #available(iOS 13.4, *) { try fileHandle.write(contentsOf: typeNameStartData) }
|
|
else { fileHandle.write(typeNameStartData) }
|
|
|
|
// Write the logs
|
|
try value.forEach { path, _ in
|
|
let logData: Data = try Data(contentsOf: URL(fileURLWithPath: path))
|
|
if #available(iOS 13.4, *) { try fileHandle.write(contentsOf: logData) }
|
|
else { fileHandle.write(logData) }
|
|
|
|
// Extension logs have been writen to the app logs, remove them now
|
|
try? FileManager.default.removeItem(atPath: path)
|
|
}
|
|
|
|
// Write the type end separator
|
|
if #available(iOS 13.4, *) { try fileHandle.write(contentsOf: typeNameEndData) }
|
|
else { fileHandle.write(typeNameEndData) }
|
|
}
|
|
}
|
|
catch { SNLog("Unable to write extension logs to current log file") }
|
|
}
|
|
}
|
|
}
|