diff --git a/OpenHABCore/Sources/OpenHABCore/Util/Logger.swift b/OpenHABCore/Sources/OpenHABCore/Util/Logger.swift new file mode 100644 index 00000000..3eea80b5 --- /dev/null +++ b/OpenHABCore/Sources/OpenHABCore/Util/Logger.swift @@ -0,0 +1,216 @@ +// Copyright (c) 2010-2024 Contributors to the openHAB project +// +// See the NOTICE file(s) distributed with this work for additional +// information. +// +// This program and the accompanying materials are made available under the +// terms of the Eclipse Public License 2.0 which is available at +// http://www.eclipse.org/legal/epl-2.0 +// +// SPDX-License-Identifier: EPL-2.0 + +import CoreTransferable +import OSLog + +// Thanks to https://useyourloaf.com/blog/fetching-oslog-messages-in-swift/ + +// swiftlint:disable:next file_types_order +private extension OSLogEntryLog.Level { + var description: String { + switch self { + case .undefined: "undefined" + case .debug: "debug" + case .info: "info" + case .notice: "notice" + case .error: "error" + case .fault: "fault" + @unknown default: "default" + } + } +} + +public extension Logger { + static func fetch(since date: Date, + predicateFormat: String) async throws -> [String] { + let store = try OSLogStore(scope: .currentProcessIdentifier) + let position = store.position(date: date) + let predicate = NSPredicate(format: predicateFormat) + + let entries = try store + .getEntries( + at: position, + matching: predicate + ) + + let dateFormatter = DateFormatter() + dateFormatter.dateFormat = "yyyy-MM-dd HH:mm:ss" + + var logs: [String] = [] + for entry in entries { + try Task.checkCancellation() + if let log = entry as? OSLogEntryLog { + var attributedMessage = AttributedString(dateFormatter.string(from: entry.date)) + attributedMessage.font = .headline + + logs.append(""" + \(entry.date.formatted(.iso8601)): \ + \(log.category):\(log.level.description): \ + \(entry.composedMessage)\n + """) + } else { + logs.append("\(entry.date): \(entry.composedMessage)\n") + } + } + + if logs.isEmpty { logs = ["Nothing found"] } + return logs + } +} + +public protocol LogServiceProtocol { + func fetchLogs(with template: NSPredicate) async -> String +} + +public class LogService { + static let shared = Logger() + private var fileHandle: FileHandle! + + // Return the folder URL, and create the folder if it doesn't exist yet. + // Return nil to trigger a crash if the folder creation fails. + // + private var _folderURL: URL? + private var folderURL: URL! { + guard _folderURL == nil else { return _folderURL } + + var folderURL = FileManager.default.urls(for: .cachesDirectory, in: .userDomainMask).last! + folderURL.appendPathComponent("Logs") + + if !FileManager.default.fileExists(atPath: folderURL.path) { + do { + try FileManager.default.createDirectory(at: folderURL, withIntermediateDirectories: true) + } catch { + print("Failed to create the log folder: \(folderURL)! \n\(error)") + return nil // To trigger crash. + } + } + _folderURL = folderURL + return folderURL + } + + // Return the file URL, and create the file if it doesn't exist yet. + // Return nil to trigger a crash if the file creation fails. + // + private var _fileURL: URL? + private var fileURL: URL! { + guard _fileURL == nil else { return _fileURL } + + let dateFormatter = DateFormatter() + dateFormatter.dateStyle = .medium + let dateString = dateFormatter.string(from: Date()) + + var fileURL: URL = folderURL + fileURL.appendPathComponent("\(dateString).log") + + if !FileManager.default.fileExists(atPath: fileURL.path) { + if !FileManager.default.createFile(atPath: fileURL.path, contents: nil, attributes: nil) { + print("Failed to create the log file: \(fileURL)!") + return nil // To trigger crash. + } + } + _fileURL = fileURL + return fileURL + } + + // Use this dispatch queue to make the log file access thread-safe. + // Public methods use performBlockAndWait to access the resource; private methods don't. + // + private lazy var ioQueue: DispatchQueue = .init(label: "ioQueue") + + public init() { + fileHandle = try? FileHandle(forUpdating: fileURL) + assert(fileHandle != nil, "Failed to create the file handle!") + } + + private func performBlockAndWait(_ block: () -> T) -> T { + ioQueue.sync { + block() + } + } + + // Get the current log file URL. + // + func getFileURL() -> URL { + performBlockAndWait { fileURL } + } + + func writeLogs() async { + let template = NSPredicate( + format: "(subsystem BEGINSWITH $PREFIX)" + ) + + let logData = await fetchLogs(with: template) + + if let data = logData.data(using: .utf8) { + performBlockAndWait { + self.fileHandle.write(data) + } + } + } + + // Read the file content and return it as a string. + // + func content() -> String { + performBlockAndWait { + fileHandle.seek(toFileOffset: 0) // Read from the very beginning. + return String(data: fileHandle.availableData, encoding: .utf8) ?? "" + } + } + + // Clear all logs. Reset the folder and file URL for later use. + // + func clearLogs() { + performBlockAndWait { + self.fileHandle.closeFile() + do { + try FileManager.default.removeItem(at: self.folderURL) + } catch { + print("Failed to clear the log folder!\n\(error)") + } + + // Create a new file handle. + // + self._folderURL = nil + self._fileURL = nil + self.fileHandle = try? FileHandle(forUpdating: self.fileURL) + assert(self.fileHandle != nil, "Failed to create the file handle!") + } + } +} + +extension LogService: LogServiceProtocol { + public func fetchLogs(with template: NSPredicate) async -> String { + let calendar = Calendar.current + guard let hourAgo = calendar.date( + byAdding: .hour, + value: -1, + to: Date.now + ) else { + return "Invalid calendar" + } + + do { + let predicate = template.withSubstitutionVariables( + [ + "PREFIX": "org.openhab" + ]) + + let logs = try await Logger.fetch( + since: hourAgo, + predicateFormat: predicate.predicateFormat + ) + return logs.joined() + } catch { + return error.localizedDescription + } + } +} diff --git a/openHAB.xcodeproj/project.pbxproj b/openHAB.xcodeproj/project.pbxproj index 3fc9f5af..abcd9ca5 100644 --- a/openHAB.xcodeproj/project.pbxproj +++ b/openHAB.xcodeproj/project.pbxproj @@ -80,6 +80,7 @@ DA0F37D023D4ACC7007EAB48 /* SliderRow.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA0F37CF23D4ACC7007EAB48 /* SliderRow.swift */; }; DA15BFBD23C6726400BD8ADA /* ObservableOpenHABDataObject.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA15BFBC23C6726400BD8ADA /* ObservableOpenHABDataObject.swift */; }; DA162BEC2CD3B53E0040DAE5 /* LogsViewer.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA162BEB2CD3B53E0040DAE5 /* LogsViewer.swift */; }; + DA162BF02CD4CC730040DAE5 /* LogsViewer.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA162BEF2CD4CC730040DAE5 /* LogsViewer.swift */; }; DA19E25B22FD801D002F8F2F /* OpenHABGeneralTests.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA19E25A22FD801D002F8F2F /* OpenHABGeneralTests.swift */; }; DA21EAE22339621C001AB415 /* Throttler.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA21EAE12339621C001AB415 /* Throttler.swift */; }; DA242C622C83588600AFB10D /* SettingsView.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA242C612C83588600AFB10D /* SettingsView.swift */; }; @@ -106,17 +107,18 @@ DA6B2EF52C89F8F200DF77CF /* ColorPickerView.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA6B2EF42C89F8F200DF77CF /* ColorPickerView.swift */; }; DA6B2EF72C8B92E800DF77CF /* SelectionView.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA6B2EF62C8B92E800DF77CF /* SelectionView.swift */; }; DA7224D223828D3400712D20 /* PreviewConstants.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA7224D123828D3300712D20 /* PreviewConstants.swift */; }; - DA72E1B8236DEA0900B8EF3A /* AppMessageService.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA72E1B5236DEA0900B8EF3A /* AppMessageService.swift */; }; + DA72E1B8236DEA0900B8EF3A /* WatchConnectivitySessionManager.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA72E1B5236DEA0900B8EF3A /* WatchConnectivitySessionManager.swift */; }; DA7E1E4B2233986E002AEFD8 /* PlayerView.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA7E1E47222EB00B002AEFD8 /* PlayerView.swift */; }; DA817E7A234BF39B00C91824 /* CHANGELOG.md in Resources */ = {isa = PBXBuildFile; fileRef = DA817E79234BF39B00C91824 /* CHANGELOG.md */; }; DA88F8C622EC377200B408E5 /* ReleaseNotes.md in Resources */ = {isa = PBXBuildFile; fileRef = DA88F8C522EC377100B408E5 /* ReleaseNotes.md */; }; + DA8F986B2CDA4CAA001F5E8A /* ContentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA8F986A2CDA4CAA001F5E8A /* ContentView.swift */; }; DA9721C324E29A8F0092CCFD /* UserDefaultsBacked.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA9721C224E29A8F0092CCFD /* UserDefaultsBacked.swift */; }; DA9F81872C85020F00B47B72 /* RTFTextView.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA9F81862C85020F00B47B72 /* RTFTextView.swift */; }; DAA070932B5181210060BB0E /* OpenHABImageDownloaderOperation.swift in Sources */ = {isa = PBXBuildFile; fileRef = DAA070922B5181210060BB0E /* OpenHABImageDownloaderOperation.swift */; }; DAA42BA821DC97E000244B2A /* NotificationTableViewCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = DAA42BA721DC97DF00244B2A /* NotificationTableViewCell.swift */; }; DAA42BAA21DC983B00244B2A /* VideoUITableViewCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = DAA42BA921DC983B00244B2A /* VideoUITableViewCell.swift */; }; DAA42BAC21DC984A00244B2A /* WebUITableViewCell.swift in Sources */ = {isa = PBXBuildFile; fileRef = DAA42BAB21DC984A00244B2A /* WebUITableViewCell.swift */; }; - DAAC30872CBBF0420041927F /* ContentView.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA0775262346705F0086C685 /* ContentView.swift */; }; + DAAC30872CBBF0420041927F /* SitemapView.swift in Sources */ = {isa = PBXBuildFile; fileRef = DA0775262346705F0086C685 /* SitemapView.swift */; }; DAC65FC7236EDF3900F4501E /* SpinnerViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = DAC65FC6236EDF3900F4501E /* SpinnerViewController.swift */; }; DAC6608D236F771600F4501E /* PreferencesSwiftUIView.swift in Sources */ = {isa = PBXBuildFile; fileRef = DAC6608C236F771600F4501E /* PreferencesSwiftUIView.swift */; }; DAC9395522B00E7600C5F423 /* XCTestCaseExtension.swift in Sources */ = {isa = PBXBuildFile; fileRef = DAC9395422B00E7600C5F423 /* XCTestCaseExtension.swift */; }; @@ -336,7 +338,7 @@ DA0775152346705D0086C685 /* openHABWatch.app */ = {isa = PBXFileReference; explicitFileType = wrapper.application; includeInIndex = 0; path = openHABWatch.app; sourceTree = BUILT_PRODUCTS_DIR; }; DA07751A2346705F0086C685 /* Assets.xcassets */ = {isa = PBXFileReference; lastKnownFileType = folder.assetcatalog; path = Assets.xcassets; sourceTree = ""; }; DA07751C2346705F0086C685 /* Info.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = Info.plist; sourceTree = ""; }; - DA0775262346705F0086C685 /* ContentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ContentView.swift; sourceTree = ""; }; + DA0775262346705F0086C685 /* SitemapView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SitemapView.swift; sourceTree = ""; }; DA07752A2346705F0086C685 /* OpenHABWatchAppDelegate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OpenHABWatchAppDelegate.swift; sourceTree = ""; }; DA07752C2346705F0086C685 /* NotificationController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NotificationController.swift; sourceTree = ""; }; DA07752E2346705F0086C685 /* NotificationView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = NotificationView.swift; sourceTree = ""; }; @@ -348,6 +350,7 @@ DA0F37CF23D4ACC7007EAB48 /* SliderRow.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = SliderRow.swift; sourceTree = ""; }; DA15BFBC23C6726400BD8ADA /* ObservableOpenHABDataObject.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ObservableOpenHABDataObject.swift; sourceTree = ""; }; DA162BEB2CD3B53E0040DAE5 /* LogsViewer.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LogsViewer.swift; sourceTree = ""; }; + DA162BEF2CD4CC730040DAE5 /* LogsViewer.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = LogsViewer.swift; sourceTree = ""; }; DA19E25A22FD801D002F8F2F /* OpenHABGeneralTests.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OpenHABGeneralTests.swift; sourceTree = ""; }; DA1C2E4B230DC28F00FACFB0 /* Appfile */ = {isa = PBXFileReference; lastKnownFileType = text; path = Appfile; sourceTree = ""; }; DA1C2E4C230DC28F00FACFB0 /* SnapshotHelper.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SnapshotHelper.swift; sourceTree = ""; }; @@ -404,10 +407,11 @@ DA6B2EF42C89F8F200DF77CF /* ColorPickerView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ColorPickerView.swift; sourceTree = ""; }; DA6B2EF62C8B92E800DF77CF /* SelectionView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SelectionView.swift; sourceTree = ""; }; DA7224D123828D3300712D20 /* PreviewConstants.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PreviewConstants.swift; sourceTree = ""; }; - DA72E1B5236DEA0900B8EF3A /* AppMessageService.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = AppMessageService.swift; sourceTree = ""; }; + DA72E1B5236DEA0900B8EF3A /* WatchConnectivitySessionManager.swift */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.swift; path = WatchConnectivitySessionManager.swift; sourceTree = ""; }; DA7E1E47222EB00B002AEFD8 /* PlayerView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = PlayerView.swift; sourceTree = ""; }; DA817E79234BF39B00C91824 /* CHANGELOG.md */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = net.daringfireball.markdown; path = CHANGELOG.md; sourceTree = ""; }; DA88F8C522EC377100B408E5 /* ReleaseNotes.md */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = net.daringfireball.markdown; path = ReleaseNotes.md; sourceTree = ""; }; + DA8F986A2CDA4CAA001F5E8A /* ContentView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = ContentView.swift; sourceTree = ""; }; DA9721C224E29A8F0092CCFD /* UserDefaultsBacked.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UserDefaultsBacked.swift; sourceTree = ""; }; DA9F81862C85020F00B47B72 /* RTFTextView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = RTFTextView.swift; sourceTree = ""; }; DAA070922B5181210060BB0E /* OpenHABImageDownloaderOperation.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = OpenHABImageDownloaderOperation.swift; sourceTree = ""; }; @@ -589,7 +593,7 @@ 1224F7C9228A8ED100750965 /* External */ = { isa = PBXGroup; children = ( - DA72E1B5236DEA0900B8EF3A /* AppMessageService.swift */, + DA72E1B5236DEA0900B8EF3A /* WatchConnectivitySessionManager.swift */, ); name = External; path = openHABWatch/External; @@ -666,6 +670,7 @@ isa = PBXGroup; children = ( DAD0855F2AE47824001D36BE /* OpenHABWatch.swift */, + DA8F986A2CDA4CAA001F5E8A /* ContentView.swift */, DA0775252346705F0086C685 /* Extension */, 1224F7C9228A8ED100750965 /* External */, 1224F7C8228A8EC600750965 /* Domain */, @@ -776,7 +781,7 @@ DA658720236F841F007E2E7F /* Views */ = { isa = PBXGroup; children = ( - DA0775262346705F0086C685 /* ContentView.swift */, + DA0775262346705F0086C685 /* SitemapView.swift */, DA162BEB2CD3B53E0040DAE5 /* LogsViewer.swift */, DAC6608C236F771600F4501E /* PreferencesSwiftUIView.swift */, DAF457A323DB7A820018B495 /* Rows */, @@ -871,6 +876,7 @@ 653B54BF285C0AC700298ECD /* OpenHABRootViewController.swift */, 65570A7C2476D16A00D524EA /* OpenHABWebViewController.swift */, DFB2624318830A3600D3244D /* OpenHABSitemapViewController.swift */, + DA162BEF2CD4CC730040DAE5 /* LogsViewer.swift */, DAC65FC6236EDF3900F4501E /* SpinnerViewController.swift */, DA6B2EF62C8B92E800DF77CF /* SelectionView.swift */, DA242C612C83588600AFB10D /* SettingsView.swift */, @@ -1093,6 +1099,7 @@ 39C91164B60A5677322E8DE2 /* Frameworks */, DA07751D2346705F0086C685 /* Sources */, 93F38C61238034AE001B1451 /* Embed Frameworks */, + DA162BEE2CD3EDFE0040DAE5 /* ShellScript */, ); buildRules = ( ); @@ -1397,6 +1404,23 @@ /* End PBXResourcesBuildPhase section */ /* Begin PBXShellScriptBuildPhase section */ + DA162BEE2CD3EDFE0040DAE5 /* ShellScript */ = { + isa = PBXShellScriptBuildPhase; + buildActionMask = 2147483647; + files = ( + ); + inputFileListPaths = ( + ); + inputPaths = ( + ); + outputFileListPaths = ( + ); + outputPaths = ( + ); + runOnlyForDeploymentPostprocessing = 0; + shellPath = /bin/sh; + shellScript = "cd BuildTools\nSDKROOT=(xcrun --sdk macosx --show-sdk-path)\n\nswift package plugin --allow-writing-to-package-directory --allow-writing-to-directory \"$SRCROOT\" swiftformat \"$SRCROOT\" --config ./.swiftformat --cache /private/tmp/\nswift package plugin --allow-writing-to-package-directory --allow-writing-to-directory ../ swiftlint --cache-path /private/tmp/\n# Type a script or drag a script file from your workspace to insert its path.\n"; + }; DAF0A2902C56FE9F00A14A6A /* Run swiftformat & swiftlint */ = { isa = PBXShellScriptBuildPhase; alwaysOutOfDate = 1; @@ -1474,14 +1498,15 @@ DA0749E023E0BF510057FA83 /* ColorSelection.swift in Sources */, DA65871F236F83CE007E2E7F /* UserDefaultsExtension.swift in Sources */, DA9721C324E29A8F0092CCFD /* UserDefaultsBacked.swift in Sources */, - DA72E1B8236DEA0900B8EF3A /* AppMessageService.swift in Sources */, + DA72E1B8236DEA0900B8EF3A /* WatchConnectivitySessionManager.swift in Sources */, DA07752B2346705F0086C685 /* OpenHABWatchAppDelegate.swift in Sources */, DAF4581623DC48400018B495 /* GenericRow.swift in Sources */, 9350F17A23814FAC00054BA8 /* ObservableOpenHABSitemapPage.swift in Sources */, DAF457A023DA3E1C0018B495 /* SegmentRow.swift in Sources */, DAF4578923D79AA50018B495 /* DetailTextLabelView.swift in Sources */, - DAAC30872CBBF0420041927F /* ContentView.swift in Sources */, + DAAC30872CBBF0420041927F /* SitemapView.swift in Sources */, DA15BFBD23C6726400BD8ADA /* ObservableOpenHABDataObject.swift in Sources */, + DA8F986B2CDA4CAA001F5E8A /* ContentView.swift in Sources */, DAC9AF4924F966FA006DAE93 /* LazyView.swift in Sources */, DA0776F0234788010086C685 /* UserData.swift in Sources */, DAC6608D236F771600F4501E /* PreferencesSwiftUIView.swift in Sources */, @@ -1540,6 +1565,7 @@ 1224F78F228A89FD00750965 /* WatchMessageService.swift in Sources */, DAA42BAC21DC984A00244B2A /* WebUITableViewCell.swift in Sources */, DF4B84131886DAC400F34902 /* FrameUITableViewCell.swift in Sources */, + DA162BF02CD4CC730040DAE5 /* LogsViewer.swift in Sources */, DF4B84161886EACA00F34902 /* GenericUITableViewCell.swift in Sources */, 935B484625342B8E00E44CF0 /* URL+Static.swift in Sources */, B7D5ECE121499E55001B0EC6 /* MapViewTableViewCell.swift in Sources */, @@ -1682,13 +1708,11 @@ CLANG_WARN_STRICT_PROTOTYPES = YES; CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; CODE_SIGN_ENTITLEMENTS = openHABIntents/openHABIntents.entitlements; - CODE_SIGN_IDENTITY = "Apple Distribution"; - "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "Apple Distribution"; - CODE_SIGN_STYLE = Manual; + CODE_SIGN_IDENTITY = "Apple Development"; + CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 29; DEBUG_INFORMATION_FORMAT = dwarf; - DEVELOPMENT_TEAM = ""; - "DEVELOPMENT_TEAM[sdk=iphoneos*]" = PBAPXHRAM9; + DEVELOPMENT_TEAM = PBAPXHRAM9; GCC_C_LANGUAGE_STANDARD = gnu11; GCC_NO_COMMON_BLOCKS = YES; GCC_OPTIMIZATION_LEVEL = 0; @@ -1705,8 +1729,7 @@ MTL_FAST_MATH = YES; PRODUCT_BUNDLE_IDENTIFIER = org.openhab.app.openHABIntents; PRODUCT_NAME = "$(TARGET_NAME)"; - PROVISIONING_PROFILE_SPECIFIER = "match AppStore org.openhab.app.openHABIntents"; - "PROVISIONING_PROFILE_SPECIFIER[sdk=iphoneos*]" = "match AppStore org.openhab.app.openHABIntents"; + PROVISIONING_PROFILE_SPECIFIER = ""; SKIP_INSTALL = YES; SWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG; SWIFT_OPTIMIZATION_LEVEL = "-Onone"; @@ -1769,13 +1792,11 @@ CLANG_WARN_STRICT_PROTOTYPES = YES; CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; CODE_SIGN_ENTITLEMENTS = NotificationService/NotificationService.entitlements; - CODE_SIGN_IDENTITY = "Apple Distribution"; - "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "Apple Distribution"; - CODE_SIGN_STYLE = Manual; + CODE_SIGN_IDENTITY = "Apple Development"; + CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 29; DEBUG_INFORMATION_FORMAT = dwarf; - DEVELOPMENT_TEAM = ""; - "DEVELOPMENT_TEAM[sdk=iphoneos*]" = PBAPXHRAM9; + DEVELOPMENT_TEAM = PBAPXHRAM9; ENABLE_USER_SCRIPT_SANDBOXING = YES; GCC_C_LANGUAGE_STANDARD = gnu17; GCC_NO_COMMON_BLOCKS = YES; @@ -1797,8 +1818,7 @@ MTL_FAST_MATH = YES; PRODUCT_BUNDLE_IDENTIFIER = org.openhab.app.NotificationService; PRODUCT_NAME = "$(TARGET_NAME)"; - PROVISIONING_PROFILE_SPECIFIER = "match AppStore org.openhab.app.NotificationService"; - "PROVISIONING_PROFILE_SPECIFIER[sdk=iphoneos*]" = "match AppStore org.openhab.app.NotificationService"; + PROVISIONING_PROFILE_SPECIFIER = ""; SKIP_INSTALL = YES; SWIFT_ACTIVE_COMPILATION_CONDITIONS = "DEBUG $(inherited)"; SWIFT_EMIT_LOC_STRINGS = YES; @@ -1953,13 +1973,11 @@ CLANG_WARN_STRICT_PROTOTYPES = YES; CLANG_WARN_UNGUARDED_AVAILABILITY = YES_AGGRESSIVE; CODE_SIGN_ENTITLEMENTS = "openHABWatch Extension/openHABWatch Extension.entitlements"; - CODE_SIGN_IDENTITY = "Apple Distribution"; - "CODE_SIGN_IDENTITY[sdk=watchos*]" = "Apple Distribution"; - CODE_SIGN_STYLE = Manual; + CODE_SIGN_IDENTITY = "Apple Development"; + CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 29; DEBUG_INFORMATION_FORMAT = dwarf; - DEVELOPMENT_TEAM = ""; - "DEVELOPMENT_TEAM[sdk=watchos*]" = PBAPXHRAM9; + DEVELOPMENT_TEAM = PBAPXHRAM9; GCC_C_LANGUAGE_STANDARD = "compiler-default"; GCC_NO_COMMON_BLOCKS = YES; GCC_OPTIMIZATION_LEVEL = 0; @@ -1981,8 +1999,7 @@ MTL_FAST_MATH = YES; PRODUCT_BUNDLE_IDENTIFIER = org.openhab.app.watchkitapp; PRODUCT_NAME = "$(TARGET_NAME)"; - PROVISIONING_PROFILE_SPECIFIER = "match AppStore org.openhab.app.watchkitapp"; - "PROVISIONING_PROFILE_SPECIFIER[sdk=watchos*]" = "match AppStore org.openhab.app.watchkitapp"; + PROVISIONING_PROFILE_SPECIFIER = ""; SDKROOT = watchos; SKIP_INSTALL = YES; SWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG; @@ -2409,12 +2426,10 @@ CLANG_CXX_LANGUAGE_STANDARD = "$(inherited)"; CLANG_ENABLE_MODULES = YES; CODE_SIGN_ENTITLEMENTS = openHAB/openHAB.entitlements; - CODE_SIGN_IDENTITY = "Apple Distribution"; - "CODE_SIGN_IDENTITY[sdk=iphoneos*]" = "Apple Distribution"; - CODE_SIGN_STYLE = Manual; + CODE_SIGN_IDENTITY = "Apple Development"; + CODE_SIGN_STYLE = Automatic; CURRENT_PROJECT_VERSION = 29; - DEVELOPMENT_TEAM = ""; - "DEVELOPMENT_TEAM[sdk=iphoneos*]" = PBAPXHRAM9; + DEVELOPMENT_TEAM = PBAPXHRAM9; GCC_PRECOMPILE_PREFIX_HEADER = YES; GCC_PREFIX_HEADER = "openHAB/openHAB-Prefix.pch"; INFOPLIST_FILE = "openHAB/openHAB-Info.plist"; @@ -2436,8 +2451,7 @@ PRODUCT_BUNDLE_IDENTIFIER = org.openhab.app; PRODUCT_NAME = "$(TARGET_NAME)"; PROVISIONING_PROFILE = ""; - PROVISIONING_PROFILE_SPECIFIER = "match AppStore org.openhab.app"; - "PROVISIONING_PROFILE_SPECIFIER[sdk=iphoneos*]" = "match AppStore org.openhab.app"; + PROVISIONING_PROFILE_SPECIFIER = ""; SUPPORTS_MACCATALYST = NO; SWIFT_INSTALL_OBJC_HEADER = NO; SWIFT_OBJC_BRIDGING_HEADER = ""; diff --git a/openHAB/LogsViewer.swift b/openHAB/LogsViewer.swift new file mode 100644 index 00000000..f2bf252b --- /dev/null +++ b/openHAB/LogsViewer.swift @@ -0,0 +1,65 @@ +// Copyright (c) 2010-2024 Contributors to the openHAB project +// +// See the NOTICE file(s) distributed with this work for additional +// information. +// +// This program and the accompanying materials are made available under the +// terms of the Eclipse Public License 2.0 which is available at +// http://www.eclipse.org/legal/epl-2.0 +// +// SPDX-License-Identifier: EPL-2.0 + +import Foundation +import OpenHABCore +import OSLog +import SwiftUI + +// swiftlint:disable:next file_types_order +struct LogsViewer: View { + let template = NSPredicate( + format: "(subsystem BEGINSWITH $PREFIX)" + ) + let myFont = Font + .system(size: 10) + .monospaced() + + var logService: LogServiceProtocol + + var body: some View { + List { + Text(text) + .font(myFont) + } + .toolbar { + ToolbarItem(placement: .primaryAction) { + ShareLink( + item: text, + preview: SharePreview("Logs", image: Image(.openHABIcon)) + ) { + Label("Share Logs", systemSymbol: .squareAndArrowUp) + } + } + } + .navigationTitle("Logs") + .task { + text = await logService.fetchLogs(with: template) + } + } + + @State private var text = "Loading..." +} + +#if DEBUG +struct MockLogService: LogServiceProtocol { + func fetchLogs(with template: NSPredicate) async -> String { + """ + Mocked Data + Test data + """ + } +} +#endif + +#Preview { + LogsViewer(logService: MockLogService()) +} diff --git a/openHAB/SettingsView.swift b/openHAB/SettingsView.swift index cd728014..5adf27cb 100644 --- a/openHAB/SettingsView.swift +++ b/openHAB/SettingsView.swift @@ -43,6 +43,8 @@ struct SettingsView: View { @State private var sitemaps: [OpenHABSitemap] = [] + @State private var exportShown = false + @Environment(\.dismiss) private var dismiss var appData: OpenHABDataObject? { @@ -287,6 +289,14 @@ struct SettingsView: View { } } + Section(header: Text(LocalizedStringKey("debug"))) { + NavigationLink { + LogsViewer(logService: LogService()) + } label: { + Text("Logs") + } + } + Section(header: Text(LocalizedStringKey("about_settings"))) { LabeledContent("App Version", value: appVersion) diff --git a/openHABWatch/ContentView.swift b/openHABWatch/ContentView.swift new file mode 100644 index 00000000..ffb693f0 --- /dev/null +++ b/openHABWatch/ContentView.swift @@ -0,0 +1,48 @@ +// Copyright (c) 2010-2024 Contributors to the openHAB project +// +// See the NOTICE file(s) distributed with this work for additional +// information. +// +// This program and the accompanying materials are made available under the +// terms of the Eclipse Public License 2.0 which is available at +// http://www.eclipse.org/legal/epl-2.0 +// +// SPDX-License-Identifier: EPL-2.0 + +import OpenHABCore +import SwiftUI + +struct ContentView: View { + @ObservedObject var viewModel: UserData + @EnvironmentObject var settings: ObservableOpenHABDataObject + @State var title = "openHAB" + + var body: some View { + TabView { + NavigationStack { + SitemapView(viewModel: viewModel) + } + .tabItem { + Label("Sitemap", systemSymbol: .circleFill) + } + NavigationStack { + PreferencesSwiftUIView() + } + .tabItem { + Label("Preferences", systemSymbol: .circleFill) + } + NavigationStack { + LogsViewer(logService: LogService()) + } + .tabItem { + Label("Debug", systemSymbol: .circleFill) + } + } + .tabViewStyle(.page) + } +} + +#Preview { + ContentView(viewModel: .init()) + .environmentObject(ObservableOpenHABDataObject()) +} diff --git a/openHABWatch/Domain/UserData.swift b/openHABWatch/Domain/UserData.swift index e4009828..ddd913c9 100644 --- a/openHABWatch/Domain/UserData.swift +++ b/openHABWatch/Domain/UserData.swift @@ -76,7 +76,7 @@ final class UserData: ObservableObject { self?.logger.error("openHABTracked: \(activeConnection.configuration.url)") if !ObservableOpenHABDataObject.shared.haveReceivedAppContext { - AppMessageService.singleton.requestApplicationContext() + WatchConnectivitySessionManager.singleton.requestApplicationContext() self?.errorDescription = NSLocalizedString("settings_not_received", comment: "") self?.showAlert = true return diff --git a/openHABWatch/Extension/OpenHABWatchAppDelegate.swift b/openHABWatch/Extension/OpenHABWatchAppDelegate.swift index 584d2f42..5cdb8be1 100644 --- a/openHABWatch/Extension/OpenHABWatchAppDelegate.swift +++ b/openHABWatch/Extension/OpenHABWatchAppDelegate.swift @@ -20,7 +20,7 @@ class OpenHABWatchAppDelegate: NSObject { let delegate: WCSessionDelegate override init() { - delegate = AppMessageService.singleton + delegate = WatchConnectivitySessionManager.singleton session = .default session.delegate = delegate session.activate() diff --git a/openHABWatch/External/AppMessageService.swift b/openHABWatch/External/WatchConnectivitySessionManager.swift similarity index 92% rename from openHABWatch/External/AppMessageService.swift rename to openHABWatch/External/WatchConnectivitySessionManager.swift index c29c1e78..9f1da0c2 100644 --- a/openHABWatch/External/AppMessageService.swift +++ b/openHABWatch/External/WatchConnectivitySessionManager.swift @@ -16,8 +16,8 @@ import WatchConnectivity import WatchKit // This class handles values that are passed from the ios app. -class AppMessageService: NSObject, WCSessionDelegate { - static let singleton = AppMessageService() +class WatchConnectivitySessionManager: NSObject, WCSessionDelegate { + static let singleton = WatchConnectivitySessionManager() private let logger = Logger(subsystem: "org.openhab.app.watchkitapp", category: "AppMessageService") @@ -130,3 +130,17 @@ class AppMessageService: NSObject, WCSessionDelegate { } } } + + +extension WatchConnectivitySessionManager { + + func transferFile(file: URL, metadata: [String : AnyObject]) -> WCSessionFileTransfer? { + return WCSession.default.transferFile(file, metadata: metadata) + } + + func session(_ session: WCSession, didFinish fileTransfer: WCSessionFileTransfer, error: Error?) { + // handle filed transfer completion + } + +} + diff --git a/openHABWatch/OpenHABWatch.swift b/openHABWatch/OpenHABWatch.swift index 0aaf26ed..8fb89e2d 100644 --- a/openHABWatch/OpenHABWatch.swift +++ b/openHABWatch/OpenHABWatch.swift @@ -9,6 +9,7 @@ // // SPDX-License-Identifier: EPL-2.0 +import OpenHABCore import SDWebImage import SDWebImageSVGCoder import SwiftUI @@ -23,28 +24,14 @@ struct OpenHABWatch: App { var body: some Scene { WindowGroup { - TabView { - ContentView(viewModel: userData) - .tabItem { - Label("Sitemap", systemSymbol: .circleFill) - } - PreferencesSwiftUIView() - .tabItem { - Label("Preferences", systemSymbol: .circleFill) - } - LogsViewer() - .tabItem { - Label("Debug", systemSymbol: .circleFill) - } - } - .tabViewStyle(.page) - .environmentObject(settings) - .task { - let center = UNUserNotificationCenter.current() - _ = try? await center.requestAuthorization( - options: [.alert, .sound, .badge] - ) - } + ContentView(viewModel: userData) + .environmentObject(settings) + .task { + let center = UNUserNotificationCenter.current() + _ = try? await center.requestAuthorization( + options: [.alert, .sound, .badge] + ) + } } WKNotificationScene(controller: NotificationController.self, category: "openHABNotification") } diff --git a/openHABWatch/Views/LogsViewer.swift b/openHABWatch/Views/LogsViewer.swift index d4ac8cbc..36b5bed5 100644 --- a/openHABWatch/Views/LogsViewer.swift +++ b/openHABWatch/Views/LogsViewer.swift @@ -1,111 +1,62 @@ +// Copyright (c) 2010-2024 Contributors to the openHAB project // -// LogView.swift -// openHABWatch +// See the NOTICE file(s) distributed with this work for additional +// information. // -// Created by Tim Müller-Seydlitz on 31.10.24. -// Copyright © 2024 openHAB e.V. All rights reserved. +// This program and the accompanying materials are made available under the +// terms of the Eclipse Public License 2.0 which is available at +// http://www.eclipse.org/legal/epl-2.0 // +// SPDX-License-Identifier: EPL-2.0 import Foundation +import OpenHABCore import OSLog import SwiftUI // Thanks to https://useyourloaf.com/blog/fetching-oslog-messages-in-swift/ -extension OSLogEntryLog.Level { - fileprivate var description: String { - switch self { - case .undefined: "undefined" - case .debug: "debug" - case .info: "info" - case .notice: "notice" - case .error: "error" - case .fault: "fault" - @unknown default: "default" - } - } -} - -extension Logger { - static public func fetch(since date: Date, - predicateFormat: String) async throws -> [String] { - let store = try OSLogStore(scope: .currentProcessIdentifier) - let position = store.position(date: date) - let predicate = NSPredicate(format: predicateFormat) - let entries = try store.getEntries( - at: position, - matching: predicate - ) - - let dateFormatter = DateFormatter() - dateFormatter.dateFormat = "yyyy-MM-dd HH:mm:ss" - - var logs: [String] = [] - for entry in entries { - try Task.checkCancellation() - if let log = entry as? OSLogEntryLog { - var attributedMessage = AttributedString(dateFormatter.string(from: entry.date)) - attributedMessage.font = .headline - - logs.append(""" - \(dateFormatter.string(from: entry.date)): \ - \(log.category):\(log.level.description): \ - \(entry.composedMessage)\n - """) - } else { - logs.append("\(entry.date): \(entry.composedMessage)\n") - } - } - - if logs.isEmpty { logs = ["Nothing found"] } - return logs - } -} - +// swiftlint:disable:next file_types_order struct LogsViewer: View { - @State private var text = "Loading..." - - static private let template = NSPredicate(format: - "(subsystem BEGINSWITH $PREFIX)") - + let template = NSPredicate( + format: "(subsystem BEGINSWITH $PREFIX)" + ) let myFont = Font - .system(size: 10) - .monospaced() - - private func fetchLogs() async -> String { - let calendar = Calendar.current - guard let dayAgo = calendar.date(byAdding: .day, - value: -1, to: Date.now) else { - return "Invalid calendar" - } - - do { - let predicate = Self.template.withSubstitutionVariables( - [ - "PREFIX": "org.openhab" - ]) + .system(size: 10) + .monospaced() - let logs = try await Logger.fetch(since: dayAgo, - predicateFormat: predicate.predicateFormat) - return logs.joined() - } catch { - return error.localizedDescription - } - } + var logService: LogServiceProtocol var body: some View { - - ScrollView { - Text(text) - .font(myFont) - .padding() + List { + Text(text) + .font(myFont) } + .toolbar { + ToolbarItem(placement: .primaryAction) { + ShareLink( + item: text, + preview: SharePreview("Logs") + ) { + Label("Share Logs", systemSymbol: .squareAndArrowUp) + } + } + } + .navigationTitle("Logs") .task { - text = await fetchLogs() + text = await logService.fetchLogs(with: template) } } + + @State private var text = "Loading..." +} + +struct MockLogService: LogServiceProtocol { + func fetchLogs(with template: NSPredicate) async -> String { + "Mocked Data" + } } #Preview { - LogsViewer() + LogsViewer(logService: MockLogService()) } diff --git a/openHABWatch/Views/PreferencesSwiftUIView.swift b/openHABWatch/Views/PreferencesSwiftUIView.swift index 2629d9e9..af312a82 100644 --- a/openHABWatch/Views/PreferencesSwiftUIView.swift +++ b/openHABWatch/Views/PreferencesSwiftUIView.swift @@ -32,11 +32,14 @@ struct PreferencesSwiftUIView: View { LabeledContent(LocalizedStringKey("username"), value: settings.openHABUsername) LabeledContent(LocalizedStringKey("version"), value: applicationVersionNumber) } - - Button { AppMessageService.singleton.requestApplicationContext() - } label: { Label("sync_prefs", systemSymbol: .arrowTriangle2Circlepath) + .toolbar { + ToolbarItem(placement: .primaryAction) { + Button { WatchConnectivitySessionManager.singleton.requestApplicationContext() + } label: { Label("sync_prefs", systemSymbol: .arrowTriangle2Circlepath) + } + } } - .buttonStyle(.borderedProminent) + .navigationTitle("Preferences") } } diff --git a/openHABWatch/Views/ContentView.swift b/openHABWatch/Views/SitemapView.swift similarity index 86% rename from openHABWatch/Views/ContentView.swift rename to openHABWatch/Views/SitemapView.swift index 96b4565f..e6899794 100644 --- a/openHABWatch/Views/ContentView.swift +++ b/openHABWatch/Views/SitemapView.swift @@ -13,7 +13,7 @@ import OpenHABCore import os.log import SwiftUI -struct ContentView: View { +struct SitemapView: View { @ObservedObject var viewModel: UserData @EnvironmentObject var settings: ObservableOpenHABDataObject @State var title = "openHAB" @@ -21,17 +21,16 @@ struct ContentView: View { var body: some View { ZStack { ScrollView { - HStack { - Text(viewModel.openHABSitemapPage?.title ?? "Sitemap without title") - .font(.body) - .lineLimit(1) - Spacer() - } ForEach(viewModel.widgets) { widget in rowWidget(widget: widget) } + + if viewModel.widgets.isEmpty { + Text("No sitemap to show") + .foregroundStyle(.gray) + } } - .navigationBarTitle(Text(title)) + .navigationTitle(Text(title)) .actionSheet(isPresented: $viewModel.showCertificateAlert) { ActionSheet( title: Text(NSLocalizedString("warning", comment: "")), @@ -111,15 +110,18 @@ struct ContentView: View { } #Preview { - Group { - ContentView(viewModel: UserData()) + SitemapView(viewModel: UserData()) - .environmentObject({ () -> UserData in - let envObj = UserData() - return envObj - }()) + .environmentObject({ () -> UserData in + let envObj = UserData() + return envObj + }()) - ContentView(viewModel: UserData()) - } - .environmentObject(ObservableOpenHABDataObject()) + .environmentObject(ObservableOpenHABDataObject()) +} + +#Preview { + SitemapView(viewModel: UserData()) + + .environmentObject(ObservableOpenHABDataObject()) }