From f3232cacfb464cb7a881f708e74d0a7ef879e724 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 25 Mar 2024 11:11:53 -0500 Subject: [PATCH 001/141] v.1.2.0 --- CHANGELOG.md | 11 +++++++++++ Nudge/Info.plist | 4 ++-- 2 files changed, 13 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 939d5fe7..ac376504 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,6 +4,17 @@ All notable changes to this project will be documented in this file. The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). +## [1.2.0] - 2024-05-08 +Requires macOS x.0 and higher. + +### Changed +Upcoming + +### Fixed +Upcoming +### Added +Upcoming + ## [1.1.16] - 2024-03-13 This will be the **final Nudge release** for macOS 11 and potentially other versions of macOS. diff --git a/Nudge/Info.plist b/Nudge/Info.plist index c7214285..90c42a07 100644 --- a/Nudge/Info.plist +++ b/Nudge/Info.plist @@ -15,9 +15,9 @@ CFBundlePackageType $(PRODUCT_BUNDLE_PACKAGE_TYPE) CFBundleShortVersionString - 1.1.16 + 1.2.0 CFBundleVersion - 1.1.16 + 1.2.0 LSApplicationCategoryType public.app-category.utilities LSMinimumSystemVersion From c4dd6b2c35643507215678bbdbc0efc53199a733 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 25 Mar 2024 11:16:20 -0500 Subject: [PATCH 002/141] bump to macos12 and fix marketing version --- Nudge.xcodeproj/project.pbxproj | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/Nudge.xcodeproj/project.pbxproj b/Nudge.xcodeproj/project.pbxproj index 636edf4f..b300f470 100644 --- a/Nudge.xcodeproj/project.pbxproj +++ b/Nudge.xcodeproj/project.pbxproj @@ -580,7 +580,7 @@ GCC_WARN_UNUSED_FUNCTION = YES; GCC_WARN_UNUSED_VARIABLE = YES; LOCALIZATION_PREFERS_STRING_CATALOGS = YES; - MACOSX_DEPLOYMENT_TARGET = 11.0; + MACOSX_DEPLOYMENT_TARGET = 12.0; MTL_ENABLE_DEBUG_INFO = INCLUDE_SOURCE; MTL_FAST_MATH = YES; ONLY_ACTIVE_ARCH = YES; @@ -640,7 +640,7 @@ GCC_WARN_UNUSED_FUNCTION = YES; GCC_WARN_UNUSED_VARIABLE = YES; LOCALIZATION_PREFERS_STRING_CATALOGS = YES; - MACOSX_DEPLOYMENT_TARGET = 11.0; + MACOSX_DEPLOYMENT_TARGET = 12.0; MTL_ENABLE_DEBUG_INFO = NO; MTL_FAST_MATH = YES; SDKROOT = macosx; @@ -669,8 +669,8 @@ "$(inherited)", "@executable_path/../Frameworks", ); - MACOSX_DEPLOYMENT_TARGET = 11.0; - MARKETING_VERSION = 1.1.12; + MACOSX_DEPLOYMENT_TARGET = 12.0; + MARKETING_VERSION = 1.2.0; PRODUCT_BUNDLE_IDENTIFIER = com.github.macadmins.Nudge; PRODUCT_NAME = "$(TARGET_NAME)"; PROVISIONING_PROFILE_SPECIFIER = ""; @@ -700,8 +700,8 @@ "$(inherited)", "@executable_path/../Frameworks", ); - MACOSX_DEPLOYMENT_TARGET = 11.0; - MARKETING_VERSION = 1.1.12; + MACOSX_DEPLOYMENT_TARGET = 12.0; + MARKETING_VERSION = 1.2.0; PRODUCT_BUNDLE_IDENTIFIER = com.github.macadmins.Nudge; PRODUCT_NAME = "$(TARGET_NAME)"; PROVISIONING_PROFILE_SPECIFIER = ""; @@ -728,7 +728,7 @@ "@executable_path/../Frameworks", "@loader_path/../Frameworks", ); - MACOSX_DEPLOYMENT_TARGET = 11.0; + MACOSX_DEPLOYMENT_TARGET = 12.0; MARKETING_VERSION = 0.0.2; PRODUCT_BUNDLE_IDENTIFIER = com.github.macadmins.NudgeTests; PRODUCT_NAME = "$(TARGET_NAME)"; @@ -756,7 +756,7 @@ "@executable_path/../Frameworks", "@loader_path/../Frameworks", ); - MACOSX_DEPLOYMENT_TARGET = 11.0; + MACOSX_DEPLOYMENT_TARGET = 12.0; MARKETING_VERSION = 0.0.2; PRODUCT_BUNDLE_IDENTIFIER = com.github.macadmins.NudgeTests; PRODUCT_NAME = "$(TARGET_NAME)"; From 1520f7bab0db4d719613b25decf34b3ab91355b5 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 26 Mar 2024 12:22:08 -0500 Subject: [PATCH 003/141] add remote image support to companyLogo/screenShot --- Nudge/UI/Common/CompanyLogo.swift | 23 +++++++++++++--- Nudge/UI/StandardMode/RightSide.swift | 31 ++++++++++++++++------ Nudge/UI/StandardMode/ScreenShotZoom.swift | 21 +++++++++++++-- Nudge/Utilities/Extensions.swift | 18 ++++++++++--- Nudge/Utilities/Utils.swift | 8 ++++++ 5 files changed, 84 insertions(+), 17 deletions(-) diff --git a/Nudge/UI/Common/CompanyLogo.swift b/Nudge/UI/Common/CompanyLogo.swift index 2c7c1699..751744d2 100644 --- a/Nudge/UI/Common/CompanyLogo.swift +++ b/Nudge/UI/Common/CompanyLogo.swift @@ -28,8 +28,25 @@ struct CompanyLogo: View { } private var companyImage: some View { - Image(nsImage: ImageManager().getCorrectImage(path: companyLogoPath, type: "CompanyLogo")) - .customResizable(width: uiConstants.logoWidth, height: uiConstants.logoHeight) + AsyncImage(url: UIUtilities().createCorrectURLType(from: companyLogoPath)) { phase in + switch phase { + case .empty: + Image(systemName: "square.dashed") + .customResizable(width: uiConstants.logoWidth, height: uiConstants.logoHeight) + .customFontWeight(fontWeight: .ultraLight) + .opacity(0.05) + case .failure: + Image(systemName: "questionmark.square.dashed") + .customResizable(width: uiConstants.logoWidth, height: uiConstants.logoHeight) + .customFontWeight(fontWeight: .ultraLight) + .opacity(0.05) + case .success(let image): + image + .customResizable(width: uiConstants.logoWidth, height: uiConstants.logoHeight) + @unknown default: + EmptyView() + } + } } private var defaultImage: some View { @@ -53,7 +70,7 @@ struct CompanyLogo: View { } private func shouldShowCompanyLogo() -> Bool { - companyLogoPath.starts(with: "data:") || FileManager.default.fileExists(atPath: companyLogoPath) + ["data:", "https://", "http://", "file://"].contains(where: companyLogoPath.starts(with:)) || FileManager.default.fileExists(atPath: companyLogoPath) } } diff --git a/Nudge/UI/StandardMode/RightSide.swift b/Nudge/UI/StandardMode/RightSide.swift index 60d71f5b..0541b1fa 100644 --- a/Nudge/UI/StandardMode/RightSide.swift +++ b/Nudge/UI/StandardMode/RightSide.swift @@ -112,7 +112,7 @@ struct StandardModeRightSide: View { private var screenshotDisplay: some View { Group { - if shouldShowScreenshot { + if shouldShowScreenshot() { screenshotButton } else { EmptyView() @@ -122,8 +122,25 @@ struct StandardModeRightSide: View { private var screenshotButton: some View { Button(action: { appState.screenShotZoomViewIsPresented = true }) { - Image(nsImage: ImageManager().getCorrectImage(path: screenShotPath, type: "ScreenShot")) - .customResizable(maxHeight: UIConstants.screenshotMaxHeight) + AsyncImage(url: UIUtilities().createCorrectURLType(from: screenShotPath)) { phase in + switch phase { + case .empty: + Image(systemName: "square.dashed") + .customResizable(maxHeight: UIConstants.screenshotMaxHeight) + .customFontWeight(fontWeight: .ultraLight) + .opacity(0.05) + case .failure: + Image(systemName: "questionmark.square.dashed") + .customResizable(maxHeight: UIConstants.screenshotMaxHeight) + .customFontWeight(fontWeight: .ultraLight) + .opacity(0.05) + case .success(let image): + image + .customResizable(maxHeight: UIConstants.screenshotMaxHeight) + @unknown default: + EmptyView() + } + } } .buttonStyle(.plain) .help(UserInterfaceVariables.screenShotAltText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) @@ -132,11 +149,9 @@ struct StandardModeRightSide: View { } .onHoverEffect() } - - private var shouldShowScreenshot: Bool { - // Logic to determine if the screenshot should be shown - let imagePath = ImageManager().getScreenShotPath(colorScheme: colorScheme) - return FileManager.default.fileExists(atPath: imagePath) || imagePath.starts(with: "data:") || forceScreenShotIconMode() + + private func shouldShowScreenshot() -> Bool { + ["data:", "https://", "http://", "file://"].contains(where: screenShotPath.starts(with:)) || FileManager.default.fileExists(atPath: screenShotPath) || forceScreenShotIconMode() } } diff --git a/Nudge/UI/StandardMode/ScreenShotZoom.swift b/Nudge/UI/StandardMode/ScreenShotZoom.swift index fc62bbb1..60f1f12b 100644 --- a/Nudge/UI/StandardMode/ScreenShotZoom.swift +++ b/Nudge/UI/StandardMode/ScreenShotZoom.swift @@ -52,8 +52,25 @@ struct ScreenShotZoom: View { } private var screenShotImage: some View { - Image(nsImage: ImageManager().getCorrectImage(path: screenShotPath, type: "ScreenShot")) - .customResizable(maxHeight: 675) + AsyncImage(url: UIUtilities().createCorrectURLType(from: screenShotPath)) { phase in + switch phase { + case .empty: + Image(systemName: "square.dashed") + .customResizable(maxHeight: 675) + .customFontWeight(fontWeight: .ultraLight) + .opacity(0.05) + case .failure: + Image(systemName: "questionmark.square.dashed") + .customResizable(maxHeight: 675) + .customFontWeight(fontWeight: .ultraLight) + .opacity(0.05) + case .success(let image): + image + .scaledToFit() + @unknown default: + EmptyView() + } + } } } diff --git a/Nudge/Utilities/Extensions.swift b/Nudge/Utilities/Extensions.swift index 30f7e583..6b79958f 100644 --- a/Nudge/Utilities/Extensions.swift +++ b/Nudge/Utilities/Extensions.swift @@ -42,13 +42,23 @@ extension FixedWidthInteger { // Image Extension extension Image { - func customResizable(width: CGFloat? = nil, height: CGFloat? = nil, maxHeight: CGFloat? = nil) -> some View { + func customResizable(width: CGFloat? = nil, height: CGFloat? = nil, minHeight: CGFloat? = nil, minWidth: CGFloat? = nil, maxHeight: CGFloat? = nil, maxWidth: CGFloat? = nil) -> some View { self .resizable() - .aspectRatio(contentMode: .fit) .scaledToFit() - .frame(width: width, height: height) - .frame(maxHeight: maxHeight) + .frame(width: width, height: height, alignment: .center) + .frame(minWidth: minWidth, maxWidth: maxWidth, minHeight: minHeight, maxHeight: maxHeight) + } +} + +extension View { + @ViewBuilder + func customFontWeight(fontWeight: Font.Weight? = nil) -> some View { + if #available(macOS 13.0, *), let weight = fontWeight { + self.fontWeight(weight) + } else { + self + } } } diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 8af2223f..068584e5 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -800,6 +800,14 @@ struct UIUtilities { NSApp.windows.first?.center() } + func createCorrectURLType(from input: String) -> URL? { + // Checks if the input contains "://", a simple heuristic to decide if it's a web URL + let isWebURL = ["data:", "https://", "http://", "file://"].contains(where: input.starts(with:)) + + // Returns a URL initialized appropriately based on the input type + return isWebURL ? URL(string: input) : URL(fileURLWithPath: input) + } + private func determineUpdateURL() -> URL? { if let actionButtonPath = FeatureVariables.actionButtonPath { if actionButtonPath.isEmpty { From f38ae97da1118c998dd1b9b5fef642b69d448193 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 26 Mar 2024 12:44:56 -0500 Subject: [PATCH 004/141] tweak screenshot zoom again --- Nudge/UI/StandardMode/ScreenShotZoom.swift | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Nudge/UI/StandardMode/ScreenShotZoom.swift b/Nudge/UI/StandardMode/ScreenShotZoom.swift index 60f1f12b..ea60794a 100644 --- a/Nudge/UI/StandardMode/ScreenShotZoom.swift +++ b/Nudge/UI/StandardMode/ScreenShotZoom.swift @@ -24,7 +24,7 @@ struct ScreenShotZoom: View { Spacer() // Vertically align Screenshot to center } .background(Color(NSColor.windowBackgroundColor)) - .frame(maxWidth: 900) + .frame(minWidth: 850, maxWidth: 1200, minHeight: 900, maxHeight: 1200) } private var closeButton: some View { @@ -66,7 +66,7 @@ struct ScreenShotZoom: View { .opacity(0.05) case .success(let image): image - .scaledToFit() + .customResizable() @unknown default: EmptyView() } From c86a76f5e8bf780aedd6f0e538d9bb35c4ef46a0 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 26 Mar 2024 12:45:03 -0500 Subject: [PATCH 005/141] move to web assets --- Example Assets/com.github.macadmins.Nudge.tester.json | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index 400583c4..a11315a6 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -31,11 +31,11 @@ "randomDelay": false }, "userInterface": { - "iconDarkPath": "", - "iconLightPath": "", + "iconDarkPath": "https://github.com/macadmins/nudge/blob/main/assets/NudgeIconInverted.png?raw=true", + "iconLightPath": "https://github.com/macadmins/nudge/blob/main/assets/NudgeIcon.png?raw=true", + "screenShotDarkPath": "https://github.com/macadmins/nudge/blob/main/assets/standard_mode/demo_dark_1_icon.png?raw=true", + "screenShotLightPath": "https://github.com/macadmins/nudge/blob/main/assets/standard_mode/demo_light_1_icon.png?raw=true", "simpleMode": false, - "screenShotDarkPath": "", - "screenShotLightPath": "", "updateElements": [ { "_language": "en", From 0cfd0649543a6213ed2682c7f51eae9a89558d77 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 26 Mar 2024 12:53:58 -0500 Subject: [PATCH 006/141] mark macOS 12 and higher for now --- CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index ac376504..d8781556 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,7 +5,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). ## [1.2.0] - 2024-05-08 -Requires macOS x.0 and higher. +Requires macOS 12.0 and higher. Further releases and feature requests may make this macOS 13 and higher depending on code complexity. ### Changed Upcoming From 62178c295f3a94d210d7012892d39e40c66756c6 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 27 Mar 2024 10:19:09 -0500 Subject: [PATCH 007/141] allow window to be movable by admin (allowMovableWindow) --- Nudge/Preferences/DefaultPreferencesNudge.swift | 8 +++++++- Nudge/Preferences/PreferencesStructure.swift | 4 +++- Nudge/UI/Main.swift | 9 ++++++++- Nudge/Utilities/Utils.swift | 5 +++++ 4 files changed, 23 insertions(+), 3 deletions(-) diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 10092ca9..838135b4 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -178,13 +178,19 @@ struct UserExperienceVariables { static var allowGracePeriods: Bool { PrefsWrapper.allowGracePeriods } - + static var allowLaterDeferralButton: Bool { userExperienceProfile?["allowLaterDeferralButton"] as? Bool ?? userExperienceJSON?.allowLaterDeferralButton ?? true } + static var allowMovableWindow: Bool { + userExperienceProfile?["allowMovableWindow"] as? Bool ?? + userExperienceJSON?.allowMovableWindow ?? + false + } + static var allowUserQuitDeferrals: Bool { userExperienceProfile?["allowUserQuitDeferrals"] as? Bool ?? userExperienceJSON?.allowUserQuitDeferrals ?? diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index d4f64507..aa47637d 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -229,7 +229,7 @@ extension AboutUpdateURL { // MARK: - UserExperience struct UserExperience: Codable { - var allowGracePeriods, allowLaterDeferralButton, allowUserQuitDeferrals: Bool? + var allowGracePeriods, allowLaterDeferralButton, allowMovableWindow, allowUserQuitDeferrals: Bool? var allowedDeferrals, allowedDeferralsUntilForcedSecondaryQuitButton, approachingRefreshCycle, approachingWindowTime: Int? var calendarDeferralUnit: String? var elapsedRefreshCycle, gracePeriodInstallDelay, gracePeriodLaunchDelay: Int? @@ -264,6 +264,7 @@ extension UserExperience { func with( allowGracePeriods: Bool? = nil, allowLaterDeferralButton: Bool? = nil, + allowMovableWindow: Bool? = nil, allowUserQuitDeferrals: Bool? = nil, allowedDeferrals: Int? = nil, allowedDeferralsUntilForcedSecondaryQuitButton: Int? = nil, @@ -287,6 +288,7 @@ extension UserExperience { return UserExperience( allowGracePeriods: allowGracePeriods ?? self.allowGracePeriods, allowLaterDeferralButton: allowLaterDeferralButton ?? self.allowLaterDeferralButton, + allowMovableWindow: allowMovableWindow ?? self.allowMovableWindow, allowUserQuitDeferrals: allowUserQuitDeferrals ?? self.allowUserQuitDeferrals, allowedDeferrals: allowedDeferrals ?? self.allowedDeferrals, allowedDeferralsUntilForcedSecondaryQuitButton: allowedDeferralsUntilForcedSecondaryQuitButton ?? self.allowedDeferralsUntilForcedSecondaryQuitButton, diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 9714903f..1eb101d7 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -85,7 +85,7 @@ struct ContentView: View { window?.standardWindowButton(.miniaturizeButton)?.isHidden = true window?.standardWindowButton(.zoomButton)?.isHidden = true window?.center() - window?.isMovable = false + window?.isMovable = UserExperienceVariables.allowMovableWindow window?.collectionBehavior = [.fullScreenAuxiliary] window?.delegate = UIConstants.windowDelegate } @@ -215,16 +215,19 @@ class AppDelegate: NSObject, NSApplicationDelegate { } @objc func screenParametersChanged(_ notification: Notification) { + if UserExperienceVariables.allowMovableWindow { return } LogManager.info("Screen parameters changed - Notification Center", logger: utilsLog) UIUtilities().centerNudge() } @objc func screenProfileChanged(_ notification: Notification) { + if UserExperienceVariables.allowMovableWindow { return } LogManager.info("Display has changed profiles - Notification Center", logger: utilsLog) UIUtilities().centerNudge() } @objc func spacesStateChanged(_ notification: Notification) { + if UserExperienceVariables.allowMovableWindow { return } UIUtilities().centerNudge() LogManager.info("Spaces state changed", logger: utilsLog) nudgePrimaryState.afterFirstStateChange = true @@ -452,6 +455,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { forName: NSWindow.didChangeScreenNotification, object: NSApplication.shared, queue: .main) { _ in + if UserExperienceVariables.allowMovableWindow { return } print("Window object frame moved - Notification Center") UIUtilities().centerNudge() } @@ -548,14 +552,17 @@ class AppDelegate: NSObject, NSApplicationDelegate { class WindowDelegate: NSObject, NSWindowDelegate { func windowDidMove(_ notification: Notification) { + if UserExperienceVariables.allowMovableWindow { return } print("Window attempted to move - Window Delegate") UIUtilities().centerNudge() } func windowDidChangeScreen(_ notification: Notification) { + if UserExperienceVariables.allowMovableWindow { return } print("Window moved screens - Window Delegate") UIUtilities().centerNudge() } func windowDidChangeScreenProfile(_ notification: Notification) { + if UserExperienceVariables.allowMovableWindow { return } print("Display has changed profiles - Window Delegate") UIUtilities().centerNudge() } diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 068584e5..9e91e115 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -23,6 +23,11 @@ struct AppStateManager { LoggerUtilities().logUserQuitDeferrals() LoggerUtilities().logUserDeferrals() + // When the window is allowed to be moved, all of the other controls no longer force centering, so we need to force centering when re-activating. + if UserExperienceVariables.allowMovableWindow { + UIUtilities().centerNudge() + } + if DateManager().pastRequiredInstallationDate() && OptionalFeatureVariables.aggressiveUserFullScreenExperience { UIUtilities().centerNudge() NSApp.activate(ignoringOtherApps: true) From 9d6838c587c1fdca44e8d756571bbc296e9a8dbf Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 27 Mar 2024 17:15:38 -0500 Subject: [PATCH 008/141] first real attempt at getting gdmf data --- Nudge.xcodeproj/project.pbxproj | 16 +++ Nudge/3rd Party Assets/gdmf.swift | 222 ++++++++++++++++++++++++++++++ Nudge/UI/Defaults.swift | 1 + Nudge/UI/Main.swift | 4 + Nudge/Utilities/Utils.swift | 84 +++++++++++ 5 files changed, 327 insertions(+) create mode 100644 Nudge/3rd Party Assets/gdmf.swift diff --git a/Nudge.xcodeproj/project.pbxproj b/Nudge.xcodeproj/project.pbxproj index b300f470..d3296dad 100644 --- a/Nudge.xcodeproj/project.pbxproj +++ b/Nudge.xcodeproj/project.pbxproj @@ -19,6 +19,9 @@ 5836861C25DAD01C0004514C /* SoftwareUpdate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5836861B25DAD01C0004514C /* SoftwareUpdate.swift */; }; 6316F0E72832CA0700E1354D /* Schema in Resources */ = {isa = PBXBuildFile; fileRef = 6316F0E62832CA0700E1354D /* Schema */; }; 6347351D2B45DC2400C3401D /* CloseButton.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6347351C2B45DC2400C3401D /* CloseButton.swift */; }; + 634CE1092BB47480002C26C4 /* gdmf.swift in Sources */ = {isa = PBXBuildFile; fileRef = 634CE1082BB47480002C26C4 /* gdmf.swift */; }; + 634CE10A2BB47480002C26C4 /* gdmf.swift in Sources */ = {isa = PBXBuildFile; fileRef = 634CE1082BB47480002C26C4 /* gdmf.swift */; }; + 634CE10B2BB47480002C26C4 /* gdmf.swift in Sources */ = {isa = PBXBuildFile; fileRef = 634CE1082BB47480002C26C4 /* gdmf.swift */; }; 636B9C0226CACCAB0007BE3B /* DeferView.swift in Sources */ = {isa = PBXBuildFile; fileRef = 636B9C0126CACCAB0007BE3B /* DeferView.swift */; }; 636C4B4A25D1BECE0004A791 /* DefaultPreferencesNudge.swift in Sources */ = {isa = PBXBuildFile; fileRef = 636C4B4925D1BECE0004A791 /* DefaultPreferencesNudge.swift */; }; 636C4B7625D4306A0004A791 /* UILogic.swift in Sources */ = {isa = PBXBuildFile; fileRef = 636C4B7525D4306A0004A791 /* UILogic.swift */; }; @@ -91,6 +94,7 @@ 5836861B25DAD01C0004514C /* SoftwareUpdate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SoftwareUpdate.swift; sourceTree = ""; }; 6316F0E62832CA0700E1354D /* Schema */ = {isa = PBXFileReference; lastKnownFileType = folder; path = Schema; sourceTree = ""; }; 6347351C2B45DC2400C3401D /* CloseButton.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CloseButton.swift; sourceTree = ""; }; + 634CE1082BB47480002C26C4 /* gdmf.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = gdmf.swift; sourceTree = ""; }; 636B9C0126CACCAB0007BE3B /* DeferView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DeferView.swift; sourceTree = ""; }; 636C4B4925D1BECE0004A791 /* DefaultPreferencesNudge.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DefaultPreferencesNudge.swift; sourceTree = ""; }; 636C4B7525D4306A0004A791 /* UILogic.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UILogic.swift; sourceTree = ""; }; @@ -177,6 +181,14 @@ path = Common; sourceTree = ""; }; + 634CE1032BB47433002C26C4 /* 3rd Party Assets */ = { + isa = PBXGroup; + children = ( + 634CE1082BB47480002C26C4 /* gdmf.swift */, + ); + path = "3rd Party Assets"; + sourceTree = ""; + }; 639B6B3925DF1FEB00E38EC1 /* Preferences */ = { isa = PBXGroup; children = ( @@ -257,6 +269,7 @@ 63D7D0E125C9E9A400236281 /* Nudge */ = { isa = PBXGroup; children = ( + 634CE1032BB47433002C26C4 /* 3rd Party Assets */, 73CC1D7729B81EE500FBF8E2 /* com.github.macadmins.Nudge.SMAppService.plist */, 639B6B5425DF374600E38EC1 /* UI */, 639B6B3925DF1FEB00E38EC1 /* Preferences */, @@ -467,6 +480,7 @@ 63C39C1B2A38D33C0049EF62 /* Extensions.swift in Sources */, 0B0CCEDA25CE1C7C00A93D43 /* OSVersion.swift in Sources */, 639E198A25CD9E21008F618B /* Utils.swift in Sources */, + 634CE1092BB47480002C26C4 /* gdmf.swift in Sources */, 636C4B7625D4306A0004A791 /* UILogic.swift in Sources */, 41AD2B0026DE65B1004C52B1 /* QuitButtons.swift in Sources */, 636C4B4A25D1BECE0004A791 /* DefaultPreferencesNudge.swift in Sources */, @@ -497,6 +511,7 @@ files = ( 0BC9972C25CE2DFC0019FC8F /* OSVersionTests.swift in Sources */, 63D7D0F625C9E9A500236281 /* NudgeTests.swift in Sources */, + 634CE10A2BB47480002C26C4 /* gdmf.swift in Sources */, ); runOnlyForDeploymentPostprocessing = 0; }; @@ -505,6 +520,7 @@ buildActionMask = 2147483647; files = ( 63D7D10125C9E9A500236281 /* NudgeUITests.swift in Sources */, + 634CE10B2BB47480002C26C4 /* gdmf.swift in Sources */, ); runOnlyForDeploymentPostprocessing = 0; }; diff --git a/Nudge/3rd Party Assets/gdmf.swift b/Nudge/3rd Party Assets/gdmf.swift new file mode 100644 index 00000000..c548b1f8 --- /dev/null +++ b/Nudge/3rd Party Assets/gdmf.swift @@ -0,0 +1,222 @@ +// +// gdmf.swift +// Nudge +// +// Created by Erik Gomez on 3/27/24. +// + +import Foundation + +// Define the root structure +struct GDMFAssetInfo: Codable { + let publicAssetSets: AssetSets + let assetSets: AssetSets + let publicRapidSecurityResponses: AssetSets? + + enum CodingKeys: String, CodingKey { + case publicAssetSets = "PublicAssetSets" + case assetSets = "AssetSets" + case publicRapidSecurityResponses = "PublicRapidSecurityResponses" + } +} + +// Represents both PublicAssetSets and AssetSets +struct AssetSets: Codable { + let iOS: [Asset]? + let xrOS: [Asset]? + let macOS: [Asset]? + let visionOS: [Asset]? + + enum CodingKeys: String, CodingKey { + case iOS = "iOS" + case xrOS = "xrOS" + case macOS = "macOS" + case visionOS = "visionOS" + } +} + +// Represents an individual asset +struct Asset: Codable { + let productVersion: String + let build: String + let postingDate: String + let expirationDate: String + let supportedDevices: [String] + + enum CodingKeys: String, CodingKey { + case productVersion = "ProductVersion" + case build = "Build" + case postingDate = "PostingDate" + case expirationDate = "ExpirationDate" + case supportedDevices = "SupportedDevices" + } +} + +extension GDMFAssetInfo { + init(data: Data) throws { + let decoder = JSONDecoder() + decoder.dateDecodingStrategy = .iso8601 // Use ISO 8601 date format + self = try decoder.decode(GDMFAssetInfo.self, from: data) + } + + init(_ json: String, using encoding: String.Encoding = .utf8) throws { + guard let data = json.data(using: encoding) else { + throw NSError(domain: "JSONDecoding", code: 0, userInfo: nil) + } + try self.init(data: data) + } + + init(fromURL url: URL) throws { + try self.init(data: try Data(contentsOf: url)) + } + + func with( + PublicAssetSets: AssetSets, + AssetSets: AssetSets, + PublicRapidSecurityResponses: AssetSets + ) -> GDMFAssetInfo { + return GDMFAssetInfo( + publicAssetSets: PublicAssetSets, + assetSets: AssetSets, + publicRapidSecurityResponses: PublicRapidSecurityResponses + ) + } +} + +// https://arvindcs.medium.com/ssl-pinning-in-ios-30ee13f3202d +class GDMFPinnedSSL: NSObject { + static let shared = GDMFPinnedSSL() + + // Create an array to store the public keys of the trusted certificates + // To get these certs, download them as .cer, convert to .der, then base64 encode + //// openssl x509 -in Apple\ Server\ Authentication\ CA.cer -outform der -out Apple\ Server\ Authentication\ CA.der + /// base64 -i Apple\ Server\ Authentication\ CA.der + let trustedCertificates: [SecCertificate] = [ + // Apple Root CA + SecCertificateCreateWithData(nil, Data(base64Encoded: "MIIEuzCCA6OgAwIBAgIBAjANBgkqhkiG9w0BAQUFADBiMQswCQYDVQQGEwJVUzETMBEGA1UEChMKQXBwbGUgSW5jLjEmMCQGA1UECxMdQXBwbGUgQ2VydGlmaWNhdGlvbiBBdXRob3JpdHkxFjAUBgNVBAMTDUFwcGxlIFJvb3QgQ0EwHhcNMDYwNDI1MjE0MDM2WhcNMzUwMjA5MjE0MDM2WjBiMQswCQYDVQQGEwJVUzETMBEGA1UEChMKQXBwbGUgSW5jLjEmMCQGA1UECxMdQXBwbGUgQ2VydGlmaWNhdGlvbiBBdXRob3JpdHkxFjAUBgNVBAMTDUFwcGxlIFJvb3QgQ0EwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQDkkakJH5HbHkdQ6wXtXnmELes2oldMVeyLGYne+Uts9QerIjAC6Bg++FAJ039BqJj50cpmnCRrEdCju+QbKsMflZ56DKRHi1vUFjczy8QPTc4UadHJGXL1XQ7Vf1+b8iUDulWPTV0N8WQ1IxVLFVkds5T39pyez1C6wVhQZ48ItCD3y6wsIG9wtj8BMIy3Q88PnT3zK0koGsj+zrW5DtleHNbLPbU6rfQPDgCSC7EhFi501TwN22IWq6NxkkdTVcGvL0Gz+PvjcM3mo0xFfh9Ma1CWQYnEdGILEINBhzOKgbEwWOxaBDKMaLOPHd5lc/9nXmW8Sdh2nzMUZaF3lMktAgMBAAGjggF6MIIBdjAOBgNVHQ8BAf8EBAMCAQYwDwYDVR0TAQH/BAUwAwEB/zAdBgNVHQ4EFgQUK9BpR5R2Cf70a40uQKb3R01/CF4wHwYDVR0jBBgwFoAUK9BpR5R2Cf70a40uQKb3R01/CF4wggERBgNVHSAEggEIMIIBBDCCAQAGCSqGSIb3Y2QFATCB8jAqBggrBgEFBQcCARYeaHR0cHM6Ly93d3cuYXBwbGUuY29tL2FwcGxlY2EvMIHDBggrBgEFBQcCAjCBthqBs1JlbGlhbmNlIG9uIHRoaXMgY2VydGlmaWNhdGUgYnkgYW55IHBhcnR5IGFzc3VtZXMgYWNjZXB0YW5jZSBvZiB0aGUgdGhlbiBhcHBsaWNhYmxlIHN0YW5kYXJkIHRlcm1zIGFuZCBjb25kaXRpb25zIG9mIHVzZSwgY2VydGlmaWNhdGUgcG9saWN5IGFuZCBjZXJ0aWZpY2F0aW9uIHByYWN0aWNlIHN0YXRlbWVudHMuMA0GCSqGSIb3DQEBBQUAA4IBAQBcNplMLXi37Yyb3PN3m/J20ncwT8EfhYOFG5k9RzfyqZtAjizUsZAS2L70c5vu0mQPy3lPNNiiPvl4/2vIB+x9OYOLUyDTOMSxv5pPCmv/K/xZpwUJfBdAVhEedNO3iyM7R6PVbyTi69G3cN8PReEnyvFteO3ntRcXqNx+IjXKJdXZD9Zr1KIkIxH3oayPc4FgxhtbCS+SsvhESPBgOJ4V9T0mZyCKM2r3DYLP3uujL/lTaltkwGMzd/c6ByxW69oPIQ7aunMZT7XZNn/Bh1XZp5m5MkL72NVxnn6hUrcbvZNCJBIqxw8dtk2cXmPIS4AXUKqK1drk/NAJBzewdXUh")! as CFData)!, + // Apple Server Authentication CA + SecCertificateCreateWithData(nil, Data(base64Encoded: "MIID+DCCAuCgAwIBAgIII2l0BK3LgxQwDQYJKoZIhvcNAQELBQAwYjELMAkGA1UEBhMCVVMxEzARBgNVBAoTCkFwcGxlIEluYy4xJjAkBgNVBAsTHUFwcGxlIENlcnRpZmljYXRpb24gQXV0aG9yaXR5MRYwFAYDVQQDEw1BcHBsZSBSb290IENBMB4XDTE0MDMwODAxNTMwNFoXDTI5MDMwODAxNTMwNFowbTEnMCUGA1UEAwweQXBwbGUgU2VydmVyIEF1dGhlbnRpY2F0aW9uIENBMSAwHgYDVQQLDBdDZXJ0aWZpY2F0aW9uIEF1dGhvcml0eTETMBEGA1UECgwKQXBwbGUgSW5jLjELMAkGA1UEBhMCVVMwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQC5Jhawy4ercRWSjt+qPuGA11O6pGDMfIVy9zB8CU9XDUr/4V7JS1ATAmSxvTk10dcEUcEY+iL6rt+YGNa/Tk1DEPoliJ/TQIV25SKBtlRFc5qL45xIGoZ6w1Hi2pX4pH3bMN5sDsTF9WyY56b6VyAdGXN6Ds1jD7cniC7hmmiCuEBsYxYkZivnsuJUfeeIOaIbgT4C0znYl3dKMgzWCgqzBJvxcm9jqBUebDfoD9tTkNYpXLxqV5tGeAo+JOqaP6HYP/XbbqhsgrXdmTjsklaUpsVzJtGuCLLGUueOdkuJuFQPbuDZQtsqZYdGFLuWuFe7UeaEE/cNobaJrHzRIXSrAgMBAAGjgaYwgaMwHQYDVR0OBBYEFCzFbVLdMe+M7AiB7d/cykMARQHQMA8GA1UdEwEB/wQFMAMBAf8wHwYDVR0jBBgwFoAUK9BpR5R2Cf70a40uQKb3R01/CF4wLgYDVR0fBCcwJTAjoCGgH4YdaHR0cDovL2NybC5hcHBsZS5jb20vcm9vdC5jcmwwDgYDVR0PAQH/BAQDAgEGMBAGCiqGSIb3Y2QGAgwEAgUAMA0GCSqGSIb3DQEBCwUAA4IBAQAj8QZ+UEGBol7TcKRJka/YzGeMoSV9xJqTOS/YafsbQVtE19lryzslCRry9OPHnOiwW/Df3SIlERWTuUle2gxmel7Xb/Bj1GWMxHpUfVZPZZr92sSyyLC4oct94EeoQBW4FhntW2GO36rQzdI6wH46nyJO39/0ThrNk//Q8EVVZDM+1OXaaKATinYwJ9S/+B529vnDAO+xg+pTbVw1xw0HAbr4Ybn+xZprQ2GBA+u6X3Cd6G+UJEvczpKoLqI1PONJ4BZ3otxruY0YQrk2lkMyxst2mTU22FbGmF3Db6V+lcLVegoCIGZ4kvJnpCMN6Am9zCExEKC9vrXdTN1GA5mZ")! as CFData)! + ] + + func pin(url: URL, completion: @escaping (Data?, URLResponse?, Error?) -> Void) { + let request = URLRequest(url: url) + let session = URLSession(configuration: .default, delegate: self, delegateQueue: nil) + let task = session.dataTask(with: request, completionHandler: completion) + task.resume() + } + +// func pin2(url: URL, completion: @escaping (Result) -> Void) { +// let request = URLRequest(url: url) +// let session = URLSession(configuration: .default, delegate: self, delegateQueue: nil) +// let task = session.dataTask(with: request) { data, response, error in +// if let error = error { +// completion(.failure(error)) +// } else if let data = data { +// completion(.success(data)) +// } else { +// completion(.failure(URLError(.badServerResponse))) +// } +// } +// task.resume() +// } + +// func pin3(url: URL, completion: @escaping (Result) -> Void) { +// let request = URLRequest(url: url) +// let session = URLSession(configuration: .default, delegate: self, delegateQueue: nil) +// let task = session.dataTask(with: request) { data, response, error in +// if let error = error { +// completion(.failure(error)) +// return +// } +// +// guard let data = data else { +// completion(.failure(URLError(.cannotParseResponse))) +// return +// } +// +// do { +// let assetInfo = try JSONDecoder().decode(GDMFAssetInfo.self, from: data) +// completion(.success(assetInfo)) +// } catch { +// completion(.failure(error)) +// } +// } +// task.resume() +// } + +// func fetchAssetInfoSynchronously(urlString: String, completion: @escaping (GDMFAssetInfo?, Error?) -> Void) { +// guard let url = URL(string: urlString) else { +// completion(nil, URLError(.badURL)) +// return +// } +// +// DispatchQueue.global(qos: .userInitiated).async { +// let semaphore = DispatchSemaphore(value: 0) +// var assetInfo: GDMFAssetInfo? +// var requestError: Error? +// +// let request = URLRequest(url: url) +// let task = URLSession.shared.dataTask(with: request) { data, response, error in +// defer { semaphore.signal() } +// +// if let error = error { +// requestError = error +// return +// } +// +// guard let data = data else { +// requestError = URLError(.badServerResponse) +// return +// } +// +// do { +// // print(String(data: data, encoding: .utf8)) +// assetInfo = try JSONDecoder().decode(GDMFAssetInfo.self, from: data) +// } catch { +// requestError = error +// } +// } +// +// task.resume() +// semaphore.wait() // This will block the background thread until the semaphore is signaled +// +// DispatchQueue.main.async { +// completion(assetInfo, requestError) +// } +// } +// } +} + +extension GDMFPinnedSSL: URLSessionDelegate { + func urlSession(_ session: URLSession, didReceive challenge: URLAuthenticationChallenge, completionHandler: @escaping (URLSession.AuthChallengeDisposition, URLCredential?) -> Void) { + // Check if the certificate is trusted + if let serverTrust = challenge.protectionSpace.serverTrust, + SecTrustGetCertificateCount(serverTrust) > 0 { + if SecTrustGetCertificateCount(serverTrust) > 1 { + // Convert certificate stores to maps so they can be compared + let trustedCertificatesData = trustedCertificates.map { SecCertificateCopyData($0) as Data } + let serverCertificatesArray = SecTrustCopyCertificateChain(serverTrust)! as! [SecCertificate] + let serverCertificatesData = serverCertificatesArray.map { SecCertificateCopyData($0) as Data } + + if !trustedCertificatesData.filter(serverCertificatesData.contains).isEmpty { + completionHandler(.useCredential, URLCredential(trust: serverTrust)) + return + } + } else { + // Single certs we just loop through the internal nudge trust and compare if any exist + let serverCertificate = SecTrustCopyCertificateChain(serverTrust)! + let serverCertificateData = SecCertificateCopyData(serverCertificate as! SecCertificate) as Data + + for trustedCertificate in trustedCertificates { + let trustedCertificateData = SecCertificateCopyData(trustedCertificate) as Data + if serverCertificateData == trustedCertificateData { + completionHandler(.useCredential, URLCredential(trust: serverTrust)) + return + } + } + } + } + // If the certificate is not trusted, cancel the request + completionHandler(.cancelAuthenticationChallenge, nil) + } +} diff --git a/Nudge/UI/Defaults.swift b/Nudge/UI/Defaults.swift index 11314570..324c4a89 100644 --- a/Nudge/UI/Defaults.swift +++ b/Nudge/UI/Defaults.swift @@ -80,6 +80,7 @@ class AppState: ObservableObject { @Published var userRequiredMinimumOSVersion = Globals.nudgeDefaults.object(forKey: "requiredMinimumOSVersion") as? String ?? "0.0" @Published var userSessionDeferrals = Globals.nudgeDefaults.object(forKey: "userSessionDeferrals") as? Int ?? 0 @Published var backgroundBlur = [BackgroundBlurWindowController]() + @Published var gdmfAssets: GDMFAssetInfo? @Published var screenCurrentlyLocked = false @Published var locale = Locale.current @Published var nudgeCustomEventDate = DateManager().getCurrentDate() diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 1eb101d7..959f3723 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -142,6 +142,9 @@ class AppDelegate: NSObject, NSApplicationDelegate { handleKeyboardEvents() handleApplicationLaunchesIfNeeded() checkFullScreenStateOnFirstLaunch() + DispatchQueue.main.asyncAfter(deadline: .now() + 2.0) { + print(nudgePrimaryState.gdmfAssets) + } } func applicationDidResignActive(_ notification: Notification) { @@ -174,6 +177,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { // Pre-Launch Logic func applicationWillFinishLaunching(_ notification: Notification) { // print("applicationWillFinishLaunching") + NetworkFileManager().getGDMFAssets() handleSMAppService() checkForBadProfilePath() handleCommandLineArguments() diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 9e91e115..cabd4e57 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -704,6 +704,90 @@ struct NetworkFileManager { } } + func getGDMFAssets() { + // Define the URL you want to pin to + if let url = URL(string: "https://gdmf.apple.com/v2/pmv") { + // Call the pin method + GDMFPinnedSSL.shared.pin(url: url) { data, response, error in + if let error = error { + print("Error: \(error.localizedDescription)") + } else if let data = data { + do { + let assetInfo = try GDMFAssetInfo(data: data) + // print(assetInfo) + DispatchQueue.main.async { + nudgePrimaryState.gdmfAssets = assetInfo + } + } catch { + print("Failed to decode JSON: \(error.localizedDescription)") + } + } else { + print("Unknown error") + } + } + } else { + print("Invalid URL") + } + } + +// func getGDMFAssets2() -> GDMFAssetInfo? { +// if let url = URL(string: "https://gdmf.apple.com/v2/pmv") { +// // Call the pin method +// GDMFPinnedSSL.shared.pin1(url: url) { result in +// switch result { +// case .success(let assetInfo): +// print(assetInfo) +// case .failure(let error): +// print("Error: \(error.localizedDescription)") +// } +// } +// } +// return nil +// } +// +// func getGDMFAssetsSync() -> GDMFAssetInfo? { +// var info: GDMFAssetInfo? +// GDMFPinnedSSL.shared.fetchAssetInfoSynchronously(urlString: "https://gdmf.apple.com/v2/pmv") { assetInfo, error in +// if let error = error { +// print("Error: \(error.localizedDescription)") +// } else if let assetInfo = assetInfo { +// print("Asset Info: \(assetInfo)") +// info = assetInfo +// } else { +// print("Unknown error occurred") +// } +// } +// return info +// } +// GDMFPinnedSSL.shared.pin(url: url) { data, response, error in +// // Ensure there is no error +// guard error == nil else { +// print("Error: \(error!.localizedDescription)") +// return nil +// } +// +// // Ensure there is data returned +// guard let data = data else { +// print("No data received") +// return nil +// } +// // Process the JSON data +// do { +// // Assuming `AssetInfo` is the struct model that conforms to Decodable +// let assetInfo = try JSONDecoder().decode(GDMFAssetInfo.self, from: data) +// // Now you have your model and can use it as needed +// return assetInfo +// } catch { +// print("Failed to decode JSON: \(error.localizedDescription)") +// return nil +// } +// } +// } else { +// print("invalid URL") +// return nil +// } +// } + func getBackupMajorUpgradeAppPath() -> String { switch VersionManager.getMajorRequiredNudgeOSVersion() { case 12: From f83eab8d36b9011c8eff609e488cff2a95bd239c Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 27 Mar 2024 17:15:53 -0500 Subject: [PATCH 009/141] return board-id for gdmf comparison --- Nudge/Utilities/Utils.swift | 48 +++++++++++++++++++++++++++++++++++++ 1 file changed, 48 insertions(+) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index cabd4e57..ae482f44 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -8,6 +8,7 @@ import AppKit import CoreMediaIO import Foundation +import IOKit #if canImport(ServiceManagement) import ServiceManagement #endif @@ -477,6 +478,49 @@ struct DateManager { } struct DeviceManager { + // print(DeviceManager().getBoardID()) + func getBoardID() -> String? { + var service: io_service_t = IOServiceGetMatchingService(kIOMainPortDefault, IOServiceMatching("IOPlatformExpertDevice")) + + defer { + IOObjectRelease(service) + } + + guard service != 0 else { + print("Failed to get IOPlatformExpertDevice service") + return nil + } + + guard let property = IORegistryEntryCreateCFProperty(service, "board-id" as CFString, kCFAllocatorDefault, 0)?.takeRetainedValue() else { + print("Failed to get board-id property") + return nil + } + +// print(CFGetTypeID(property)) +// print(CFStringGetTypeID()) +// if let propertyDescription = CFCopyTypeIDDescription(CFGetTypeID(property)) { +// print("Property type is:", propertyDescription) +// } + + // Check if the property is of type CFData + if CFGetTypeID(property) == CFDataGetTypeID(), let data = property as? Data { + // Attempt to convert the data to a string + let boardID = String(data: data, encoding: .utf8)?.trimmingCharacters(in: CharacterSet(charactersIn: "\0")) + return boardID + } else { + print("Failed to check board-id property") + return nil + } + } + + func getSysctlValue(for key: String) -> String? { + var size: size_t = 0 + sysctlbyname(key, nil, &size, nil, 0) + var value = [CChar](repeating: 0, count: size) + sysctlbyname(key, &value, &size, nil, 0) + return String(cString: value) + } + func getCPUTypeInt() -> Int { // https://stackoverflow.com/a/63539782 var cputype = cpu_type_t() @@ -485,6 +529,10 @@ struct DeviceManager { return result == -1 ? -1 : Int(cputype) } + func getHardwareModel() -> String { + getSysctlValue(for: "hw.model") ?? "" + } + func getCPUTypeString() -> String { // https://stackoverflow.com/a/63539782 let type = getCPUTypeInt() From ce83f3c5d16b282f60025e9aebd7de844b9f9958 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 27 Mar 2024 17:19:34 -0500 Subject: [PATCH 010/141] get rid of some compiler warnings --- Nudge/Utilities/Utils.swift | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index ae482f44..317e016e 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -238,7 +238,7 @@ struct CameraManager { var address = CMIOObjectPropertyAddress( mSelector: CMIOObjectPropertySelector(kCMIOObjectPropertyName), mScope: CMIOObjectPropertyScope(kCMIOObjectPropertyScopeGlobal), - mElement: CMIOObjectPropertyElement(kCMIOObjectPropertyElementMaster)) + mElement: CMIOObjectPropertyElement(kCMIOObjectPropertyElementMain)) var nameCFString: CFString? let propsize = UInt32(MemoryLayout>.size) @@ -260,7 +260,7 @@ struct CameraUtilities { var opa = CMIOObjectPropertyAddress( mSelector: CMIOObjectPropertySelector(kCMIOHardwarePropertyDevices), mScope: CMIOObjectPropertyScope(kCMIOObjectPropertyScopeGlobal), - mElement: CMIOObjectPropertyElement(kCMIOObjectPropertyElementMaster)) + mElement: CMIOObjectPropertyElement(kCMIOObjectPropertyElementMain)) var dataSize: UInt32 = 0 var dataUsed: UInt32 = 0 @@ -480,7 +480,7 @@ struct DateManager { struct DeviceManager { // print(DeviceManager().getBoardID()) func getBoardID() -> String? { - var service: io_service_t = IOServiceGetMatchingService(kIOMainPortDefault, IOServiceMatching("IOPlatformExpertDevice")) + let service: io_service_t = IOServiceGetMatchingService(kIOMainPortDefault, IOServiceMatching("IOPlatformExpertDevice")) defer { IOObjectRelease(service) @@ -570,7 +570,7 @@ struct DeviceManager { } private func getPropertyFromPlatformExpert(key: String) -> String? { - let platformExpert = IOServiceGetMatchingService(kIOMasterPortDefault, IOServiceMatching("IOPlatformExpertDevice")) + let platformExpert = IOServiceGetMatchingService(kIOMainPortDefault, IOServiceMatching("IOPlatformExpertDevice")) defer { IOObjectRelease(platformExpert) } guard platformExpert > 0, @@ -1156,7 +1156,7 @@ var cameras: [CameraManager] { var opa = CMIOObjectPropertyAddress( mSelector: CMIOObjectPropertySelector(kCMIOHardwarePropertyDevices), mScope: CMIOObjectPropertyScope(kCMIOObjectPropertyScopeGlobal), - mElement: CMIOObjectPropertyElement(kCMIOObjectPropertyElementMaster)) + mElement: CMIOObjectPropertyElement(kCMIOObjectPropertyElementMain)) var dataSize: UInt32 = 0 var dataUsed: UInt32 = 0 From 9a3e8a43c170069f433b3f2a3bbfd0e827512c5f Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Thu, 28 Mar 2024 15:04:22 -0500 Subject: [PATCH 011/141] support for apple silicon vs intel hardware for gdmf --- Nudge/UI/Defaults.swift | 2 ++ Nudge/UI/Main.swift | 21 ++++++++++++++++++++- Nudge/Utilities/Logger.swift | 1 + Nudge/Utilities/SoftwareUpdate.swift | 12 ++++++++++++ Nudge/Utilities/Utils.swift | 22 +++++++++++----------- 5 files changed, 46 insertions(+), 12 deletions(-) diff --git a/Nudge/UI/Defaults.swift b/Nudge/UI/Defaults.swift index 324c4a89..1026b157 100644 --- a/Nudge/UI/Defaults.swift +++ b/Nudge/UI/Defaults.swift @@ -26,6 +26,8 @@ struct Globals { static let configProfile = ConfigurationManager().getConfigurationAsProfile() static let nudgeDefaults = UserDefaults.standard static let nudgeJSONPreferences = NetworkFileManager().getNudgeJSONPreferences() + // Device Properties + var hardwareID: String = "" } struct Intervals { diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 959f3723..91ed77a0 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -143,7 +143,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { handleApplicationLaunchesIfNeeded() checkFullScreenStateOnFirstLaunch() DispatchQueue.main.asyncAfter(deadline: .now() + 2.0) { - print(nudgePrimaryState.gdmfAssets) + print(nudgePrimaryState.gdmfAssets as Any) } } @@ -178,6 +178,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { func applicationWillFinishLaunching(_ notification: Notification) { // print("applicationWillFinishLaunching") NetworkFileManager().getGDMFAssets() + self.getSoftwareUpdateDeviceID() handleSMAppService() checkForBadProfilePath() handleCommandLineArguments() @@ -552,6 +553,24 @@ class AppDelegate: NSObject, NSApplicationDelegate { SoftwareUpdate().download() } } + + func getSoftwareUpdateDeviceID() { + if DeviceManager().getCPUTypeString() == "Apple Silicon" { + // There is no local bridge + globals.hardwareID = DeviceManager().getIORegInfo(serviceTarget: "target-sub-type") ?? "Unknown" + } else { + // Attempt localbridge for T2, if it fails, it's likely a T1 or lower + let bridgeID = SoftwareUpdate().getSoftwareUpdateDeviceID() + let boardID = DeviceManager().getIORegInfo(serviceTarget: "board-id") + if bridgeID.isEmpty { + // Fallback to boardID for T1 + globals.hardwareID = boardID ?? "Unknown" + } else { + // T2 uses bridge ID for it's update brain via gdmf + globals.hardwareID = bridgeID + } + } + } } class WindowDelegate: NSObject, NSWindowDelegate { diff --git a/Nudge/Utilities/Logger.swift b/Nudge/Utilities/Logger.swift index 9bd156f5..5d3cd5a6 100644 --- a/Nudge/Utilities/Logger.swift +++ b/Nudge/Utilities/Logger.swift @@ -44,6 +44,7 @@ let loggingLog = LogManager.createLogger(category: "logging") let prefsProfileLog = LogManager.createLogger(category: "preferences-profile") let prefsJSONLog = LogManager.createLogger(category: "preferences-json") let uiLog = LogManager.createLogger(category: "user-interface") +let softwareupdateDeviceLog = LogManager.createLogger(category: "softwareupdate-device") let softwareupdateListLog = LogManager.createLogger(category: "softwareupdate-list") let softwareupdateDownloadLog = LogManager.createLogger(category: "softwareupdate-download") diff --git a/Nudge/Utilities/SoftwareUpdate.swift b/Nudge/Utilities/SoftwareUpdate.swift index fed909fa..7ebba8db 100644 --- a/Nudge/Utilities/SoftwareUpdate.swift +++ b/Nudge/Utilities/SoftwareUpdate.swift @@ -9,6 +9,18 @@ import Foundation import os class SoftwareUpdate { + func getSoftwareUpdateDeviceID() -> String { + let (output, error, exitCode) = runProcess(launchPath: "/usr/libexec/remotectl", arguments: ["get-property", "localbridge", "HWModel"]) + + if exitCode != 0 { + LogManager.error("Error assessing DeviceID: \(error)", logger: softwareupdateDeviceLog) + return error + } else { + LogManager.info("SoftwareUpdateDeviceID: \(output)", logger: softwareupdateDeviceLog) + return output + } + } + func list() -> String { let (output, error, exitCode) = runProcess(launchPath: "/usr/sbin/softwareupdate", arguments: ["--list", "--all"]) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 317e016e..918cf747 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -478,8 +478,7 @@ struct DateManager { } struct DeviceManager { - // print(DeviceManager().getBoardID()) - func getBoardID() -> String? { + func getIORegInfo(serviceTarget: String) -> String? { let service: io_service_t = IOServiceGetMatchingService(kIOMainPortDefault, IOServiceMatching("IOPlatformExpertDevice")) defer { @@ -487,12 +486,12 @@ struct DeviceManager { } guard service != 0 else { - print("Failed to get IOPlatformExpertDevice service") + LogManager.error("Failed to fetch IOPlatformExpertDevice service.", logger: utilsLog) return nil } - guard let property = IORegistryEntryCreateCFProperty(service, "board-id" as CFString, kCFAllocatorDefault, 0)?.takeRetainedValue() else { - print("Failed to get board-id property") + guard let property = IORegistryEntryCreateCFProperty(service, serviceTarget as CFString, kCFAllocatorDefault, 0)?.takeRetainedValue() else { + LogManager.error("Failed to fetch \(serviceTarget) property.", logger: utilsLog) return nil } @@ -505,10 +504,10 @@ struct DeviceManager { // Check if the property is of type CFData if CFGetTypeID(property) == CFDataGetTypeID(), let data = property as? Data { // Attempt to convert the data to a string - let boardID = String(data: data, encoding: .utf8)?.trimmingCharacters(in: CharacterSet(charactersIn: "\0")) - return boardID + let serviceTargetProperty = String(data: data, encoding: .utf8)?.trimmingCharacters(in: CharacterSet(charactersIn: "\0")) + return serviceTargetProperty } else { - print("Failed to check board-id property") + LogManager.error("Failed to check \(serviceTarget) property.", logger: utilsLog) return nil } } @@ -544,13 +543,13 @@ struct DeviceManager { switch cpuArch { case Int(CPU_TYPE_X86) /* Intel */: - LogManager.debug("CPU Type is Intel", logger: utilsLog) + LogManager.debug("CPU Type: Intel", logger: utilsLog) return "Intel" case Int(CPU_TYPE_ARM) /* Apple Silicon */: - LogManager.debug("CPU Type is Apple Silicon", logger: utilsLog) + LogManager.debug("CPU Type: Apple Silicon", logger: utilsLog) return "Apple Silicon" default: - LogManager.debug("Unknown CPU Type", logger: utilsLog) + LogManager.debug("CPU Type: Unknown", logger: utilsLog) return "unknown" } } @@ -765,6 +764,7 @@ struct NetworkFileManager { // print(assetInfo) DispatchQueue.main.async { nudgePrimaryState.gdmfAssets = assetInfo + // perhaps process this as a lookup for all OS versions a board ID can contain } } catch { print("Failed to decode JSON: \(error.localizedDescription)") From 16426c29e45b13a259e72d0a85c21fe122e14ab3 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 29 Mar 2024 15:38:46 -0500 Subject: [PATCH 012/141] refactor gdmf and hardmodel --- Nudge/3rd Party Assets/gdmf.swift | 116 ++++-------- Nudge/UI/Defaults.swift | 4 +- Nudge/UI/Main.swift | 23 --- Nudge/Utilities/SoftwareUpdate.swift | 44 +---- Nudge/Utilities/Utils.swift | 257 ++++++++++++++------------- 5 files changed, 175 insertions(+), 269 deletions(-) diff --git a/Nudge/3rd Party Assets/gdmf.swift b/Nudge/3rd Party Assets/gdmf.swift index c548b1f8..1baeb122 100644 --- a/Nudge/3rd Party Assets/gdmf.swift +++ b/Nudge/3rd Party Assets/gdmf.swift @@ -98,93 +98,47 @@ class GDMFPinnedSSL: NSObject { SecCertificateCreateWithData(nil, Data(base64Encoded: "MIID+DCCAuCgAwIBAgIII2l0BK3LgxQwDQYJKoZIhvcNAQELBQAwYjELMAkGA1UEBhMCVVMxEzARBgNVBAoTCkFwcGxlIEluYy4xJjAkBgNVBAsTHUFwcGxlIENlcnRpZmljYXRpb24gQXV0aG9yaXR5MRYwFAYDVQQDEw1BcHBsZSBSb290IENBMB4XDTE0MDMwODAxNTMwNFoXDTI5MDMwODAxNTMwNFowbTEnMCUGA1UEAwweQXBwbGUgU2VydmVyIEF1dGhlbnRpY2F0aW9uIENBMSAwHgYDVQQLDBdDZXJ0aWZpY2F0aW9uIEF1dGhvcml0eTETMBEGA1UECgwKQXBwbGUgSW5jLjELMAkGA1UEBhMCVVMwggEiMA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQC5Jhawy4ercRWSjt+qPuGA11O6pGDMfIVy9zB8CU9XDUr/4V7JS1ATAmSxvTk10dcEUcEY+iL6rt+YGNa/Tk1DEPoliJ/TQIV25SKBtlRFc5qL45xIGoZ6w1Hi2pX4pH3bMN5sDsTF9WyY56b6VyAdGXN6Ds1jD7cniC7hmmiCuEBsYxYkZivnsuJUfeeIOaIbgT4C0znYl3dKMgzWCgqzBJvxcm9jqBUebDfoD9tTkNYpXLxqV5tGeAo+JOqaP6HYP/XbbqhsgrXdmTjsklaUpsVzJtGuCLLGUueOdkuJuFQPbuDZQtsqZYdGFLuWuFe7UeaEE/cNobaJrHzRIXSrAgMBAAGjgaYwgaMwHQYDVR0OBBYEFCzFbVLdMe+M7AiB7d/cykMARQHQMA8GA1UdEwEB/wQFMAMBAf8wHwYDVR0jBBgwFoAUK9BpR5R2Cf70a40uQKb3R01/CF4wLgYDVR0fBCcwJTAjoCGgH4YdaHR0cDovL2NybC5hcHBsZS5jb20vcm9vdC5jcmwwDgYDVR0PAQH/BAQDAgEGMBAGCiqGSIb3Y2QGAgwEAgUAMA0GCSqGSIb3DQEBCwUAA4IBAQAj8QZ+UEGBol7TcKRJka/YzGeMoSV9xJqTOS/YafsbQVtE19lryzslCRry9OPHnOiwW/Df3SIlERWTuUle2gxmel7Xb/Bj1GWMxHpUfVZPZZr92sSyyLC4oct94EeoQBW4FhntW2GO36rQzdI6wH46nyJO39/0ThrNk//Q8EVVZDM+1OXaaKATinYwJ9S/+B529vnDAO+xg+pTbVw1xw0HAbr4Ybn+xZprQ2GBA+u6X3Cd6G+UJEvczpKoLqI1PONJ4BZ3otxruY0YQrk2lkMyxst2mTU22FbGmF3Db6V+lcLVegoCIGZ4kvJnpCMN6Am9zCExEKC9vrXdTN1GA5mZ")! as CFData)! ] - func pin(url: URL, completion: @escaping (Data?, URLResponse?, Error?) -> Void) { + func pinAsynch(url: URL, completion: @escaping (Data?, URLResponse?, Error?) -> Void) { let request = URLRequest(url: url) let session = URLSession(configuration: .default, delegate: self, delegateQueue: nil) let task = session.dataTask(with: request, completionHandler: completion) task.resume() } -// func pin2(url: URL, completion: @escaping (Result) -> Void) { -// let request = URLRequest(url: url) -// let session = URLSession(configuration: .default, delegate: self, delegateQueue: nil) -// let task = session.dataTask(with: request) { data, response, error in -// if let error = error { -// completion(.failure(error)) -// } else if let data = data { -// completion(.success(data)) -// } else { -// completion(.failure(URLError(.badServerResponse))) -// } -// } -// task.resume() -// } - -// func pin3(url: URL, completion: @escaping (Result) -> Void) { -// let request = URLRequest(url: url) -// let session = URLSession(configuration: .default, delegate: self, delegateQueue: nil) -// let task = session.dataTask(with: request) { data, response, error in -// if let error = error { -// completion(.failure(error)) -// return -// } -// -// guard let data = data else { -// completion(.failure(URLError(.cannotParseResponse))) -// return -// } -// -// do { -// let assetInfo = try JSONDecoder().decode(GDMFAssetInfo.self, from: data) -// completion(.success(assetInfo)) -// } catch { -// completion(.failure(error)) -// } -// } -// task.resume() -// } - -// func fetchAssetInfoSynchronously(urlString: String, completion: @escaping (GDMFAssetInfo?, Error?) -> Void) { -// guard let url = URL(string: urlString) else { -// completion(nil, URLError(.badURL)) -// return -// } -// -// DispatchQueue.global(qos: .userInitiated).async { -// let semaphore = DispatchSemaphore(value: 0) -// var assetInfo: GDMFAssetInfo? -// var requestError: Error? -// -// let request = URLRequest(url: url) -// let task = URLSession.shared.dataTask(with: request) { data, response, error in -// defer { semaphore.signal() } -// -// if let error = error { -// requestError = error -// return -// } -// -// guard let data = data else { -// requestError = URLError(.badServerResponse) -// return -// } -// -// do { -// // print(String(data: data, encoding: .utf8)) -// assetInfo = try JSONDecoder().decode(GDMFAssetInfo.self, from: data) -// } catch { -// requestError = error -// } -// } -// -// task.resume() -// semaphore.wait() // This will block the background thread until the semaphore is signaled -// -// DispatchQueue.main.async { -// completion(assetInfo, requestError) -// } -// } -// } + func pinSync(url: URL, maxRetries: Int = 3) -> (data: Data?, response: URLResponse?, error: Error?) { + let semaphore = DispatchSemaphore(value: 0) + let request = URLRequest(url: url) + let session = URLSession(configuration: .default, delegate: self, delegateQueue: nil) + var attempts = 0 + + var responseData: Data? + var response: URLResponse? + var responseError: Error? + + // Retry loop + while attempts < maxRetries { + attempts += 1 + let task = session.dataTask(with: request) { data, resp, error in + responseData = data + response = resp + responseError = error + semaphore.signal() + } + task.resume() + + semaphore.wait() + + // Break the loop if the task succeeded or return an error other than a timeout + if responseError == nil || (responseError! as NSError).code != NSURLErrorTimedOut { + break + } else if attempts < maxRetries { + // Reset the error to try again + responseError = nil + } + } + + return (responseData, response, responseError) + } } extension GDMFPinnedSSL: URLSessionDelegate { diff --git a/Nudge/UI/Defaults.swift b/Nudge/UI/Defaults.swift index 1026b157..5850cdab 100644 --- a/Nudge/UI/Defaults.swift +++ b/Nudge/UI/Defaults.swift @@ -27,7 +27,8 @@ struct Globals { static let nudgeDefaults = UserDefaults.standard static let nudgeJSONPreferences = NetworkFileManager().getNudgeJSONPreferences() // Device Properties - var hardwareID: String = "" + static let gdmfAssets = NetworkFileManager().getGDMFAssets() + static let hardwareModelID = DeviceManager().getHardwareModelID() } struct Intervals { @@ -82,7 +83,6 @@ class AppState: ObservableObject { @Published var userRequiredMinimumOSVersion = Globals.nudgeDefaults.object(forKey: "requiredMinimumOSVersion") as? String ?? "0.0" @Published var userSessionDeferrals = Globals.nudgeDefaults.object(forKey: "userSessionDeferrals") as? Int ?? 0 @Published var backgroundBlur = [BackgroundBlurWindowController]() - @Published var gdmfAssets: GDMFAssetInfo? @Published var screenCurrentlyLocked = false @Published var locale = Locale.current @Published var nudgeCustomEventDate = DateManager().getCurrentDate() diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 91ed77a0..1eb101d7 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -142,9 +142,6 @@ class AppDelegate: NSObject, NSApplicationDelegate { handleKeyboardEvents() handleApplicationLaunchesIfNeeded() checkFullScreenStateOnFirstLaunch() - DispatchQueue.main.asyncAfter(deadline: .now() + 2.0) { - print(nudgePrimaryState.gdmfAssets as Any) - } } func applicationDidResignActive(_ notification: Notification) { @@ -177,8 +174,6 @@ class AppDelegate: NSObject, NSApplicationDelegate { // Pre-Launch Logic func applicationWillFinishLaunching(_ notification: Notification) { // print("applicationWillFinishLaunching") - NetworkFileManager().getGDMFAssets() - self.getSoftwareUpdateDeviceID() handleSMAppService() checkForBadProfilePath() handleCommandLineArguments() @@ -553,24 +548,6 @@ class AppDelegate: NSObject, NSApplicationDelegate { SoftwareUpdate().download() } } - - func getSoftwareUpdateDeviceID() { - if DeviceManager().getCPUTypeString() == "Apple Silicon" { - // There is no local bridge - globals.hardwareID = DeviceManager().getIORegInfo(serviceTarget: "target-sub-type") ?? "Unknown" - } else { - // Attempt localbridge for T2, if it fails, it's likely a T1 or lower - let bridgeID = SoftwareUpdate().getSoftwareUpdateDeviceID() - let boardID = DeviceManager().getIORegInfo(serviceTarget: "board-id") - if bridgeID.isEmpty { - // Fallback to boardID for T1 - globals.hardwareID = boardID ?? "Unknown" - } else { - // T2 uses bridge ID for it's update brain via gdmf - globals.hardwareID = bridgeID - } - } - } } class WindowDelegate: NSObject, NSWindowDelegate { diff --git a/Nudge/Utilities/SoftwareUpdate.swift b/Nudge/Utilities/SoftwareUpdate.swift index 7ebba8db..c3c551e1 100644 --- a/Nudge/Utilities/SoftwareUpdate.swift +++ b/Nudge/Utilities/SoftwareUpdate.swift @@ -9,20 +9,8 @@ import Foundation import os class SoftwareUpdate { - func getSoftwareUpdateDeviceID() -> String { - let (output, error, exitCode) = runProcess(launchPath: "/usr/libexec/remotectl", arguments: ["get-property", "localbridge", "HWModel"]) - - if exitCode != 0 { - LogManager.error("Error assessing DeviceID: \(error)", logger: softwareupdateDeviceLog) - return error - } else { - LogManager.info("SoftwareUpdateDeviceID: \(output)", logger: softwareupdateDeviceLog) - return output - } - } - func list() -> String { - let (output, error, exitCode) = runProcess(launchPath: "/usr/sbin/softwareupdate", arguments: ["--list", "--all"]) + let (output, error, exitCode) = SubProcessUtilities().runProcess(launchPath: "/usr/sbin/softwareupdate", arguments: ["--list", "--all"]) if exitCode != 0 { LogManager.error("Error listing software updates: \(error)", logger: softwareupdateListLog) @@ -46,7 +34,7 @@ class SoftwareUpdate { if OptionalFeatureVariables.attemptToFetchMajorUpgrade, !majorUpgradeAppPathExists, !majorUpgradeBackupAppPathExists { LogManager.notice("Device requires major upgrade - attempting download", logger: softwareupdateListLog) - let (output, error, exitCode) = runProcess(launchPath: "/usr/sbin/softwareupdate", arguments: ["--fetch-full-installer", "--full-installer-version", OSVersionRequirementVariables.requiredMinimumOSVersion]) + let (output, error, exitCode) = SubProcessUtilities().runProcess(launchPath: "/usr/sbin/softwareupdate", arguments: ["--fetch-full-installer", "--full-installer-version", OSVersionRequirementVariables.requiredMinimumOSVersion]) if exitCode != 0 { LogManager.error("Error downloading software update: \(error)", logger: softwareupdateDownloadLog) @@ -72,7 +60,7 @@ class SoftwareUpdate { } LogManager.notice("Software update found \(updateLabel) available for download - attempting download", logger: softwareupdateListLog) - let (output, error, exitCode) = runProcess(launchPath: "/usr/sbin/softwareupdate", arguments: ["--download", updateLabel]) + let (output, error, exitCode) = SubProcessUtilities().runProcess(launchPath: "/usr/sbin/softwareupdate", arguments: ["--download", updateLabel]) if exitCode != 0 { LogManager.error("Error downloading software updates: \(error)", logger: softwareupdateDownloadLog) @@ -98,30 +86,4 @@ class SoftwareUpdate { return updateLabel ?? "" } - - private func runProcess(launchPath: String, arguments: [String]) -> (output: String, error: String, exitCode: Int32) { - let task = Process() - task.launchPath = launchPath - task.arguments = arguments - - let outputPipe = Pipe() - let errorPipe = Pipe() - task.standardOutput = outputPipe - task.standardError = errorPipe - - do { - try task.run() - } catch { - return ("", "Error running process", -1) - } - - task.waitUntilExit() - - let outputData = outputPipe.fileHandleForReading.readDataToEndOfFile() - let errorData = errorPipe.fileHandleForReading.readDataToEndOfFile() - let output = String(decoding: outputData, as: UTF8.self) - let error = String(decoding: errorData, as: UTF8.self) - - return (output, error, task.terminationStatus) - } } diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 918cf747..cd06a085 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -478,48 +478,6 @@ struct DateManager { } struct DeviceManager { - func getIORegInfo(serviceTarget: String) -> String? { - let service: io_service_t = IOServiceGetMatchingService(kIOMainPortDefault, IOServiceMatching("IOPlatformExpertDevice")) - - defer { - IOObjectRelease(service) - } - - guard service != 0 else { - LogManager.error("Failed to fetch IOPlatformExpertDevice service.", logger: utilsLog) - return nil - } - - guard let property = IORegistryEntryCreateCFProperty(service, serviceTarget as CFString, kCFAllocatorDefault, 0)?.takeRetainedValue() else { - LogManager.error("Failed to fetch \(serviceTarget) property.", logger: utilsLog) - return nil - } - -// print(CFGetTypeID(property)) -// print(CFStringGetTypeID()) -// if let propertyDescription = CFCopyTypeIDDescription(CFGetTypeID(property)) { -// print("Property type is:", propertyDescription) -// } - - // Check if the property is of type CFData - if CFGetTypeID(property) == CFDataGetTypeID(), let data = property as? Data { - // Attempt to convert the data to a string - let serviceTargetProperty = String(data: data, encoding: .utf8)?.trimmingCharacters(in: CharacterSet(charactersIn: "\0")) - return serviceTargetProperty - } else { - LogManager.error("Failed to check \(serviceTarget) property.", logger: utilsLog) - return nil - } - } - - func getSysctlValue(for key: String) -> String? { - var size: size_t = 0 - sysctlbyname(key, nil, &size, nil, 0) - var value = [CChar](repeating: 0, count: size) - sysctlbyname(key, &value, &size, nil, 0) - return String(cString: value) - } - func getCPUTypeInt() -> Int { // https://stackoverflow.com/a/63539782 var cputype = cpu_type_t() @@ -528,8 +486,16 @@ struct DeviceManager { return result == -1 ? -1 : Int(cputype) } - func getHardwareModel() -> String { - getSysctlValue(for: "hw.model") ?? "" + func getBridgeModelID() -> String { + let (output, error, exitCode) = SubProcessUtilities().runProcess(launchPath: "/usr/libexec/remotectl", arguments: ["get-property", "localbridge", "HWModel"]) + + if exitCode != 0 { + LogManager.error("Error assessing DeviceID: \(error)", logger: softwareupdateDeviceLog) + return "" + } else { + LogManager.info("SoftwareUpdateDeviceID: \(output)", logger: softwareupdateDeviceLog) + return output + } } func getCPUTypeString() -> String { @@ -554,6 +520,31 @@ struct DeviceManager { } } + func getHardwareModel() -> String { + getSysctlValue(for: "hw.model") ?? "" + } + + func getHardwareModelID() -> String { + var hardwareModelID = "" + if DeviceManager().getCPUTypeString() == "Apple Silicon" { + // There is no local bridge + hardwareModelID = getIORegInfo(serviceTarget: "target-sub-type") ?? "Unknown" + } else { + // Attempt localbridge for T2, if it fails, it's likely a T1 or lower + let bridgeID = getBridgeModelID() + let boardID = getIORegInfo(serviceTarget: "board-id") + if bridgeID.isEmpty { + // Fallback to boardID for T1 + hardwareModelID = boardID ?? "Unknown" + } else { + // T2 uses bridge ID for it's update brain via gdmf + hardwareModelID = bridgeID + } + } + LogManager.debug("Hardware Model ID: \(hardwareModelID)", logger: utilsLog) + return hardwareModelID + } + func getHardwareUUID() -> String { guard !CommandLineUtilities().demoModeEnabled(), !CommandLineUtilities().unitTestingEnabled() else { @@ -562,6 +553,40 @@ struct DeviceManager { return getPropertyFromPlatformExpert(key: String(kIOPlatformUUIDKey)) ?? "" } + func getIORegInfo(serviceTarget: String) -> String? { + let service: io_service_t = IOServiceGetMatchingService(kIOMainPortDefault, IOServiceMatching("IOPlatformExpertDevice")) + + defer { + IOObjectRelease(service) + } + + guard service != 0 else { + LogManager.error("Failed to fetch IOPlatformExpertDevice service.", logger: utilsLog) + return nil + } + + guard let property = IORegistryEntryCreateCFProperty(service, serviceTarget as CFString, kCFAllocatorDefault, 0)?.takeRetainedValue() else { + LogManager.error("Failed to fetch \(serviceTarget) property.", logger: utilsLog) + return nil + } + + // print(CFGetTypeID(property)) + // print(CFStringGetTypeID()) + // if let propertyDescription = CFCopyTypeIDDescription(CFGetTypeID(property)) { + // print("Property type is:", propertyDescription) + // } + + // Check if the property is of type CFData + if CFGetTypeID(property) == CFDataGetTypeID(), let data = property as? Data { + // Attempt to convert the data to a string + let serviceTargetProperty = String(data: data, encoding: .utf8)?.trimmingCharacters(in: CharacterSet(charactersIn: "\0")) + return serviceTargetProperty + } else { + LogManager.error("Failed to check \(serviceTarget) property.", logger: utilsLog) + return nil + } + } + func getPatchOSVersion() -> Int { let PatchOSVersion = ProcessInfo().operatingSystemVersion.patchVersion LogManager.info("Patch OS Version: \(PatchOSVersion)", logger: utilsLog) @@ -587,6 +612,14 @@ struct DeviceManager { return getPropertyFromPlatformExpert(key: String(kIOPlatformSerialNumberKey)) ?? "" } + func getSysctlValue(for key: String) -> String? { + var size: size_t = 0 + sysctlbyname(key, nil, &size, nil, 0) + var value = [CChar](repeating: 0, count: size) + sysctlbyname(key, &value, &size, nil, 0) + return String(cString: value) + } + func getSystemConsoleUsername() -> String { var uid: uid_t = 0 var gid: gid_t = 0 @@ -751,90 +784,42 @@ struct NetworkFileManager { } } - func getGDMFAssets() { + func getGDMFAssets() -> GDMFAssetInfo? { // Define the URL you want to pin to if let url = URL(string: "https://gdmf.apple.com/v2/pmv") { // Call the pin method - GDMFPinnedSSL.shared.pin(url: url) { data, response, error in - if let error = error { - print("Error: \(error.localizedDescription)") - } else if let data = data { - do { - let assetInfo = try GDMFAssetInfo(data: data) - // print(assetInfo) - DispatchQueue.main.async { - nudgePrimaryState.gdmfAssets = assetInfo - // perhaps process this as a lookup for all OS versions a board ID can contain - } - } catch { - print("Failed to decode JSON: \(error.localizedDescription)") - } - } else { - print("Unknown error") + // Async Method + // GDMFPinnedSSL.shared.pinAsync(url: url) { data, response, error in + // if let error = error { + // print("Error: \(error.localizedDescription)") + // } else if let data = data { + // do { + // let assetInfo = try GDMFAssetInfo(data: data) + // return assetInfo + // } catch { + // print("Failed to decode JSON: \(error.localizedDescription)") + // } + // } else { + // print("Unknown error") + // } + // } + // Sync Method + let gdmfData = GDMFPinnedSSL.shared.pinSync(url: url) + if (gdmfData.error == nil) { + do { + let assetInfo = try GDMFAssetInfo(data: gdmfData.data!) + return assetInfo + } catch { + LogManager.error("Failed to decode gdmf JSON: \(error.localizedDescription)", logger: utilsLog) } + } else { + LogManager.error("Failed to fetch gdmf JSON: \(gdmfData.error!.localizedDescription)", logger: utilsLog) } } else { - print("Invalid URL") - } - } - -// func getGDMFAssets2() -> GDMFAssetInfo? { -// if let url = URL(string: "https://gdmf.apple.com/v2/pmv") { -// // Call the pin method -// GDMFPinnedSSL.shared.pin1(url: url) { result in -// switch result { -// case .success(let assetInfo): -// print(assetInfo) -// case .failure(let error): -// print("Error: \(error.localizedDescription)") -// } -// } -// } -// return nil -// } -// -// func getGDMFAssetsSync() -> GDMFAssetInfo? { -// var info: GDMFAssetInfo? -// GDMFPinnedSSL.shared.fetchAssetInfoSynchronously(urlString: "https://gdmf.apple.com/v2/pmv") { assetInfo, error in -// if let error = error { -// print("Error: \(error.localizedDescription)") -// } else if let assetInfo = assetInfo { -// print("Asset Info: \(assetInfo)") -// info = assetInfo -// } else { -// print("Unknown error occurred") -// } -// } -// return info -// } -// GDMFPinnedSSL.shared.pin(url: url) { data, response, error in -// // Ensure there is no error -// guard error == nil else { -// print("Error: \(error!.localizedDescription)") -// return nil -// } -// -// // Ensure there is data returned -// guard let data = data else { -// print("No data received") -// return nil -// } -// // Process the JSON data -// do { -// // Assuming `AssetInfo` is the struct model that conforms to Decodable -// let assetInfo = try JSONDecoder().decode(GDMFAssetInfo.self, from: data) -// // Now you have your model and can use it as needed -// return assetInfo -// } catch { -// print("Failed to decode JSON: \(error.localizedDescription)") -// return nil -// } -// } -// } else { -// print("invalid URL") -// return nil -// } -// } + LogManager.error("Failed to decode gdmf JSON URL string", logger: utilsLog) + } + return nil + } func getBackupMajorUpgradeAppPath() -> String { switch VersionManager.getMajorRequiredNudgeOSVersion() { @@ -875,6 +860,34 @@ struct NetworkFileManager { } } +struct SubProcessUtilities { + func runProcess(launchPath: String, arguments: [String]) -> (output: String, error: String, exitCode: Int32) { + let task = Process() + task.launchPath = launchPath + task.arguments = arguments + + let outputPipe = Pipe() + let errorPipe = Pipe() + task.standardOutput = outputPipe + task.standardError = errorPipe + + do { + try task.run() + } catch { + return ("", "Error running process", -1) + } + + task.waitUntilExit() + + let outputData = outputPipe.fileHandleForReading.readDataToEndOfFile() + let errorData = errorPipe.fileHandleForReading.readDataToEndOfFile() + let output = String(decoding: outputData, as: UTF8.self) + let error = String(decoding: errorData, as: UTF8.self) + + return (output, error, task.terminationStatus) + } +} + struct SMAppManager { private func handleLegacyLaunchAgent(passedThroughCLI: Bool, action: String) { logOrPrint("Legacy Nudge LaunchAgent currently loaded. Please disable this agent before attempting to \(action) modern agent.", passedThroughCLI: passedThroughCLI, exitCode: 1) From 632451072d6f8365558bc79b1dfbe30323620d35 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 29 Mar 2024 16:39:19 -0500 Subject: [PATCH 013/141] strip newlines in modelID --- Nudge/Utilities/Utils.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index cd06a085..2ddb0985 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -542,7 +542,7 @@ struct DeviceManager { } } LogManager.debug("Hardware Model ID: \(hardwareModelID)", logger: utilsLog) - return hardwareModelID + return hardwareModelID.trimmingCharacters(in: .whitespacesAndNewlines) } func getHardwareUUID() -> String { From 9fa46e2cf2193a49d0fbfc3b78b78d15b1c12fcb Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 29 Mar 2024 16:39:36 -0500 Subject: [PATCH 014/141] Remove optional logging in serviceTargetProperty --- Nudge/Utilities/Utils.swift | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 2ddb0985..5133c9e9 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -579,8 +579,11 @@ struct DeviceManager { // Check if the property is of type CFData if CFGetTypeID(property) == CFDataGetTypeID(), let data = property as? Data { // Attempt to convert the data to a string - let serviceTargetProperty = String(data: data, encoding: .utf8)?.trimmingCharacters(in: CharacterSet(charactersIn: "\0")) - return serviceTargetProperty + if let serviceTargetProperty = String(data: data, encoding: .utf8)?.trimmingCharacters(in: CharacterSet(charactersIn: "\0")) { + LogManager.debug("\(serviceTarget): \(String(describing: serviceTargetProperty))", logger: utilsLog) + return serviceTargetProperty + } + return nil } else { LogManager.error("Failed to check \(serviceTarget) property.", logger: utilsLog) return nil From 05e052e0ebc9cc6c6f695b523c5477b8e8e65c58 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 29 Mar 2024 16:39:55 -0500 Subject: [PATCH 015/141] Add a temporary comparison loop of the macOS Assets from gdmf --- Nudge/UI/Main.swift | 17 +++++++++++++++++ 1 file changed, 17 insertions(+) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 1eb101d7..d8098e5f 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -174,6 +174,23 @@ class AppDelegate: NSObject, NSApplicationDelegate { // Pre-Launch Logic func applicationWillFinishLaunching(_ notification: Notification) { // print("applicationWillFinishLaunching") + if let macOSAssets = Globals.gdmfAssets?.publicAssetSets.macOS { + // Find the first macOS asset that matches the product version + var tempRequiredMinOSVersion = PrefsWrapper.requiredMinimumOSVersion + tempRequiredMinOSVersion = "14.4.1" // Hack so I can test this since 14.99.99 doesn't exist in GDMF + if let matchingAsset = macOSAssets.first(where: { $0.productVersion == tempRequiredMinOSVersion }) { + // Check if the specified device is in the supported devices of the matching asset + print("GDMF Matched OS Version: \(matchingAsset.productVersion)") + print("GDMF Assets: \(matchingAsset.supportedDevices)") + print("Assessed Model ID: \(Globals.hardwareModelID)") + print("DeviceGDMFSupported: \(matchingAsset.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }))") + } else { + // If no matching product version found or the device is not supported, return false + print("DeviceGDMFSupported: False") + } + } else { + print("No macOS assets available.") + } handleSMAppService() checkForBadProfilePath() handleCommandLineArguments() From 17853d2b051b5b5b3576a0eb518b2e1d9864fdc2 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 14 May 2024 12:53:48 -0500 Subject: [PATCH 016/141] initial query of sofa v1 api --- Nudge.xcodeproj/project.pbxproj | 4 + Nudge/3rd Party Assets/sofa.swift | 253 ++++++++++++++++++++++++++++++ Nudge/UI/Defaults.swift | 1 + Nudge/UI/Main.swift | 32 +++- Nudge/Utilities/Utils.swift | 22 ++- 5 files changed, 307 insertions(+), 5 deletions(-) create mode 100644 Nudge/3rd Party Assets/sofa.swift diff --git a/Nudge.xcodeproj/project.pbxproj b/Nudge.xcodeproj/project.pbxproj index d3296dad..9f563e03 100644 --- a/Nudge.xcodeproj/project.pbxproj +++ b/Nudge.xcodeproj/project.pbxproj @@ -18,6 +18,7 @@ 5836861425DACFE90004514C /* Logger.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5836861325DACFE90004514C /* Logger.swift */; }; 5836861C25DAD01C0004514C /* SoftwareUpdate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5836861B25DAD01C0004514C /* SoftwareUpdate.swift */; }; 6316F0E72832CA0700E1354D /* Schema in Resources */ = {isa = PBXBuildFile; fileRef = 6316F0E62832CA0700E1354D /* Schema */; }; + 631A6D762BF2654000DC1EF3 /* sofa.swift in Sources */ = {isa = PBXBuildFile; fileRef = 631A6D752BF2654000DC1EF3 /* sofa.swift */; }; 6347351D2B45DC2400C3401D /* CloseButton.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6347351C2B45DC2400C3401D /* CloseButton.swift */; }; 634CE1092BB47480002C26C4 /* gdmf.swift in Sources */ = {isa = PBXBuildFile; fileRef = 634CE1082BB47480002C26C4 /* gdmf.swift */; }; 634CE10A2BB47480002C26C4 /* gdmf.swift in Sources */ = {isa = PBXBuildFile; fileRef = 634CE1082BB47480002C26C4 /* gdmf.swift */; }; @@ -93,6 +94,7 @@ 5836861325DACFE90004514C /* Logger.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Logger.swift; sourceTree = ""; }; 5836861B25DAD01C0004514C /* SoftwareUpdate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SoftwareUpdate.swift; sourceTree = ""; }; 6316F0E62832CA0700E1354D /* Schema */ = {isa = PBXFileReference; lastKnownFileType = folder; path = Schema; sourceTree = ""; }; + 631A6D752BF2654000DC1EF3 /* sofa.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = sofa.swift; sourceTree = ""; }; 6347351C2B45DC2400C3401D /* CloseButton.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CloseButton.swift; sourceTree = ""; }; 634CE1082BB47480002C26C4 /* gdmf.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = gdmf.swift; sourceTree = ""; }; 636B9C0126CACCAB0007BE3B /* DeferView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DeferView.swift; sourceTree = ""; }; @@ -185,6 +187,7 @@ isa = PBXGroup; children = ( 634CE1082BB47480002C26C4 /* gdmf.swift */, + 631A6D752BF2654000DC1EF3 /* sofa.swift */, ); path = "3rd Party Assets"; sourceTree = ""; @@ -495,6 +498,7 @@ 639B6B6E25DF3C3F00E38EC1 /* SimpleMode.swift in Sources */, 639E198225CD885D008F618B /* PreferencesStructure.swift in Sources */, 639B6B3B25DF200C00E38EC1 /* Preferences.swift in Sources */, + 631A6D762BF2654000DC1EF3 /* sofa.swift in Sources */, 63D7D0E325C9E9A400236281 /* Main.swift in Sources */, 639B6B6025DF37F000E38EC1 /* ScreenShotZoom.swift in Sources */, 41AD2B0226DE6947004C52B1 /* AdditionalInfoButton.swift in Sources */, diff --git a/Nudge/3rd Party Assets/sofa.swift b/Nudge/3rd Party Assets/sofa.swift new file mode 100644 index 00000000..58d46bc1 --- /dev/null +++ b/Nudge/3rd Party Assets/sofa.swift @@ -0,0 +1,253 @@ +// +// sofa.swift +// Nudge +// +// Created by Erik Gomez on 5/13/24. +// + +import Foundation + +struct MacOSDataFeed: Codable { + let updateHash: String + let osVersions: [SofaOSVersion] + let xProtectPayloads: XProtectPayloads + let xProtectPlistConfigData: XProtectPlistConfigData + let models: [String: ModelInfo] + let installationApps: InstallationApps + + enum CodingKeys: String, CodingKey { + case updateHash = "UpdateHash" + case osVersions = "OSVersions" + case xProtectPayloads = "XProtectPayloads" + case xProtectPlistConfigData = "XProtectPlistConfigData" + case models = "Models" + case installationApps = "InstallationApps" + } +} + +struct SofaOSVersion: Codable { + let osVersion: String + let latest: LatestOS + let securityReleases: [SecurityRelease] + let supportedModels: [SupportedModel] + + enum CodingKeys: String, CodingKey { + case osVersion = "OSVersion" + case latest = "Latest" + case securityReleases = "SecurityReleases" + case supportedModels = "SupportedModels" + } +} + +struct LatestOS: Codable { + let productVersion, build: String + let releaseDate: Date? + let expirationDate: Date + let supportedDevices: [String] + + enum CodingKeys: String, CodingKey { + case productVersion = "ProductVersion" + case build = "Build" + case releaseDate = "ReleaseDate" + case expirationDate = "ExpirationDate" + case supportedDevices = "SupportedDevices" + } +} + +struct SecurityRelease: Codable { + let updateName, productVersion: String + let releaseDate: Date + let securityInfo: String + let cves: [String: Bool] + let activelyExploitedCVEs: [String] + let uniqueCVEsCount, daysSincePreviousRelease: Int + + enum CodingKeys: String, CodingKey { + case updateName = "UpdateName" + case productVersion = "ProductVersion" + case releaseDate = "ReleaseDate" + case securityInfo = "SecurityInfo" + case cves = "CVEs" + case activelyExploitedCVEs = "ActivelyExploitedCVEs" + case uniqueCVEsCount = "UniqueCVEsCount" + case daysSincePreviousRelease = "DaysSincePreviousRelease" + } +} + +struct SupportedModel: Codable { + let model: String + let url: String + let identifiers: [String: String] + + enum CodingKeys: String, CodingKey { + case model = "Model" + case url = "URL" + case identifiers = "Identifiers" + } +} + +struct XProtectPayloads: Codable { + let xProtectFramework, pluginService: String + let releaseDate: Date + + enum CodingKeys: String, CodingKey { + case xProtectFramework = "com.apple.XProtectFramework.XProtect" + case pluginService = "com.apple.XprotectFramework.PluginService" + case releaseDate = "ReleaseDate" + } +} + +struct XProtectPlistConfigData: Codable { + let xProtect: String + let releaseDate: Date + + enum CodingKeys: String, CodingKey { + case xProtect = "com.apple.XProtect" + case releaseDate = "ReleaseDate" + } +} + +struct ModelInfo: Codable { + let marketingName: String + let supportedOS: [String] + let osVersions: [Int] + + enum CodingKeys: String, CodingKey { + case marketingName = "MarketingName" + case supportedOS = "SupportedOS" + case osVersions = "OSVersions" + } +} + +struct InstallationApps: Codable { + let latestUMA: UMA + let allPreviousUMA: [UMA] + let latestMacIPSW: MacIPSW + + enum CodingKeys: String, CodingKey { + case latestUMA = "LatestUMA" + case allPreviousUMA = "AllPreviousUMA" + case latestMacIPSW = "LatestMacIPSW" + } +} + +struct UMA: Codable { + let title, version, build, appleSlug, url: String + + enum CodingKeys: String, CodingKey { + case title, version, build + case appleSlug = "apple_slug" + case url + } +} + +struct MacIPSW: Codable { + let macosIpswURL: String + let macosIpswBuild, macosIpswVersion, macosIpswAppleSlug: String + + enum CodingKeys: String, CodingKey { + case macosIpswURL = "macos_ipsw_url" + case macosIpswBuild = "macos_ipsw_build" + case macosIpswVersion = "macos_ipsw_version" + case macosIpswAppleSlug = "macos_ipsw_apple_slug" + } +} + +extension MacOSDataFeed { + init(data: Data) throws { + let decoder = JSONDecoder() + decoder.dateDecodingStrategy = .flexibleISO8601 // Use custom ISO 8601 date format because of weird strings returning in v1 of sofa api + self = try decoder.decode(MacOSDataFeed.self, from: data) + } + + init(_ json: String, using encoding: String.Encoding = .utf8) throws { + guard let data = json.data(using: encoding) else { + throw NSError(domain: "JSONDecoding", code: 0, userInfo: nil) + } + try self.init(data: data) + } + + init(fromURL url: URL) throws { + try self.init(data: try Data(contentsOf: url)) + } + + func with( + updateHash: String, + osVersions: [SofaOSVersion], + xProtectPayloads: XProtectPayloads, + xProtectPlistConfigData: XProtectPlistConfigData, + models: [String: ModelInfo], + installationApps: InstallationApps + ) -> MacOSDataFeed { + return MacOSDataFeed( + updateHash: updateHash, + osVersions: osVersions, + xProtectPayloads: xProtectPayloads, + xProtectPlistConfigData: xProtectPlistConfigData, + models: models, + installationApps: installationApps + ) + } +} + +extension JSONDecoder.DateDecodingStrategy { + static var flexibleISO8601: JSONDecoder.DateDecodingStrategy { + return .custom { decoder -> Date in + let container = try decoder.singleValueContainer() + let dateString = try container.decode(String.self) + + let dateFormatter = DateFormatter() + dateFormatter.locale = Locale(identifier: "en_US_POSIX") + dateFormatter.timeZone = TimeZone(secondsFromGMT: 0) + + // List of date formats to try + let dateFormats = ["yyyy-MM-dd'T'HH:mm:ssZ", "yyyy-MM-dd'T'HH:mm:ss.SSSZ"] + for format in dateFormats { + dateFormatter.dateFormat = format + if let date = dateFormatter.date(from: dateString) { + return date + } + } + + // Return POSIX epoch start date if no valid date is found + return Date(timeIntervalSince1970: 0) + } + } +} + +class SOFA: NSObject, URLSessionDelegate { + func URLSync(url: URL, maxRetries: Int = 3) -> (data: Data?, response: URLResponse?, error: Error?) { + let semaphore = DispatchSemaphore(value: 0) + let request = URLRequest(url: url) + let session = URLSession(configuration: .default, delegate: self, delegateQueue: nil) + var attempts = 0 + + var responseData: Data? + var response: URLResponse? + var responseError: Error? + + // Retry loop + while attempts < maxRetries { + attempts += 1 + let task = session.dataTask(with: request) { data, resp, error in + responseData = data + response = resp + responseError = error + semaphore.signal() + } + task.resume() + + semaphore.wait() + + // Break the loop if the task succeeded or return an error other than a timeout + if responseError == nil || (responseError! as NSError).code != NSURLErrorTimedOut { + break + } else if attempts < maxRetries { + // Reset the error to try again + responseError = nil + } + } + + return (responseData, response, responseError) + } +} diff --git a/Nudge/UI/Defaults.swift b/Nudge/UI/Defaults.swift index 5850cdab..cab45a92 100644 --- a/Nudge/UI/Defaults.swift +++ b/Nudge/UI/Defaults.swift @@ -28,6 +28,7 @@ struct Globals { static let nudgeJSONPreferences = NetworkFileManager().getNudgeJSONPreferences() // Device Properties static let gdmfAssets = NetworkFileManager().getGDMFAssets() + static let sofaAssets = NetworkFileManager().getSOFAAssets() static let hardwareModelID = DeviceManager().getHardwareModelID() } diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index d8098e5f..01103db0 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -174,11 +174,11 @@ class AppDelegate: NSObject, NSApplicationDelegate { // Pre-Launch Logic func applicationWillFinishLaunching(_ notification: Notification) { // print("applicationWillFinishLaunching") - if let macOSAssets = Globals.gdmfAssets?.publicAssetSets.macOS { + if let macOSGDMFAssets = Globals.gdmfAssets?.publicAssetSets.macOS { // Find the first macOS asset that matches the product version var tempRequiredMinOSVersion = PrefsWrapper.requiredMinimumOSVersion - tempRequiredMinOSVersion = "14.4.1" // Hack so I can test this since 14.99.99 doesn't exist in GDMF - if let matchingAsset = macOSAssets.first(where: { $0.productVersion == tempRequiredMinOSVersion }) { + tempRequiredMinOSVersion = "14.5" // Hack so I can test this since 14.99.99 doesn't exist in GDMF + if let matchingAsset = macOSGDMFAssets.first(where: { $0.productVersion == tempRequiredMinOSVersion }) { // Check if the specified device is in the supported devices of the matching asset print("GDMF Matched OS Version: \(matchingAsset.productVersion)") print("GDMF Assets: \(matchingAsset.supportedDevices)") @@ -189,7 +189,31 @@ class AppDelegate: NSObject, NSApplicationDelegate { print("DeviceGDMFSupported: False") } } else { - print("No macOS assets available.") + print("No macOS GDMF assets available.") + } + + if let macOSSOFAAssets = Globals.sofaAssets?.osVersions { + // Find the first macOS asset that matches the product version + var tempRequiredMinOSVersion = PrefsWrapper.requiredMinimumOSVersion + tempRequiredMinOSVersion = "14.5" // Hack so I can test this since 14.99.99 doesn't exist in GDMF + var foundMatch = false + // osVersions.map { $0.latest.productVersion } + for osVersion in macOSSOFAAssets { + if let matchingAsset = osVersion.securityReleases.first(where: { $0.productVersion == tempRequiredMinOSVersion }) { + foundMatch = true + // Check if the specified device is in the supported devices of the matching asset + print("SOFA Matched OS Version: \(matchingAsset.productVersion)") + // print("SOFA Assets: \(matchingAsset.supportedDevices)") + print("Assessed Model ID: \(Globals.hardwareModelID)") + // print("DeviceSOFASupported: \(matchingAsset.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }))") + } + } + if !foundMatch { + // If no matching product version found or the device is not supported, return false + print("DeviceSOFASupported: False") + } + } else { + print("No macOS SOFA assets available.") } handleSMAppService() checkForBadProfilePath() diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 5133c9e9..80c9cfab 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -787,7 +787,7 @@ struct NetworkFileManager { } } - func getGDMFAssets() -> GDMFAssetInfo? { + func getGDMFAssets() -> GDMFAssetInfo? { // Define the URL you want to pin to if let url = URL(string: "https://gdmf.apple.com/v2/pmv") { // Call the pin method @@ -824,6 +824,26 @@ struct NetworkFileManager { return nil } + func getSOFAAssets() -> MacOSDataFeed? { + if let url = URL(string: "https://sofa.macadmins.io/v1/macos_data_feed.json") { + let sofaData = SOFA().URLSync(url: url) + if (sofaData.error == nil) { + do { + let assetInfo = try MacOSDataFeed(data: sofaData.data!) + return assetInfo + } catch { + LogManager.error("Failed to decode sofa JSON: \(error.localizedDescription)", logger: utilsLog) + LogManager.error("Failed to decode sofa JSON: \(error)", logger: utilsLog) + } + } else { + LogManager.error("Failed to fetch sofa JSON: \(sofaData.error!.localizedDescription)", logger: utilsLog) + } + } else { + LogManager.error("Failed to decode sofa JSON URL string", logger: utilsLog) + } + return nil + } + func getBackupMajorUpgradeAppPath() -> String { switch VersionManager.getMajorRequiredNudgeOSVersion() { case 12: From b09d4936b30ecc53f3c2e0033906ed4697d525a6 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 14 May 2024 14:52:12 -0500 Subject: [PATCH 017/141] remove need for custom decoding strategy for sofa thanks Henry for the quick fix --- Nudge/3rd Party Assets/sofa.swift | 27 +-------------------------- 1 file changed, 1 insertion(+), 26 deletions(-) diff --git a/Nudge/3rd Party Assets/sofa.swift b/Nudge/3rd Party Assets/sofa.swift index 58d46bc1..105b2b02 100644 --- a/Nudge/3rd Party Assets/sofa.swift +++ b/Nudge/3rd Party Assets/sofa.swift @@ -156,7 +156,7 @@ struct MacIPSW: Codable { extension MacOSDataFeed { init(data: Data) throws { let decoder = JSONDecoder() - decoder.dateDecodingStrategy = .flexibleISO8601 // Use custom ISO 8601 date format because of weird strings returning in v1 of sofa api + decoder.dateDecodingStrategy = .iso8601 // Use ISO 8601 date format self = try decoder.decode(MacOSDataFeed.self, from: data) } @@ -190,31 +190,6 @@ extension MacOSDataFeed { } } -extension JSONDecoder.DateDecodingStrategy { - static var flexibleISO8601: JSONDecoder.DateDecodingStrategy { - return .custom { decoder -> Date in - let container = try decoder.singleValueContainer() - let dateString = try container.decode(String.self) - - let dateFormatter = DateFormatter() - dateFormatter.locale = Locale(identifier: "en_US_POSIX") - dateFormatter.timeZone = TimeZone(secondsFromGMT: 0) - - // List of date formats to try - let dateFormats = ["yyyy-MM-dd'T'HH:mm:ssZ", "yyyy-MM-dd'T'HH:mm:ss.SSSZ"] - for format in dateFormats { - dateFormatter.dateFormat = format - if let date = dateFormatter.date(from: dateString) { - return date - } - } - - // Return POSIX epoch start date if no valid date is found - return Date(timeIntervalSince1970: 0) - } - } -} - class SOFA: NSObject, URLSessionDelegate { func URLSync(url: URL, maxRetries: Int = 3) -> (data: Data?, response: URLResponse?, error: Error?) { let semaphore = DispatchSemaphore(value: 0) From 2deab45644f3311804f7745fbf9e87634970528a Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 15 May 2024 15:12:52 -0500 Subject: [PATCH 018/141] new version of sofa has SupportedDevices enabled --- Nudge/3rd Party Assets/sofa.swift | 2 ++ Nudge/UI/Main.swift | 4 ++-- 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/Nudge/3rd Party Assets/sofa.swift b/Nudge/3rd Party Assets/sofa.swift index 105b2b02..085a3b1f 100644 --- a/Nudge/3rd Party Assets/sofa.swift +++ b/Nudge/3rd Party Assets/sofa.swift @@ -58,6 +58,7 @@ struct SecurityRelease: Codable { let updateName, productVersion: String let releaseDate: Date let securityInfo: String + let supportedDevices: [String] let cves: [String: Bool] let activelyExploitedCVEs: [String] let uniqueCVEsCount, daysSincePreviousRelease: Int @@ -67,6 +68,7 @@ struct SecurityRelease: Codable { case productVersion = "ProductVersion" case releaseDate = "ReleaseDate" case securityInfo = "SecurityInfo" + case supportedDevices = "SupportedDevices" case cves = "CVEs" case activelyExploitedCVEs = "ActivelyExploitedCVEs" case uniqueCVEsCount = "UniqueCVEsCount" diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 01103db0..b1ae2558 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -203,9 +203,9 @@ class AppDelegate: NSObject, NSApplicationDelegate { foundMatch = true // Check if the specified device is in the supported devices of the matching asset print("SOFA Matched OS Version: \(matchingAsset.productVersion)") - // print("SOFA Assets: \(matchingAsset.supportedDevices)") + print("SOFA Assets: \(matchingAsset.supportedDevices)") print("Assessed Model ID: \(Globals.hardwareModelID)") - // print("DeviceSOFASupported: \(matchingAsset.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }))") + print("DeviceSOFASupported: \(matchingAsset.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }))") } } if !foundMatch { From 0f31977f22ccb852611ec8090065dfedd7d99dff Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 15 May 2024 15:31:30 -0500 Subject: [PATCH 019/141] new utilizeSOFAFeed preference - defaults to false --- Example Assets/com.github.macadmins.Nudge.tester.json | 3 ++- Nudge/Preferences/DefaultPreferencesNudge.swift | 6 ++++++ Nudge/Preferences/PreferencesStructure.swift | 8 +++++--- Nudge/Utilities/Utils.swift | 3 +++ 4 files changed, 16 insertions(+), 4 deletions(-) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index a11315a6..39a8f706 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -14,7 +14,8 @@ "com.microsoft.VSCode", "us.zoom.xos" ], - "terminateApplicationsOnLaunch": false + "terminateApplicationsOnLaunch": false, + "utilizeSOFAFeed": true }, "osVersionRequirements": [ { diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 838135b4..f5d5cde6 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -142,6 +142,12 @@ struct OptionalFeatureVariables { optionalFeaturesJSON?.terminateApplicationsOnLaunch ?? false } + + static var utilizeSOFAFeed: Bool { + optionalFeaturesProfile?["utilizeSOFAFeedh"] as? Bool ?? + optionalFeaturesJSON?.utilizeSOFAFeed ?? + false + } } // OS Version Requirements diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index aa47637d..a825ffff 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -53,7 +53,7 @@ struct OptionalFeatures: Codable { var acceptableApplicationBundleIDs, acceptableAssertionApplicationNames: [String]? var acceptableAssertionUsage, acceptableCameraUsage, acceptableScreenSharingUsage, aggressiveUserExperience, aggressiveUserFullScreenExperience, asynchronousSoftwareUpdate, attemptToBlockApplicationLaunches, attemptToFetchMajorUpgrade: Bool? var blockedApplicationBundleIDs: [String]? - var disableSoftwareUpdateWorkflow, enforceMinorUpdates, terminateApplicationsOnLaunch: Bool? + var disableSoftwareUpdateWorkflow, enforceMinorUpdates, terminateApplicationsOnLaunch, utilizeSOFAFeed: Bool? } // MARK: OptionalFeatures convenience initializers and mutators @@ -88,7 +88,8 @@ extension OptionalFeatures { blockedApplicationBundleIDs: [String]? = nil, disableSoftwareUpdateWorkflow: Bool? = nil, enforceMinorUpdates: Bool? = nil, - terminateApplicationsOnLaunch: Bool? = nil + terminateApplicationsOnLaunch: Bool? = nil, + utilizeSOFAFeed: Bool? = nil ) -> OptionalFeatures { return OptionalFeatures( acceptableApplicationBundleIDs: acceptableApplicationBundleIDs ?? self.acceptableApplicationBundleIDs, @@ -104,7 +105,8 @@ extension OptionalFeatures { blockedApplicationBundleIDs: blockedApplicationBundleIDs ?? self.blockedApplicationBundleIDs, disableSoftwareUpdateWorkflow: disableSoftwareUpdateWorkflow ?? self.disableSoftwareUpdateWorkflow, enforceMinorUpdates: enforceMinorUpdates ?? self.enforceMinorUpdates, - terminateApplicationsOnLaunch: terminateApplicationsOnLaunch ?? self.terminateApplicationsOnLaunch + terminateApplicationsOnLaunch: terminateApplicationsOnLaunch ?? self.terminateApplicationsOnLaunch, + utilizeSOFAFeed: utilizeSOFAFeed ?? self.utilizeSOFAFeed ) } } diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 80c9cfab..3b6d92af 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -825,6 +825,9 @@ struct NetworkFileManager { } func getSOFAAssets() -> MacOSDataFeed? { + if !OptionalFeatureVariables.utilizeSOFAFeed { + return nil + } if let url = URL(string: "https://sofa.macadmins.io/v1/macos_data_feed.json") { let sofaData = SOFA().URLSync(url: url) if (sofaData.error == nil) { From b20dd4409a1c509be2cf95a0b5b007a53f21de11 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 15 May 2024 16:05:28 -0500 Subject: [PATCH 020/141] start implementing unsupported ui --- .../com.github.macadmins.Nudge.tester.json | 7 ++++- .../Preferences/DefaultPreferencesNudge.swift | 31 +++++++++++++++++++ Nudge/Preferences/PreferencesStructure.swift | 19 ++++++++++-- Nudge/UI/Defaults.swift | 1 + Nudge/UI/Main.swift | 31 ++++--------------- Nudge/UI/StandardMode/RightSide.swift | 5 +-- Nudge/Utilities/Preferences.swift | 10 ++++++ 7 files changed, 73 insertions(+), 31 deletions(-) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index 39a8f706..92375ec7 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -21,7 +21,7 @@ { "aboutUpdateURL": "https://apple.com", "requiredInstallationDate": "2025-01-01T00:00:00Z", - "requiredMinimumOSVersion": "14.99.99" + "requiredMinimumOSVersion": "14.5" } ], "userExperience": { @@ -45,10 +45,15 @@ "customDeferralDropdownText": "customDeferralDropdownText", "informationButtonText": "informationButtonText", "mainContentHeader": "mainContentHeader", + "mainContentHeaderUnsupported": "mainContentHeaderUnsupported", "mainContentNote": "mainContentNote", + "mainContentNoteUnsupported": "mainContentNoteUnsupported", "mainContentSubHeader": "mainContentSubHeader", + "mainContentSubHeaderUnsupported": "mainContentSubHeaderUnsupported", "mainContentText": "mainContentText", + "mainContentTextUnsupported": "mainContentTextUnsupported", "mainHeader": "mainHeader", + "mainHeaderUnsupported": "mainHeaderUnsupported", "oneDayDeferralButtonText": "oneDayDeferralButtonText", "oneHourDeferralButtonText": "oneHourDeferralButtonText", "primaryQuitButtonText": "primaryQuitButtonText", diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index f5d5cde6..810d2b48 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -373,24 +373,48 @@ struct UserInterfaceVariables { "Your device will restart during this update" } + static var mainContentHeaderUnsupported: String { + userInterfaceUpdateElementsProfile?["mainContentHeaderUnsupported"] as? String ?? + userInterfaceUpdateElementsJSON?.mainContentHeaderUnsupported ?? + "Your device will restart during this update" + } + static var mainContentNote: String { userInterfaceUpdateElementsProfile?["mainContentNote"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentNote ?? "Important Notes" } + static var mainContentNoteUnsupported: String { + userInterfaceUpdateElementsProfile?["mainContentNoteUnsupported"] as? String ?? + userInterfaceUpdateElementsJSON?.mainContentNoteUnsupported ?? + "Important Notes" + } + static var mainContentSubHeader: String { userInterfaceUpdateElementsProfile?["mainContentSubHeader"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentSubHeader ?? "Updates can take around 30 minutes to complete" } + static var mainContentSubHeaderUnsupported: String { + userInterfaceUpdateElementsProfile?["mainContentSubHeaderUnsupported"] as? String ?? + userInterfaceUpdateElementsJSON?.mainContentSubHeaderUnsupported ?? + "Updates can take around 30 minutes to complete" + } + static var mainContentText: String { userInterfaceUpdateElementsProfile?["mainContentText"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentText ?? "A fully up-to-date device is required to ensure that IT can accurately protect your device.\n\nIf you do not update your device, you may lose access to some items necessary for your day-to-day tasks.\n\nTo begin the update, simply click on the Update Device button and follow the provided steps." } + static var mainContentTextUnsupported: String { + userInterfaceUpdateElementsProfile?["mainContentTextUnsupported"] as? String ?? + userInterfaceUpdateElementsJSON?.mainContentTextUnsupported ?? + "A fully up-to-date device is required to ensure that IT can accurately protect your device.\n\nIf you do not update your device, you may lose access to some items necessary for your day-to-day tasks.\n\nTo begin the update, simply click on the Update Device button and follow the provided steps." + } + static var primaryQuitButtonText: String { userInterfaceUpdateElementsProfile?["primaryQuitButtonText"] as? String ?? userInterfaceUpdateElementsJSON?.primaryQuitButtonText ?? @@ -421,6 +445,13 @@ struct UserInterfaceVariables { "A friendly reminder from your local IT team" } + static var subHeaderUnsupported: String { + userInterfaceUpdateElementsProfile?["subHeaderUnsupported"] as? String ?? + userInterfaceUpdateElementsJSON?.subHeaderUnsupported ?? + "A friendly reminder from your local IT team" + } + + static var customDeferralDropdownText: String { userInterfaceUpdateElementsProfile?["customDeferralDropdownText"] as? String ?? userInterfaceUpdateElementsJSON?.customDeferralDropdownText ?? diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index a825ffff..40bce2a8 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -376,11 +376,12 @@ extension UserInterface { struct UpdateElement: Codable { var language, actionButtonText, customDeferralButtonText, customDeferralDropdownText, informationButtonText: String? var mainContentHeader, mainContentNote, mainContentSubHeader, mainContentText, mainHeader: String? - var oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, secondaryQuitButtonText, subHeader, screenShotAltText: String? - + var mainContentHeaderUnsupported, mainContentNoteUnsupported, mainContentSubHeaderUnsupported, mainContentTextUnsupported, mainHeaderUnsupported: String? + var oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, secondaryQuitButtonText, subHeader, subHeaderUnsupported, screenShotAltText: String? + enum CodingKeys: String, CodingKey { case language = "_language" - case actionButtonText, customDeferralButtonText, customDeferralDropdownText, informationButtonText, mainContentHeader, mainContentNote, mainContentSubHeader, mainContentText, mainHeader, oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, secondaryQuitButtonText, subHeader, screenShotAltText + case actionButtonText, customDeferralButtonText, customDeferralDropdownText, informationButtonText, mainContentHeader, mainContentNote, mainContentSubHeader, mainContentText, mainHeader, mainContentHeaderUnsupported, mainContentNoteUnsupported, mainContentSubHeaderUnsupported, mainContentTextUnsupported, mainHeaderUnsupported, oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, secondaryQuitButtonText, subHeader, subHeaderUnsupported, screenShotAltText } } @@ -413,11 +414,17 @@ extension UpdateElement { mainContentSubHeader: String? = nil, mainContentText: String? = nil, mainHeader: String? = nil, + mainContentHeaderUnsupported: String? = nil, + mainContentNoteUnsupported: String? = nil, + mainContentSubHeaderUnsupported: String? = nil, + mainContentTextUnsupported: String? = nil, + mainHeaderUnsupported: String? = nil, oneDayDeferralButtonText: String? = nil, oneHourDeferralButtonText: String? = nil, primaryQuitButtonText: String? = nil, secondaryQuitButtonText: String? = nil, subHeader: String? = nil, + subHeaderUnsupported: String? = nil, screenShotAltText: String? = nil ) -> UpdateElement { return UpdateElement( @@ -431,11 +438,17 @@ extension UpdateElement { mainContentSubHeader: mainContentSubHeader ?? self.mainContentSubHeader, mainContentText: mainContentText ?? self.mainContentText, mainHeader: mainHeader ?? self.mainHeader, + mainContentHeaderUnsupported: mainContentHeaderUnsupported ?? self.mainContentHeaderUnsupported, + mainContentNoteUnsupported: mainContentNoteUnsupported ?? self.mainContentNoteUnsupported, + mainContentSubHeaderUnsupported: mainContentSubHeaderUnsupported ?? self.mainContentSubHeaderUnsupported, + mainContentTextUnsupported: mainContentTextUnsupported ?? self.mainContentTextUnsupported, + mainHeaderUnsupported: mainHeaderUnsupported ?? self.mainHeaderUnsupported, oneDayDeferralButtonText: oneDayDeferralButtonText ?? self.oneDayDeferralButtonText, oneHourDeferralButtonText: oneHourDeferralButtonText ?? self.oneHourDeferralButtonText, primaryQuitButtonText: primaryQuitButtonText ?? self.primaryQuitButtonText, secondaryQuitButtonText: secondaryQuitButtonText ?? self.secondaryQuitButtonText, subHeader: subHeader ?? self.subHeader, + subHeaderUnsupported: subHeaderUnsupported ?? self.subHeaderUnsupported, screenShotAltText: screenShotAltText ?? self.screenShotAltText ) } diff --git a/Nudge/UI/Defaults.swift b/Nudge/UI/Defaults.swift index cab45a92..175f21e8 100644 --- a/Nudge/UI/Defaults.swift +++ b/Nudge/UI/Defaults.swift @@ -70,6 +70,7 @@ class AppState: ObservableObject { @Published var daysRemaining = DateManager().getNumberOfDaysBetween() @Published var deferralCountPastThreshold = false @Published var deferRunUntil = Globals.nudgeDefaults.object(forKey: "deferRunUntil") as? Date + @Published var deviceSupportedByOSVersion = true @Published var hasClickedSecondaryQuitButton = false @Published var hasLoggedDeferralCountPastThreshold = false @Published var hasLoggedDeferralCountPastThresholdDualQuitButtons = false diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index b1ae2558..237792ed 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -174,38 +174,19 @@ class AppDelegate: NSObject, NSApplicationDelegate { // Pre-Launch Logic func applicationWillFinishLaunching(_ notification: Notification) { // print("applicationWillFinishLaunching") - if let macOSGDMFAssets = Globals.gdmfAssets?.publicAssetSets.macOS { - // Find the first macOS asset that matches the product version - var tempRequiredMinOSVersion = PrefsWrapper.requiredMinimumOSVersion - tempRequiredMinOSVersion = "14.5" // Hack so I can test this since 14.99.99 doesn't exist in GDMF - if let matchingAsset = macOSGDMFAssets.first(where: { $0.productVersion == tempRequiredMinOSVersion }) { - // Check if the specified device is in the supported devices of the matching asset - print("GDMF Matched OS Version: \(matchingAsset.productVersion)") - print("GDMF Assets: \(matchingAsset.supportedDevices)") - print("Assessed Model ID: \(Globals.hardwareModelID)") - print("DeviceGDMFSupported: \(matchingAsset.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }))") - } else { - // If no matching product version found or the device is not supported, return false - print("DeviceGDMFSupported: False") - } - } else { - print("No macOS GDMF assets available.") - } - if let macOSSOFAAssets = Globals.sofaAssets?.osVersions { // Find the first macOS asset that matches the product version - var tempRequiredMinOSVersion = PrefsWrapper.requiredMinimumOSVersion - tempRequiredMinOSVersion = "14.5" // Hack so I can test this since 14.99.99 doesn't exist in GDMF var foundMatch = false // osVersions.map { $0.latest.productVersion } for osVersion in macOSSOFAAssets { - if let matchingAsset = osVersion.securityReleases.first(where: { $0.productVersion == tempRequiredMinOSVersion }) { + if let matchingAsset = osVersion.securityReleases.first(where: { $0.productVersion == PrefsWrapper.requiredMinimumOSVersion }) { foundMatch = true // Check if the specified device is in the supported devices of the matching asset - print("SOFA Matched OS Version: \(matchingAsset.productVersion)") - print("SOFA Assets: \(matchingAsset.supportedDevices)") - print("Assessed Model ID: \(Globals.hardwareModelID)") - print("DeviceSOFASupported: \(matchingAsset.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }))") +// print("SOFA Matched OS Version: \(matchingAsset.productVersion)") +// print("SOFA Assets: \(matchingAsset.supportedDevices)") +// print("Assessed Model ID: \(Globals.hardwareModelID)") + // nudgePrimaryState.deviceSupportedByOSVersion = matchingAsset.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }) + nudgePrimaryState.deviceSupportedByOSVersion = false } } if !foundMatch { diff --git a/Nudge/UI/StandardMode/RightSide.swift b/Nudge/UI/StandardMode/RightSide.swift index 0541b1fa..0001f57c 100644 --- a/Nudge/UI/StandardMode/RightSide.swift +++ b/Nudge/UI/StandardMode/RightSide.swift @@ -37,14 +37,15 @@ struct StandardModeRightSide: View { VStack(alignment: .center) { HStack { VStack(alignment: .leading, spacing: 5) { + let mainHeaderText = appState.deviceSupportedByOSVersion == true ? getMainHeader().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : getMainHeaderUnsupported().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) HStack { - Text(getMainHeader().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(mainHeaderText) .font(.largeTitle) .minimumScaleFactor(0.5) .frame(maxHeight: 25) .lineLimit(1) } - + HStack { Text(UserInterfaceVariables.subHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .font(.body) diff --git a/Nudge/Utilities/Preferences.swift b/Nudge/Utilities/Preferences.swift index 9370d230..09eaf5d8 100644 --- a/Nudge/Utilities/Preferences.swift +++ b/Nudge/Utilities/Preferences.swift @@ -239,6 +239,16 @@ func getMainHeader() -> String { getUserInterfaceUpdateElementsJSON()?.mainHeader ?? "Your device requires a security update" } +func getMainHeaderUnsupported() -> String { + if CommandLineUtilities().demoModeEnabled() { + return "Your device is no longer capable of security updates (Demo Mode)" + } else if CommandLineUtilities().unitTestingEnabled() { + return "Your device is no longer capable of security updates (Unit Testing Mode)" + } + return UserInterfaceVariables.userInterfaceUpdateElementsProfile?["mainHeaderUnsupported"] as? String ?? + getUserInterfaceUpdateElementsJSON()?.mainHeaderUnsupported ?? "Your device is no longer capable of security updates" +} + func simpleMode() -> Bool { return CommandLineUtilities().simpleModeEnabled() || UserInterfaceVariables.userInterfaceProfile?["simpleMode"] as? Bool ?? From a8f382a646eb0168398741fb9df6905421d4c1de Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 15 May 2024 16:13:54 -0500 Subject: [PATCH 021/141] change the other text ternaries --- .../com.github.macadmins.Nudge.tester.json | 1 + Nudge/UI/SimpleMode/SimpleMode.swift | 2 +- Nudge/UI/StandardMode/RightSide.swift | 13 ++++++------- 3 files changed, 8 insertions(+), 8 deletions(-) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index 92375ec7..7c25629f 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -59,6 +59,7 @@ "primaryQuitButtonText": "primaryQuitButtonText", "secondaryQuitButtonText": "secondaryQuitButtonText", "subHeader": "subHeader", + "subHeaderUnsupported": "subHeaderUnsupported", "screenShotAltText": "Click to zoom into screenshot" } ] diff --git a/Nudge/UI/SimpleMode/SimpleMode.swift b/Nudge/UI/SimpleMode/SimpleMode.swift index 4673fe32..2e5236db 100644 --- a/Nudge/UI/SimpleMode/SimpleMode.swift +++ b/Nudge/UI/SimpleMode/SimpleMode.swift @@ -30,7 +30,7 @@ struct SimpleMode: View { CompanyLogo() Spacer() - Text(getMainHeader().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(appState.deviceSupportedByOSVersion == true ? getMainHeader().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : getMainHeaderUnsupported().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .font(.title) remainingTimeView diff --git a/Nudge/UI/StandardMode/RightSide.swift b/Nudge/UI/StandardMode/RightSide.swift index 0001f57c..e6b67bb8 100644 --- a/Nudge/UI/StandardMode/RightSide.swift +++ b/Nudge/UI/StandardMode/RightSide.swift @@ -37,9 +37,8 @@ struct StandardModeRightSide: View { VStack(alignment: .center) { HStack { VStack(alignment: .leading, spacing: 5) { - let mainHeaderText = appState.deviceSupportedByOSVersion == true ? getMainHeader().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : getMainHeaderUnsupported().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) HStack { - Text(mainHeaderText) + Text(appState.deviceSupportedByOSVersion == true ? getMainHeader().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : getMainHeaderUnsupported().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .font(.largeTitle) .minimumScaleFactor(0.5) .frame(maxHeight: 25) @@ -47,7 +46,7 @@ struct StandardModeRightSide: View { } HStack { - Text(UserInterfaceVariables.subHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(appState.deviceSupportedByOSVersion == true ? UserInterfaceVariables.subHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.subHeaderUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .font(.body) .fontWeight(.bold) .lineLimit(1) @@ -65,13 +64,13 @@ struct StandardModeRightSide: View { HStack(alignment: .center) { VStack(alignment: .leading, spacing: 1) { HStack { - Text(UserInterfaceVariables.mainContentHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(appState.deviceSupportedByOSVersion == true ? UserInterfaceVariables.mainContentHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentHeaderUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .font(.callout) .fontWeight(.bold) Spacer() } HStack { - Text(UserInterfaceVariables.mainContentSubHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(appState.deviceSupportedByOSVersion == true ? UserInterfaceVariables.mainContentSubHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentSubHeaderUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .font(.callout) Spacer() } @@ -89,7 +88,7 @@ struct StandardModeRightSide: View { Divider() HStack { - Text(UserInterfaceVariables.mainContentNote.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(appState.deviceSupportedByOSVersion == true ? UserInterfaceVariables.mainContentNote.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentNoteUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .font(.callout) .fontWeight(.bold) .foregroundColor(appState.differentiateWithoutColor ? .accessibleRed : .red) @@ -99,7 +98,7 @@ struct StandardModeRightSide: View { ScrollView(.vertical) { VStack { HStack { - Text(UserInterfaceVariables.mainContentText.replacingOccurrences(of: "\\n", with: "\n").localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(appState.deviceSupportedByOSVersion == true ? UserInterfaceVariables.mainContentText.replacingOccurrences(of: "\\n", with: "\n").localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentTextUnsupported.replacingOccurrences(of: "\\n", with: "\n").localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .font(.callout) .multilineTextAlignment(.leading) Spacer() From bc956cab5b6442de0a183c222d04c40aedf0fb6e Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 17 May 2024 09:37:58 -0500 Subject: [PATCH 022/141] sort the keys --- Nudge/Preferences/PreferencesStructure.swift | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index 40bce2a8..38cfdfab 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -410,14 +410,14 @@ extension UpdateElement { customDeferralDropdownText: String? = nil, informationButtonText: String? = nil, mainContentHeader: String? = nil, - mainContentNote: String? = nil, - mainContentSubHeader: String? = nil, - mainContentText: String? = nil, - mainHeader: String? = nil, mainContentHeaderUnsupported: String? = nil, + mainContentNote: String? = nil, mainContentNoteUnsupported: String? = nil, + mainContentSubHeader: String? = nil, mainContentSubHeaderUnsupported: String? = nil, + mainContentText: String? = nil, mainContentTextUnsupported: String? = nil, + mainHeader: String? = nil, mainHeaderUnsupported: String? = nil, oneDayDeferralButtonText: String? = nil, oneHourDeferralButtonText: String? = nil, @@ -434,14 +434,14 @@ extension UpdateElement { customDeferralDropdownText: customDeferralDropdownText ?? self.customDeferralDropdownText, informationButtonText: informationButtonText ?? self.informationButtonText, mainContentHeader: mainContentHeader ?? self.mainContentHeader, - mainContentNote: mainContentNote ?? self.mainContentNote, - mainContentSubHeader: mainContentSubHeader ?? self.mainContentSubHeader, - mainContentText: mainContentText ?? self.mainContentText, - mainHeader: mainHeader ?? self.mainHeader, mainContentHeaderUnsupported: mainContentHeaderUnsupported ?? self.mainContentHeaderUnsupported, + mainContentNote: mainContentNote ?? self.mainContentNote, mainContentNoteUnsupported: mainContentNoteUnsupported ?? self.mainContentNoteUnsupported, + mainContentSubHeader: mainContentSubHeader ?? self.mainContentSubHeader, mainContentSubHeaderUnsupported: mainContentSubHeaderUnsupported ?? self.mainContentSubHeaderUnsupported, + mainContentText: mainContentText ?? self.mainContentText, mainContentTextUnsupported: mainContentTextUnsupported ?? self.mainContentTextUnsupported, + mainHeader: mainHeader ?? self.mainHeader, mainHeaderUnsupported: mainHeaderUnsupported ?? self.mainHeaderUnsupported, oneDayDeferralButtonText: oneDayDeferralButtonText ?? self.oneDayDeferralButtonText, oneHourDeferralButtonText: oneHourDeferralButtonText ?? self.oneHourDeferralButtonText, From 2afb77888ccfbc8557a468c78eadaf0685b9fcf9 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 17 May 2024 09:45:13 -0500 Subject: [PATCH 023/141] finish sorting --- Nudge/Preferences/PreferencesStructure.swift | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index 38cfdfab..464ff61b 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -375,13 +375,13 @@ extension UserInterface { // MARK: - UpdateElement struct UpdateElement: Codable { var language, actionButtonText, customDeferralButtonText, customDeferralDropdownText, informationButtonText: String? - var mainContentHeader, mainContentNote, mainContentSubHeader, mainContentText, mainHeader: String? - var mainContentHeaderUnsupported, mainContentNoteUnsupported, mainContentSubHeaderUnsupported, mainContentTextUnsupported, mainHeaderUnsupported: String? + var mainContentHeader, mainContentHeaderUnsupported, mainContentNote, mainContentNoteUnsupported, mainContentSubHeader, mainContentSubHeaderUnsupported: String? + var mainContentText, mainContentTextUnsupported, mainHeader, mainHeaderUnsupported: String? var oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, secondaryQuitButtonText, subHeader, subHeaderUnsupported, screenShotAltText: String? enum CodingKeys: String, CodingKey { case language = "_language" - case actionButtonText, customDeferralButtonText, customDeferralDropdownText, informationButtonText, mainContentHeader, mainContentNote, mainContentSubHeader, mainContentText, mainHeader, mainContentHeaderUnsupported, mainContentNoteUnsupported, mainContentSubHeaderUnsupported, mainContentTextUnsupported, mainHeaderUnsupported, oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, secondaryQuitButtonText, subHeader, subHeaderUnsupported, screenShotAltText + case actionButtonText, customDeferralButtonText, customDeferralDropdownText, informationButtonText, mainContentHeader, mainContentHeaderUnsupported, mainContentNote, mainContentNoteUnsupported, mainContentSubHeader, mainContentSubHeaderUnsupported, mainContentText, mainContentTextUnsupported, mainHeader, mainHeaderUnsupported, oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, secondaryQuitButtonText, subHeader, subHeaderUnsupported, screenShotAltText } } From cd53504025efb45ada4e0b065c7d401d0012f307 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 17 May 2024 09:59:53 -0500 Subject: [PATCH 024/141] flesh out device supported logic --- .../com.github.macadmins.Nudge.tester.json | 1 + .../Preferences/DefaultPreferencesNudge.swift | 6 +++ Nudge/Preferences/PreferencesStructure.swift | 4 +- Nudge/UI/Main.swift | 42 ++++++++++--------- Nudge/UI/SimpleMode/SimpleMode.swift | 6 ++- Nudge/UI/StandardMode/RightSide.swift | 24 ++++++----- Nudge/Utilities/UILogic.swift | 4 +- 7 files changed, 53 insertions(+), 34 deletions(-) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index 7c25629f..c1f4f6ba 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -10,6 +10,7 @@ "acceptableCameraUsage": true, "acceptableScreenSharingUsage": true, "attemptToBlockApplicationLaunches": true, + "attemptToCheckForSupportedDevice": true, "blockedApplicationBundleIDs": [ "com.microsoft.VSCode", "us.zoom.xos" diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 810d2b48..10c68fbc 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -113,6 +113,12 @@ struct OptionalFeatureVariables { false } + static var attemptToCheckForSupportedDevice: Bool { + optionalFeaturesProfile?["attemptToCheckForSupportedDevice"] as? Bool ?? + optionalFeaturesJSON?.attemptToCheckForSupportedDevice ?? + false + } + static var attemptToFetchMajorUpgrade: Bool { optionalFeaturesProfile?["attemptToFetchMajorUpgrade"] as? Bool ?? optionalFeaturesJSON?.attemptToFetchMajorUpgrade ?? diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index 464ff61b..d73221b7 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -51,7 +51,7 @@ extension NudgePreferences { // MARK: - OptionalFeatures struct OptionalFeatures: Codable { var acceptableApplicationBundleIDs, acceptableAssertionApplicationNames: [String]? - var acceptableAssertionUsage, acceptableCameraUsage, acceptableScreenSharingUsage, aggressiveUserExperience, aggressiveUserFullScreenExperience, asynchronousSoftwareUpdate, attemptToBlockApplicationLaunches, attemptToFetchMajorUpgrade: Bool? + var acceptableAssertionUsage, acceptableCameraUsage, acceptableScreenSharingUsage, aggressiveUserExperience, aggressiveUserFullScreenExperience, asynchronousSoftwareUpdate, attemptToBlockApplicationLaunches, attemptToCheckForSupportedDevice, attemptToFetchMajorUpgrade: Bool? var blockedApplicationBundleIDs: [String]? var disableSoftwareUpdateWorkflow, enforceMinorUpdates, terminateApplicationsOnLaunch, utilizeSOFAFeed: Bool? } @@ -84,6 +84,7 @@ extension OptionalFeatures { aggressiveUserFullScreenExperience: Bool? = nil, asynchronousSoftwareUpdate: Bool? = nil, attemptToBlockApplicationLaunches: Bool? = nil, + attemptToCheckForSupportedDevice: Bool? = nil, attemptToFetchMajorUpgrade: Bool? = nil, blockedApplicationBundleIDs: [String]? = nil, disableSoftwareUpdateWorkflow: Bool? = nil, @@ -101,6 +102,7 @@ extension OptionalFeatures { aggressiveUserFullScreenExperience: aggressiveUserFullScreenExperience ?? self.aggressiveUserFullScreenExperience, asynchronousSoftwareUpdate: asynchronousSoftwareUpdate ?? self.asynchronousSoftwareUpdate, attemptToBlockApplicationLaunches: attemptToBlockApplicationLaunches ?? self.attemptToBlockApplicationLaunches, + attemptToCheckForSupportedDevice: attemptToCheckForSupportedDevice ?? self.attemptToCheckForSupportedDevice, attemptToFetchMajorUpgrade: attemptToFetchMajorUpgrade ?? self.attemptToFetchMajorUpgrade, blockedApplicationBundleIDs: blockedApplicationBundleIDs ?? self.blockedApplicationBundleIDs, disableSoftwareUpdateWorkflow: disableSoftwareUpdateWorkflow ?? self.disableSoftwareUpdateWorkflow, diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 237792ed..0782e385 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -174,27 +174,31 @@ class AppDelegate: NSObject, NSApplicationDelegate { // Pre-Launch Logic func applicationWillFinishLaunching(_ notification: Notification) { // print("applicationWillFinishLaunching") - if let macOSSOFAAssets = Globals.sofaAssets?.osVersions { - // Find the first macOS asset that matches the product version - var foundMatch = false - // osVersions.map { $0.latest.productVersion } - for osVersion in macOSSOFAAssets { - if let matchingAsset = osVersion.securityReleases.first(where: { $0.productVersion == PrefsWrapper.requiredMinimumOSVersion }) { - foundMatch = true - // Check if the specified device is in the supported devices of the matching asset -// print("SOFA Matched OS Version: \(matchingAsset.productVersion)") -// print("SOFA Assets: \(matchingAsset.supportedDevices)") -// print("Assessed Model ID: \(Globals.hardwareModelID)") - // nudgePrimaryState.deviceSupportedByOSVersion = matchingAsset.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }) - nudgePrimaryState.deviceSupportedByOSVersion = false + if OptionalFeatureVariables.utilizeSOFAFeed { + if let macOSSOFAAssets = Globals.sofaAssets?.osVersions { + // Find the first macOS asset that matches the product version + var foundMatch = false + // osVersions.map { $0.latest.productVersion } + for osVersion in macOSSOFAAssets { + if let matchingAsset = osVersion.securityReleases.first(where: { $0.productVersion == PrefsWrapper.requiredMinimumOSVersion }) { + foundMatch = true + // Check if the specified device is in the supported devices of the matching asset + // print("SOFA Matched OS Version: \(matchingAsset.productVersion)") + // print("SOFA Assets: \(matchingAsset.supportedDevices)") + // print("Assessed Model ID: \(Globals.hardwareModelID)") + if OptionalFeatureVariables.attemptToCheckForSupportedDevice { + nudgePrimaryState.deviceSupportedByOSVersion = matchingAsset.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }) + nudgePrimaryState.deviceSupportedByOSVersion = false + } + } } + if !foundMatch { + // If no matching product version found or the device is not supported, return false + print("DeviceSOFASupported: False") + } + } else { + print("No macOS SOFA assets available.") } - if !foundMatch { - // If no matching product version found or the device is not supported, return false - print("DeviceSOFASupported: False") - } - } else { - print("No macOS SOFA assets available.") } handleSMAppService() checkForBadProfilePath() diff --git a/Nudge/UI/SimpleMode/SimpleMode.swift b/Nudge/UI/SimpleMode/SimpleMode.swift index 2e5236db..5ef26dce 100644 --- a/Nudge/UI/SimpleMode/SimpleMode.swift +++ b/Nudge/UI/SimpleMode/SimpleMode.swift @@ -30,7 +30,7 @@ struct SimpleMode: View { CompanyLogo() Spacer() - Text(appState.deviceSupportedByOSVersion == true ? getMainHeader().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : getMainHeaderUnsupported().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(appState.deviceSupportedByOSVersion ? getMainHeader().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : getMainHeaderUnsupported().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .font(.title) remainingTimeView @@ -40,7 +40,9 @@ struct SimpleMode: View { } Spacer() - updateButton + if appState.deviceSupportedByOSVersion { + updateButton + } Spacer() } } diff --git a/Nudge/UI/StandardMode/RightSide.swift b/Nudge/UI/StandardMode/RightSide.swift index e6b67bb8..cdba747d 100644 --- a/Nudge/UI/StandardMode/RightSide.swift +++ b/Nudge/UI/StandardMode/RightSide.swift @@ -38,7 +38,7 @@ struct StandardModeRightSide: View { HStack { VStack(alignment: .leading, spacing: 5) { HStack { - Text(appState.deviceSupportedByOSVersion == true ? getMainHeader().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : getMainHeaderUnsupported().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(appState.deviceSupportedByOSVersion ? getMainHeader().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : getMainHeaderUnsupported().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .font(.largeTitle) .minimumScaleFactor(0.5) .frame(maxHeight: 25) @@ -46,7 +46,7 @@ struct StandardModeRightSide: View { } HStack { - Text(appState.deviceSupportedByOSVersion == true ? UserInterfaceVariables.subHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.subHeaderUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(appState.deviceSupportedByOSVersion ? UserInterfaceVariables.subHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.subHeaderUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .font(.body) .fontWeight(.bold) .lineLimit(1) @@ -64,31 +64,33 @@ struct StandardModeRightSide: View { HStack(alignment: .center) { VStack(alignment: .leading, spacing: 1) { HStack { - Text(appState.deviceSupportedByOSVersion == true ? UserInterfaceVariables.mainContentHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentHeaderUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(appState.deviceSupportedByOSVersion ? UserInterfaceVariables.mainContentHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentHeaderUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .font(.callout) .fontWeight(.bold) Spacer() } HStack { - Text(appState.deviceSupportedByOSVersion == true ? UserInterfaceVariables.mainContentSubHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentSubHeaderUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(appState.deviceSupportedByOSVersion ? UserInterfaceVariables.mainContentSubHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentSubHeaderUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .font(.callout) Spacer() } } Spacer() - Button(action: { - UIUtilities().updateDevice() - }) { - Text(UserInterfaceVariables.actionButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + if appState.deviceSupportedByOSVersion { + Button(action: { + UIUtilities().updateDevice() + }) { + Text(UserInterfaceVariables.actionButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + } + .keyboardShortcut(.defaultAction) } - .keyboardShortcut(.defaultAction) } Divider() HStack { - Text(appState.deviceSupportedByOSVersion == true ? UserInterfaceVariables.mainContentNote.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentNoteUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(appState.deviceSupportedByOSVersion ? UserInterfaceVariables.mainContentNote.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentNoteUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .font(.callout) .fontWeight(.bold) .foregroundColor(appState.differentiateWithoutColor ? .accessibleRed : .red) @@ -98,7 +100,7 @@ struct StandardModeRightSide: View { ScrollView(.vertical) { VStack { HStack { - Text(appState.deviceSupportedByOSVersion == true ? UserInterfaceVariables.mainContentText.replacingOccurrences(of: "\\n", with: "\n").localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentTextUnsupported.replacingOccurrences(of: "\\n", with: "\n").localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(appState.deviceSupportedByOSVersion ? UserInterfaceVariables.mainContentText.replacingOccurrences(of: "\\n", with: "\n").localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentTextUnsupported.replacingOccurrences(of: "\\n", with: "\n").localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .font(.callout) .multilineTextAlignment(.leading) Spacer() diff --git a/Nudge/Utilities/UILogic.swift b/Nudge/Utilities/UILogic.swift index 4764ec87..8414b272 100644 --- a/Nudge/Utilities/UILogic.swift +++ b/Nudge/Utilities/UILogic.swift @@ -245,7 +245,9 @@ private func shouldActivateNudgeBasedOnAggressiveExperience(_ runningApplication } AppStateManager().activateNudge() if !CommandLineUtilities().unitTestingEnabled() { - UIUtilities().updateDevice(userClicked: false) + if nudgePrimaryState.deviceSupportedByOSVersion { + UIUtilities().updateDevice(userClicked: false) + } } return true } else { From 098d361b979fe24d4f95942804649ff1d05a5bfd Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 17 May 2024 10:09:53 -0500 Subject: [PATCH 025/141] add sofa logging events --- Nudge/UI/Main.swift | 18 +++++++++--------- Nudge/Utilities/Logger.swift | 1 + 2 files changed, 10 insertions(+), 9 deletions(-) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 0782e385..30b6e8cc 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -176,28 +176,28 @@ class AppDelegate: NSObject, NSApplicationDelegate { // print("applicationWillFinishLaunching") if OptionalFeatureVariables.utilizeSOFAFeed { if let macOSSOFAAssets = Globals.sofaAssets?.osVersions { - // Find the first macOS asset that matches the product version var foundMatch = false - // osVersions.map { $0.latest.productVersion } for osVersion in macOSSOFAAssets { if let matchingAsset = osVersion.securityReleases.first(where: { $0.productVersion == PrefsWrapper.requiredMinimumOSVersion }) { - foundMatch = true // Check if the specified device is in the supported devices of the matching asset - // print("SOFA Matched OS Version: \(matchingAsset.productVersion)") - // print("SOFA Assets: \(matchingAsset.supportedDevices)") - // print("Assessed Model ID: \(Globals.hardwareModelID)") + LogManager.notice("SOFA Matched OS Version: \(matchingAsset.productVersion)", logger: sofaLog) + LogManager.notice("SOFA Assets: \(matchingAsset.supportedDevices)", logger: sofaLog) if OptionalFeatureVariables.attemptToCheckForSupportedDevice { - nudgePrimaryState.deviceSupportedByOSVersion = matchingAsset.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }) + LogManager.notice("Assessed Model ID: \(Globals.hardwareModelID)", logger: sofaLog) + let deviceMatchFound = matchingAsset.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }) + LogManager.notice("Assessed Model ID found in SOFA Entry: \(deviceMatchFound)", logger: sofaLog) + nudgePrimaryState.deviceSupportedByOSVersion = deviceMatchFound nudgePrimaryState.deviceSupportedByOSVersion = false } + foundMatch = true } } if !foundMatch { // If no matching product version found or the device is not supported, return false - print("DeviceSOFASupported: False") + LogManager.notice("Could not find requiredMinimumOSVersion in SOFA feed", logger: sofaLog) } } else { - print("No macOS SOFA assets available.") + LogManager.notice("Could not fetch SOFA feed", logger: sofaLog) } } handleSMAppService() diff --git a/Nudge/Utilities/Logger.swift b/Nudge/Utilities/Logger.swift index 5d3cd5a6..eab199a6 100644 --- a/Nudge/Utilities/Logger.swift +++ b/Nudge/Utilities/Logger.swift @@ -47,6 +47,7 @@ let uiLog = LogManager.createLogger(category: "user-interface") let softwareupdateDeviceLog = LogManager.createLogger(category: "softwareupdate-device") let softwareupdateListLog = LogManager.createLogger(category: "softwareupdate-list") let softwareupdateDownloadLog = LogManager.createLogger(category: "softwareupdate-download") +let sofaLog = LogManager.createLogger(category: "sofa") // Log State class LogState { From 335b1bf0410cef4dd5b98316c5301977ddc3fe13 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 17 May 2024 10:33:12 -0500 Subject: [PATCH 026/141] add unsupportedURL and informationButtonTextUnsupported --- .../com.github.macadmins.Nudge.tester.json | 4 +- .../Preferences/DefaultPreferencesNudge.swift | 12 ++++ Nudge/Preferences/PreferencesStructure.swift | 68 +++++++++++++++++-- Nudge/UI/Common/InformationButton.swift | 32 ++++++--- Nudge/Utilities/Preferences.swift | 21 ++++++ Nudge/Utilities/Utils.swift | 8 +++ 6 files changed, 130 insertions(+), 15 deletions(-) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index c1f4f6ba..264b4008 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -22,7 +22,8 @@ { "aboutUpdateURL": "https://apple.com", "requiredInstallationDate": "2025-01-01T00:00:00Z", - "requiredMinimumOSVersion": "14.5" + "requiredMinimumOSVersion": "14.5", + "unsupportedURL": "https://google.com", } ], "userExperience": { @@ -45,6 +46,7 @@ "customDeferralButtonText": "customDeferralButtonText", "customDeferralDropdownText": "customDeferralDropdownText", "informationButtonText": "informationButtonText", + "informationButtonTextUnsupported": "informationButtonTextUnsupported", "mainContentHeader": "mainContentHeader", "mainContentHeaderUnsupported": "mainContentHeaderUnsupported", "mainContentNote": "mainContentNote", diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 10c68fbc..b13bbbb8 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -179,6 +179,12 @@ struct OSVersionRequirementVariables { static var requiredMinimumOSVersion: String { try! OSVersion(PrefsWrapper.requiredMinimumOSVersion).description } + + static var unsupportedURL: String { + getUnsupportedURL(OSVerReq: osVersionRequirementsProfile) ?? + getUnsupportedURL(OSVerReq: osVersionRequirementsJSON) ?? + "" + } } @@ -373,6 +379,12 @@ struct UserInterfaceVariables { "More Info" } + static var informationButtonTextUnsupported: String { + userInterfaceUpdateElementsProfile?["informationButtonTextUnsupported"] as? String ?? + userInterfaceUpdateElementsJSON?.informationButtonTextUnsupported ?? + "Replace Your Device" + } + static var mainContentHeader: String { userInterfaceUpdateElementsProfile?["mainContentHeader"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentHeader ?? diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index d73221b7..4de187db 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -120,6 +120,8 @@ struct OSVersionRequirement: Codable { var actionButtonPath, majorUpgradeAppPath: String? var requiredInstallationDate: Date? var requiredMinimumOSVersion, targetedOSVersionsRule: String? + var unsupportedURL: String? + var unsupportedURLs: [UnsupportedURL]? } // MARK: OSVersionRequirement convenience initializers and mutators @@ -130,8 +132,9 @@ extension OSVersionRequirement { self.majorUpgradeAppPath = fromDictionary["majorUpgradeAppPath"] as? String self.requiredMinimumOSVersion = fromDictionary["requiredMinimumOSVersion"] as? String self.targetedOSVersionsRule = fromDictionary["targetedOSVersionsRule"] as? String + self.unsupportedURL = fromDictionary["unsupportedURL"] as? String - // Handling AboutUpdateURLs + // Handling AboutUpdateURLs and UnsupportedURLs if let aboutURLs = fromDictionary["aboutUpdateURLs"] as? [[String: String]] { self.aboutUpdateURLs = aboutURLs.compactMap { dict in guard let language = dict["_language"], let url = dict["aboutUpdateURL"] else { return nil } @@ -141,6 +144,15 @@ extension OSVersionRequirement { self.aboutUpdateURLs = [] } + if let unsupportedURLs = fromDictionary["unsupportedURLs"] as? [[String: String]] { + self.unsupportedURLs = unsupportedURLs.compactMap { dict in + guard let language = dict["_language"], let url = dict["unsupportedURL"] else { return nil } + return UnsupportedURL(language: language, unsupportedURL: url) + } + } else { + self.unsupportedURLs = [] + } + // Handling requiredInstallationDate // Jamf JSON Schema for mobileconfigurations do not support Date types (JSON does not support it) // In order to support this, an admin would need to pass a string and then coerce it into our Date format @@ -177,7 +189,9 @@ extension OSVersionRequirement { majorUpgradeAppPath: String? = nil, requiredInstallationDate: Date? = nil, requiredMinimumOSVersion: String? = nil, - targetedOSVersionsRule: String? = nil + targetedOSVersionsRule: String? = nil, + unsupportedURL: String? = nil, + unsupportedURLs: [UnsupportedURL]? = nil ) -> OSVersionRequirement { return OSVersionRequirement( aboutUpdateURL: aboutUpdateURL ?? self.aboutUpdateURL, @@ -186,7 +200,9 @@ extension OSVersionRequirement { majorUpgradeAppPath: majorUpgradeAppPath ?? self.majorUpgradeAppPath, requiredInstallationDate: requiredInstallationDate ?? self.requiredInstallationDate, requiredMinimumOSVersion: requiredMinimumOSVersion ?? self.requiredMinimumOSVersion, - targetedOSVersionsRule: targetedOSVersionsRule ?? self.targetedOSVersionsRule + targetedOSVersionsRule: targetedOSVersionsRule ?? self.targetedOSVersionsRule, + unsupportedURL: unsupportedURL ?? self.unsupportedURL, + unsupportedURLs: unsupportedURLs ?? self.unsupportedURLs ) } } @@ -231,6 +247,46 @@ extension AboutUpdateURL { } } +// MARK: - UnsupportedURL +struct UnsupportedURL: Codable { + var language: String? + var unsupportedURL: String? + + enum CodingKeys: String, CodingKey { + case language = "_language" + case unsupportedURL + } +} + +// MARK: UnsupportedURL convenience initializers and mutators +extension UnsupportedURL { + init(data: Data) throws { + self = try JSONDecoder().decode(UnsupportedURL.self, from: data) + } + + init(_ json: String, using encoding: String.Encoding = .utf8) throws { + guard let data = json.data(using: encoding) else { + throw NSError(domain: "JSONDecoding", code: 0, userInfo: [NSLocalizedDescriptionKey: "Invalid JSON string."]) + } + try self.init(data: data) + } + + init(fromURL url: URL) throws { + let data = try Data(contentsOf: url) + try self.init(data: data) + } + + func with( + language: String? = nil, + unsupportedURL: String? = nil + ) -> UnsupportedURL { + return UnsupportedURL( + language: language ?? self.language, + unsupportedURL: unsupportedURL ?? self.unsupportedURL + ) + } +} + // MARK: - UserExperience struct UserExperience: Codable { var allowGracePeriods, allowLaterDeferralButton, allowMovableWindow, allowUserQuitDeferrals: Bool? @@ -376,14 +432,14 @@ extension UserInterface { // MARK: - UpdateElement struct UpdateElement: Codable { - var language, actionButtonText, customDeferralButtonText, customDeferralDropdownText, informationButtonText: String? + var language, actionButtonText, customDeferralButtonText, customDeferralDropdownText, informationButtonText, informationButtonTextUnsupported: String? var mainContentHeader, mainContentHeaderUnsupported, mainContentNote, mainContentNoteUnsupported, mainContentSubHeader, mainContentSubHeaderUnsupported: String? var mainContentText, mainContentTextUnsupported, mainHeader, mainHeaderUnsupported: String? var oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, secondaryQuitButtonText, subHeader, subHeaderUnsupported, screenShotAltText: String? enum CodingKeys: String, CodingKey { case language = "_language" - case actionButtonText, customDeferralButtonText, customDeferralDropdownText, informationButtonText, mainContentHeader, mainContentHeaderUnsupported, mainContentNote, mainContentNoteUnsupported, mainContentSubHeader, mainContentSubHeaderUnsupported, mainContentText, mainContentTextUnsupported, mainHeader, mainHeaderUnsupported, oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, secondaryQuitButtonText, subHeader, subHeaderUnsupported, screenShotAltText + case actionButtonText, customDeferralButtonText, customDeferralDropdownText, informationButtonText, informationButtonTextUnsupported, mainContentHeader, mainContentHeaderUnsupported, mainContentNote, mainContentNoteUnsupported, mainContentSubHeader, mainContentSubHeaderUnsupported, mainContentText, mainContentTextUnsupported, mainHeader, mainHeaderUnsupported, oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, secondaryQuitButtonText, subHeader, subHeaderUnsupported, screenShotAltText } } @@ -411,6 +467,7 @@ extension UpdateElement { customDeferralButtonText: String? = nil, customDeferralDropdownText: String? = nil, informationButtonText: String? = nil, + informationButtonTextUnsupported: String? = nil, mainContentHeader: String? = nil, mainContentHeaderUnsupported: String? = nil, mainContentNote: String? = nil, @@ -435,6 +492,7 @@ extension UpdateElement { customDeferralButtonText: customDeferralButtonText ?? self.customDeferralButtonText, customDeferralDropdownText: customDeferralDropdownText ?? self.customDeferralDropdownText, informationButtonText: informationButtonText ?? self.informationButtonText, + informationButtonTextUnsupported: informationButtonTextUnsupported ?? self.informationButtonTextUnsupported, mainContentHeader: mainContentHeader ?? self.mainContentHeader, mainContentHeaderUnsupported: mainContentHeaderUnsupported ?? self.mainContentHeaderUnsupported, mainContentNote: mainContentNote ?? self.mainContentNote, diff --git a/Nudge/UI/Common/InformationButton.swift b/Nudge/UI/Common/InformationButton.swift index 3b9bfeb0..41c742c8 100644 --- a/Nudge/UI/Common/InformationButton.swift +++ b/Nudge/UI/Common/InformationButton.swift @@ -19,17 +19,31 @@ struct InformationButton: View { } private var informationButton: some View { - guard OSVersionRequirementVariables.aboutUpdateURL != "" else { return AnyView(EmptyView()) } - - return AnyView( - Button(action: UIUtilities().openMoreInfo) { - Text(UserInterfaceVariables.informationButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) - .foregroundColor(dynamicTextColor) - } + if appState.deviceSupportedByOSVersion { + guard OSVersionRequirementVariables.aboutUpdateURL != "" else { return AnyView(EmptyView()) } + + return AnyView( + Button(action: UIUtilities().openMoreInfo) { + Text(UserInterfaceVariables.informationButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + .foregroundColor(dynamicTextColor) + } + .buttonStyle(.plain) + .help("Click for more information about the security update".localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + .onHoverEffect() + ) + } else { + guard OSVersionRequirementVariables.unsupportedURL != "" else { return AnyView(EmptyView()) } + + return AnyView( + Button(action: UIUtilities().openMoreInfoUnsupported) { + Text(UserInterfaceVariables.informationButtonTextUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + .foregroundColor(dynamicTextColor) + } .buttonStyle(.plain) - .help("Click for more information about the security update".localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + .help("Click for more information about replacing your device".localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .onHoverEffect() - ) + ) + } } private var dynamicTextColor: Color { diff --git a/Nudge/Utilities/Preferences.swift b/Nudge/Utilities/Preferences.swift index 09eaf5d8..9a037ea5 100644 --- a/Nudge/Utilities/Preferences.swift +++ b/Nudge/Utilities/Preferences.swift @@ -122,6 +122,27 @@ private func getOSVersionRequirements(from requirements: [OSVersionRequirement]? return fullMatch ?? partialMatch ?? defaultMatch ?? nil } +func getUnsupportedURL(OSVerReq: OSVersionRequirement?) -> String? { + if CommandLineUtilities().demoModeEnabled() || CommandLineUtilities().unitTestingEnabled() { + return "https://apple.com" + } + + if let update = OSVerReq?.unsupportedURL { + return update + } + + let desiredLanguage = getDesiredLanguage() + if let updates = OSVerReq?.unsupportedURLs { + for subUpdate in updates { + if subUpdate.language == desiredLanguage { + return subUpdate.unsupportedURL ?? "" + } + } + } + + return nil +} + // userExperience // Even if profile/JSON is installed, return nil if in demo-mode func getUserExperienceJSON() -> UserExperience? { diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 3b6d92af..da29da00 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -1059,6 +1059,14 @@ struct UIUtilities { NSWorkspace.shared.open(url) } + func openMoreInfoUnsupported() { + guard let url = URL(string: OSVersionRequirementVariables.unsupportedURL) else { + return + } + LogManager.notice("User clicked moreInfo button in unsupported state", logger: uiLog) + NSWorkspace.shared.open(url) + } + private func postUpdateDeviceActions(userClicked: Bool) { if userClicked { LogManager.notice("User clicked updateDevice", logger: uiLog) From fc0ce05aab8d572c2f9a7ca1d67fcbe2739288cb Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 17 May 2024 11:12:07 -0500 Subject: [PATCH 027/141] start testing for actively exploited cve info --- Nudge/UI/Main.swift | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 30b6e8cc..b476c2f3 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -178,13 +178,17 @@ class AppDelegate: NSObject, NSApplicationDelegate { if let macOSSOFAAssets = Globals.sofaAssets?.osVersions { var foundMatch = false for osVersion in macOSSOFAAssets { - if let matchingAsset = osVersion.securityReleases.first(where: { $0.productVersion == PrefsWrapper.requiredMinimumOSVersion }) { + var test = PrefsWrapper.requiredMinimumOSVersion + test = "14.4" + if let matchingAsset = osVersion.securityReleases.first(where: { $0.productVersion == test }) { // Check if the specified device is in the supported devices of the matching asset LogManager.notice("SOFA Matched OS Version: \(matchingAsset.productVersion)", logger: sofaLog) LogManager.notice("SOFA Assets: \(matchingAsset.supportedDevices)", logger: sofaLog) if OptionalFeatureVariables.attemptToCheckForSupportedDevice { LogManager.notice("Assessed Model ID: \(Globals.hardwareModelID)", logger: sofaLog) let deviceMatchFound = matchingAsset.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }) + print("CVEs: \(matchingAsset.cves)") + print("Actively Exploited CVEs: \(matchingAsset.activelyExploitedCVEs.count > 0)") LogManager.notice("Assessed Model ID found in SOFA Entry: \(deviceMatchFound)", logger: sofaLog) nudgePrimaryState.deviceSupportedByOSVersion = deviceMatchFound nudgePrimaryState.deviceSupportedByOSVersion = false From 0fb6528ed524a15b12911b473edc1b818922f413 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 17 May 2024 11:14:56 -0500 Subject: [PATCH 028/141] TODOs so I don't forget --- Nudge/UI/Main.swift | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index b476c2f3..4f746d00 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -174,6 +174,12 @@ class AppDelegate: NSObject, NSApplicationDelegate { // Pre-Launch Logic func applicationWillFinishLaunching(_ notification: Notification) { // print("applicationWillFinishLaunching") + // TODO: Implement SOFA Caching + // TODO: Implement requiredInstallationDate values SLAs for Actively Exploited vs Not based on ReleaseDate sofa key + // TODO: Implement Actively Exploited sidebar info on standardMode + // TODO: Implement "latest" for sofa vs hardcoded requiredInstallationVersion + // TODO: Add more logging to "unsupported devices" UI. + // TODO: Add localization for "unsupported devices" text fields if OptionalFeatureVariables.utilizeSOFAFeed { if let macOSSOFAAssets = Globals.sofaAssets?.osVersions { var foundMatch = false From 027450e5a91c9c510aa42857a238d747aa4c6031 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 17 May 2024 11:18:34 -0500 Subject: [PATCH 029/141] more todo notes --- Nudge/UI/Main.swift | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 4f746d00..9874349a 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -174,12 +174,14 @@ class AppDelegate: NSObject, NSApplicationDelegate { // Pre-Launch Logic func applicationWillFinishLaunching(_ notification: Notification) { // print("applicationWillFinishLaunching") - // TODO: Implement SOFA Caching + // TODO: Implement SOFA Caching with key refreshSOFAFeedTime // TODO: Implement requiredInstallationDate values SLAs for Actively Exploited vs Not based on ReleaseDate sofa key + // TODO: activelyExploitedInstallationSLA, standardInstallationSLA // TODO: Implement Actively Exploited sidebar info on standardMode // TODO: Implement "latest" for sofa vs hardcoded requiredInstallationVersion // TODO: Add more logging to "unsupported devices" UI. // TODO: Add localization for "unsupported devices" text fields + // TODO: Get someone to update JAMF JSON schema for all the new keys and wiki if OptionalFeatureVariables.utilizeSOFAFeed { if let macOSSOFAAssets = Globals.sofaAssets?.osVersions { var foundMatch = false From ad4247bfe600668fa09715387cbf60e08b357502 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 17 May 2024 12:59:33 -0500 Subject: [PATCH 030/141] handle new sofa values for "Latest" --- Nudge/3rd Party Assets/sofa.swift | 8 ++++++++ Nudge/UI/Main.swift | 29 +++++++++++++++++++++++------ 2 files changed, 31 insertions(+), 6 deletions(-) diff --git a/Nudge/3rd Party Assets/sofa.swift b/Nudge/3rd Party Assets/sofa.swift index 085a3b1f..8c066a0b 100644 --- a/Nudge/3rd Party Assets/sofa.swift +++ b/Nudge/3rd Party Assets/sofa.swift @@ -44,6 +44,10 @@ struct LatestOS: Codable { let releaseDate: Date? let expirationDate: Date let supportedDevices: [String] + let securityInfo: String + let cves: [String: Bool] + let activelyExploitedCVEs: [String] + let uniqueCVEsCount: Int enum CodingKeys: String, CodingKey { case productVersion = "ProductVersion" @@ -51,6 +55,10 @@ struct LatestOS: Codable { case releaseDate = "ReleaseDate" case expirationDate = "ExpirationDate" case supportedDevices = "SupportedDevices" + case securityInfo = "SecurityInfo" + case cves = "CVEs" + case activelyExploitedCVEs = "ActivelyExploitedCVEs" + case uniqueCVEsCount = "UniqueCVEsCount" } } diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 9874349a..cb1dcaae 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -188,20 +188,37 @@ class AppDelegate: NSObject, NSApplicationDelegate { for osVersion in macOSSOFAAssets { var test = PrefsWrapper.requiredMinimumOSVersion test = "14.4" - if let matchingAsset = osVersion.securityReleases.first(where: { $0.productVersion == test }) { + test = "latest" + if test == "latest" { // Check if the specified device is in the supported devices of the matching asset - LogManager.notice("SOFA Matched OS Version: \(matchingAsset.productVersion)", logger: sofaLog) - LogManager.notice("SOFA Assets: \(matchingAsset.supportedDevices)", logger: sofaLog) + LogManager.notice("SOFA Matched OS Version: \(osVersion.latest.productVersion)", logger: sofaLog) + LogManager.notice("SOFA Assets: \(osVersion.latest.supportedDevices)", logger: sofaLog) if OptionalFeatureVariables.attemptToCheckForSupportedDevice { LogManager.notice("Assessed Model ID: \(Globals.hardwareModelID)", logger: sofaLog) - let deviceMatchFound = matchingAsset.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }) - print("CVEs: \(matchingAsset.cves)") - print("Actively Exploited CVEs: \(matchingAsset.activelyExploitedCVEs.count > 0)") + let deviceMatchFound = osVersion.latest.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }) + print("CVEs: \(osVersion.latest.cves)") + print("Actively Exploited CVEs: \(osVersion.latest.activelyExploitedCVEs.count > 0)") LogManager.notice("Assessed Model ID found in SOFA Entry: \(deviceMatchFound)", logger: sofaLog) nudgePrimaryState.deviceSupportedByOSVersion = deviceMatchFound nudgePrimaryState.deviceSupportedByOSVersion = false } foundMatch = true + } else { + if let matchingAsset = osVersion.securityReleases.first(where: { $0.productVersion == test }) { + // Check if the specified device is in the supported devices of the matching asset + LogManager.notice("SOFA Matched OS Version: \(matchingAsset.productVersion)", logger: sofaLog) + LogManager.notice("SOFA Assets: \(matchingAsset.supportedDevices)", logger: sofaLog) + if OptionalFeatureVariables.attemptToCheckForSupportedDevice { + LogManager.notice("Assessed Model ID: \(Globals.hardwareModelID)", logger: sofaLog) + let deviceMatchFound = matchingAsset.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }) + print("CVEs: \(matchingAsset.cves)") + print("Actively Exploited CVEs: \(matchingAsset.activelyExploitedCVEs.count > 0)") + LogManager.notice("Assessed Model ID found in SOFA Entry: \(deviceMatchFound)", logger: sofaLog) + nudgePrimaryState.deviceSupportedByOSVersion = deviceMatchFound + nudgePrimaryState.deviceSupportedByOSVersion = false + } + foundMatch = true + } } } if !foundMatch { From d290f0e64f81a2281643c8675ac5ea7bc0b1cb27 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 17 May 2024 14:18:32 -0500 Subject: [PATCH 031/141] first attempt at implementing "latest" with ui --- .../com.github.macadmins.Nudge.tester.json | 2 +- .../Preferences/DefaultPreferencesNudge.swift | 6 +++++- Nudge/UI/Defaults.swift | 1 + Nudge/UI/Main.swift | 18 ++++++++++-------- Nudge/UI/StandardMode/LeftSide.swift | 2 +- Nudge/Utilities/SoftwareUpdate.swift | 8 ++++---- Nudge/Utilities/Utils.swift | 8 ++++---- 7 files changed, 26 insertions(+), 19 deletions(-) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index 264b4008..9bcfed7e 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -22,7 +22,7 @@ { "aboutUpdateURL": "https://apple.com", "requiredInstallationDate": "2025-01-01T00:00:00Z", - "requiredMinimumOSVersion": "14.5", + "requiredMinimumOSVersion": "latest", "unsupportedURL": "https://google.com", } ], diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index b13bbbb8..dbbbf4ac 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -177,7 +177,11 @@ struct OSVersionRequirementVariables { } static var requiredMinimumOSVersion: String { - try! OSVersion(PrefsWrapper.requiredMinimumOSVersion).description + if PrefsWrapper.requiredMinimumOSVersion == "latest" { + PrefsWrapper.requiredMinimumOSVersion + } else { + try! OSVersion(PrefsWrapper.requiredMinimumOSVersion).description + } } static var unsupportedURL: String { diff --git a/Nudge/UI/Defaults.swift b/Nudge/UI/Defaults.swift index 175f21e8..e92079c4 100644 --- a/Nudge/UI/Defaults.swift +++ b/Nudge/UI/Defaults.swift @@ -78,6 +78,7 @@ class AppState: ObservableObject { @Published var hoursRemaining = DateManager().getNumberOfHoursRemaining() @Published var lastRefreshTime = DateManager().getFormattedDate() @Published var requireDualQuitButtons = false + @Published var requiredMinimumOSVersion = OSVersionRequirementVariables.requiredMinimumOSVersion @Published var shouldExit = false @Published var timerCycle = 0 @Published var userDeferrals = Globals.nudgeDefaults.object(forKey: "userDeferrals") as? Int ?? 0 diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index cb1dcaae..c71d8047 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -179,6 +179,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { // TODO: activelyExploitedInstallationSLA, standardInstallationSLA // TODO: Implement Actively Exploited sidebar info on standardMode // TODO: Implement "latest" for sofa vs hardcoded requiredInstallationVersion + // TODO: Implement "latest-minor" or something for implementing all of the minor releases. // TODO: Add more logging to "unsupported devices" UI. // TODO: Add localization for "unsupported devices" text fields // TODO: Get someone to update JAMF JSON schema for all the new keys and wiki @@ -186,11 +187,9 @@ class AppDelegate: NSObject, NSApplicationDelegate { if let macOSSOFAAssets = Globals.sofaAssets?.osVersions { var foundMatch = false for osVersion in macOSSOFAAssets { - var test = PrefsWrapper.requiredMinimumOSVersion - test = "14.4" - test = "latest" - if test == "latest" { + if PrefsWrapper.requiredMinimumOSVersion == "latest" { // Check if the specified device is in the supported devices of the matching asset + nudgePrimaryState.requiredMinimumOSVersion = osVersion.latest.productVersion LogManager.notice("SOFA Matched OS Version: \(osVersion.latest.productVersion)", logger: sofaLog) LogManager.notice("SOFA Assets: \(osVersion.latest.supportedDevices)", logger: sofaLog) if OptionalFeatureVariables.attemptToCheckForSupportedDevice { @@ -200,11 +199,11 @@ class AppDelegate: NSObject, NSApplicationDelegate { print("Actively Exploited CVEs: \(osVersion.latest.activelyExploitedCVEs.count > 0)") LogManager.notice("Assessed Model ID found in SOFA Entry: \(deviceMatchFound)", logger: sofaLog) nudgePrimaryState.deviceSupportedByOSVersion = deviceMatchFound - nudgePrimaryState.deviceSupportedByOSVersion = false + // nudgePrimaryState.deviceSupportedByOSVersion = false } foundMatch = true } else { - if let matchingAsset = osVersion.securityReleases.first(where: { $0.productVersion == test }) { + if let matchingAsset = osVersion.securityReleases.first(where: { $0.productVersion == nudgePrimaryState.requiredMinimumOSVersion }) { // Check if the specified device is in the supported devices of the matching asset LogManager.notice("SOFA Matched OS Version: \(matchingAsset.productVersion)", logger: sofaLog) LogManager.notice("SOFA Assets: \(matchingAsset.supportedDevices)", logger: sofaLog) @@ -215,15 +214,18 @@ class AppDelegate: NSObject, NSApplicationDelegate { print("Actively Exploited CVEs: \(matchingAsset.activelyExploitedCVEs.count > 0)") LogManager.notice("Assessed Model ID found in SOFA Entry: \(deviceMatchFound)", logger: sofaLog) nudgePrimaryState.deviceSupportedByOSVersion = deviceMatchFound - nudgePrimaryState.deviceSupportedByOSVersion = false + // nudgePrimaryState.deviceSupportedByOSVersion = false } foundMatch = true } } + if foundMatch { + break + } } if !foundMatch { // If no matching product version found or the device is not supported, return false - LogManager.notice("Could not find requiredMinimumOSVersion in SOFA feed", logger: sofaLog) + LogManager.notice("Could not find requiredMinimumOSVersion \(nudgePrimaryState.requiredMinimumOSVersion) in SOFA feed", logger: sofaLog) } } else { LogManager.notice("Could not fetch SOFA feed", logger: sofaLog) diff --git a/Nudge/UI/StandardMode/LeftSide.swift b/Nudge/UI/StandardMode/LeftSide.swift index a57a9c05..16ad91f7 100644 --- a/Nudge/UI/StandardMode/LeftSide.swift +++ b/Nudge/UI/StandardMode/LeftSide.swift @@ -36,7 +36,7 @@ struct StandardModeLeftSide: View { private var informationStack: some View { VStack(alignment: .center, spacing: interLineSpacing) { - InfoRow(label: "Required OS Version:", value: String(OSVersionRequirementVariables.requiredMinimumOSVersion), boldText: true) + InfoRow(label: "Required OS Version:", value: String(appState.requiredMinimumOSVersion), boldText: true) InfoRow(label: "Current OS Version:", value: GlobalVariables.currentOSVersion) remainingTimeRow if UserInterfaceVariables.showDeferralCount { diff --git a/Nudge/Utilities/SoftwareUpdate.swift b/Nudge/Utilities/SoftwareUpdate.swift index c3c551e1..6955edcd 100644 --- a/Nudge/Utilities/SoftwareUpdate.swift +++ b/Nudge/Utilities/SoftwareUpdate.swift @@ -34,7 +34,7 @@ class SoftwareUpdate { if OptionalFeatureVariables.attemptToFetchMajorUpgrade, !majorUpgradeAppPathExists, !majorUpgradeBackupAppPathExists { LogManager.notice("Device requires major upgrade - attempting download", logger: softwareupdateListLog) - let (output, error, exitCode) = SubProcessUtilities().runProcess(launchPath: "/usr/sbin/softwareupdate", arguments: ["--fetch-full-installer", "--full-installer-version", OSVersionRequirementVariables.requiredMinimumOSVersion]) + let (output, error, exitCode) = SubProcessUtilities().runProcess(launchPath: "/usr/sbin/softwareupdate", arguments: ["--fetch-full-installer", "--full-installer-version", nudgePrimaryState.requiredMinimumOSVersion]) if exitCode != 0 { LogManager.error("Error downloading software update: \(error)", logger: softwareupdateDownloadLog) @@ -54,8 +54,8 @@ class SoftwareUpdate { let softwareupdateList = self.list() let updateLabel = extractUpdateLabel(from: softwareupdateList) - if !softwareupdateList.contains(OSVersionRequirementVariables.requiredMinimumOSVersion) || updateLabel.isEmpty { - LogManager.notice("Software update did not find \(OSVersionRequirementVariables.requiredMinimumOSVersion) available for download - skipping download attempt", logger: softwareupdateListLog) + if !softwareupdateList.contains(nudgePrimaryState.requiredMinimumOSVersion) || updateLabel.isEmpty { + LogManager.notice("Software update did not find \(nudgePrimaryState.requiredMinimumOSVersion) available for download - skipping download attempt", logger: softwareupdateListLog) return } @@ -77,7 +77,7 @@ class SoftwareUpdate { for line in lines { if line.contains("Label:") { let labelPart = line.split(separator: ":").map { $0.trimmingCharacters(in: .whitespaces) } - if labelPart.count > 1 && labelPart[1].contains(OSVersionRequirementVariables.requiredMinimumOSVersion) { + if labelPart.count > 1 && labelPart[1].contains(nudgePrimaryState.requiredMinimumOSVersion) { updateLabel = labelPart[1] break } diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index da29da00..9fa4a8fd 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -685,7 +685,7 @@ struct ImageManager { struct LoggerUtilities { func logRequiredMinimumOSVersion() { if !requiredMinimumOSVersionNil() { - Globals.nudgeDefaults.set(OSVersionRequirementVariables.requiredMinimumOSVersion, forKey: "requiredMinimumOSVersion") + Globals.nudgeDefaults.set(nudgePrimaryState.requiredMinimumOSVersion, forKey: "requiredMinimumOSVersion") } } @@ -1134,7 +1134,7 @@ struct UIUtilities { struct VersionManager { static func fullyUpdated() -> Bool { let currentOSVersion = GlobalVariables.currentOSVersion - let requiredMinimumOSVersion = OSVersionRequirementVariables.requiredMinimumOSVersion + let requiredMinimumOSVersion = nudgePrimaryState.requiredMinimumOSVersion let fullyUpdated = versionGreaterThanOrEqual(currentVersion: currentOSVersion, newVersion: requiredMinimumOSVersion) if fullyUpdated { LogManager.notice("Current operating system (\(currentOSVersion)) is greater than or equal to required operating system (\(requiredMinimumOSVersion))", logger: utilsLog) @@ -1150,7 +1150,7 @@ struct VersionManager { } static func getMajorRequiredNudgeOSVersion() -> Int { - guard let majorVersion = Int(OSVersionRequirementVariables.requiredMinimumOSVersion.split(separator: ".").first ?? "") else { + guard let majorVersion = Int(nudgePrimaryState.requiredMinimumOSVersion.split(separator: ".").first ?? "") else { LogManager.error("Invalid format for requiredMinimumOSVersion", logger: utilsLog) return 0 } @@ -1173,7 +1173,7 @@ struct VersionManager { } static func newNudgeEvent() -> Bool { - versionGreaterThan(currentVersion: OSVersionRequirementVariables.requiredMinimumOSVersion, newVersion: nudgePrimaryState.userRequiredMinimumOSVersion) + versionGreaterThan(currentVersion: nudgePrimaryState.requiredMinimumOSVersion, newVersion: nudgePrimaryState.userRequiredMinimumOSVersion) } // Adapted from https://stackoverflow.com/a/25453654 From c12fb995e15b6fe049d59a7e06f7d1ab4fbb4899 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 17 May 2024 14:50:40 -0500 Subject: [PATCH 032/141] add activelyExploitedInstallationSLA and standardInstallationSLA --- Nudge/Preferences/DefaultPreferencesNudge.swift | 14 +++++++++++++- Nudge/Preferences/PreferencesStructure.swift | 14 ++++++++++++-- Nudge/UI/Main.swift | 13 ++++++++----- 3 files changed, 33 insertions(+), 8 deletions(-) diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index dbbbf4ac..a5518eb5 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -169,7 +169,13 @@ struct OSVersionRequirementVariables { getAboutUpdateURL(OSVerReq: osVersionRequirementsJSON) ?? "" } - + + static var activelyExploitedInstallationSLA: Int { + osVersionRequirementsProfile?.activelyExploitedInstallationSLA ?? + osVersionRequirementsJSON?.activelyExploitedInstallationSLA ?? + 14 + } + static var majorUpgradeAppPath: String { osVersionRequirementsProfile?.majorUpgradeAppPath ?? osVersionRequirementsJSON?.majorUpgradeAppPath ?? @@ -184,6 +190,12 @@ struct OSVersionRequirementVariables { } } + static var standardInstallationSLA: Int { + osVersionRequirementsProfile?.standardInstallationSLA ?? + osVersionRequirementsJSON?.standardInstallationSLA ?? + 28 + } + static var unsupportedURL: String { getUnsupportedURL(OSVerReq: osVersionRequirementsProfile) ?? getUnsupportedURL(OSVerReq: osVersionRequirementsJSON) ?? diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index 4de187db..e9163eaf 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -117,9 +117,13 @@ extension OptionalFeatures { struct OSVersionRequirement: Codable { var aboutUpdateURL: String? var aboutUpdateURLs: [AboutUpdateURL]? - var actionButtonPath, majorUpgradeAppPath: String? + var actionButtonPath: String? + var activelyExploitedInstallationSLA: Int? + var majorUpgradeAppPath: String? var requiredInstallationDate: Date? - var requiredMinimumOSVersion, targetedOSVersionsRule: String? + var requiredMinimumOSVersion: String? + var standardInstallationSLA: Int? + var targetedOSVersionsRule: String? var unsupportedURL: String? var unsupportedURLs: [UnsupportedURL]? } @@ -129,8 +133,10 @@ extension OSVersionRequirement { init(fromDictionary: [String: AnyObject]) { self.aboutUpdateURL = fromDictionary["aboutUpdateURL"] as? String self.actionButtonPath = fromDictionary["actionButtonPath"] as? String + self.activelyExploitedInstallationSLA = fromDictionary["activelyExploitedInstallationSLA"] as? Int self.majorUpgradeAppPath = fromDictionary["majorUpgradeAppPath"] as? String self.requiredMinimumOSVersion = fromDictionary["requiredMinimumOSVersion"] as? String + self.standardInstallationSLA = fromDictionary["standardInstallationSLA"] as? Int self.targetedOSVersionsRule = fromDictionary["targetedOSVersionsRule"] as? String self.unsupportedURL = fromDictionary["unsupportedURL"] as? String @@ -186,9 +192,11 @@ extension OSVersionRequirement { aboutUpdateURL: String? = nil, aboutUpdateURLs: [AboutUpdateURL]? = nil, actionButtonPath: String? = nil, + activelyExploitedInstallationSLA: Int? = nil, majorUpgradeAppPath: String? = nil, requiredInstallationDate: Date? = nil, requiredMinimumOSVersion: String? = nil, + standardInstallationSLA: Int? = nil, targetedOSVersionsRule: String? = nil, unsupportedURL: String? = nil, unsupportedURLs: [UnsupportedURL]? = nil @@ -197,9 +205,11 @@ extension OSVersionRequirement { aboutUpdateURL: aboutUpdateURL ?? self.aboutUpdateURL, aboutUpdateURLs: aboutUpdateURLs ?? self.aboutUpdateURLs, actionButtonPath: actionButtonPath ?? self.actionButtonPath, + activelyExploitedInstallationSLA: activelyExploitedInstallationSLA ?? self.activelyExploitedInstallationSLA, majorUpgradeAppPath: majorUpgradeAppPath ?? self.majorUpgradeAppPath, requiredInstallationDate: requiredInstallationDate ?? self.requiredInstallationDate, requiredMinimumOSVersion: requiredMinimumOSVersion ?? self.requiredMinimumOSVersion, + standardInstallationSLA: standardInstallationSLA ?? self.standardInstallationSLA, targetedOSVersionsRule: targetedOSVersionsRule ?? self.targetedOSVersionsRule, unsupportedURL: unsupportedURL ?? self.unsupportedURL, unsupportedURLs: unsupportedURLs ?? self.unsupportedURLs diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index c71d8047..953123e1 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -175,10 +175,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { func applicationWillFinishLaunching(_ notification: Notification) { // print("applicationWillFinishLaunching") // TODO: Implement SOFA Caching with key refreshSOFAFeedTime - // TODO: Implement requiredInstallationDate values SLAs for Actively Exploited vs Not based on ReleaseDate sofa key - // TODO: activelyExploitedInstallationSLA, standardInstallationSLA // TODO: Implement Actively Exploited sidebar info on standardMode - // TODO: Implement "latest" for sofa vs hardcoded requiredInstallationVersion // TODO: Implement "latest-minor" or something for implementing all of the minor releases. // TODO: Add more logging to "unsupported devices" UI. // TODO: Add localization for "unsupported devices" text fields @@ -190,13 +187,19 @@ class AppDelegate: NSObject, NSApplicationDelegate { if PrefsWrapper.requiredMinimumOSVersion == "latest" { // Check if the specified device is in the supported devices of the matching asset nudgePrimaryState.requiredMinimumOSVersion = osVersion.latest.productVersion + let activelyExploited = osVersion.latest.activelyExploitedCVEs.count > 0 + LogManager.notice("SOFA Actively Exploited CVEs: \(activelyExploited)", logger: sofaLog) + let slaExtension = activelyExploited ? OSVersionRequirementVariables.activelyExploitedInstallationSLA * 86400 : OSVersionRequirementVariables.standardInstallationSLA * 86400 + requiredInstallationDate = osVersion.latest.releaseDate?.addingTimeInterval(TimeInterval(slaExtension)) ?? DateManager().getCurrentDate().addingTimeInterval(1209600) + LogManager.notice("Extending requiredInstallationDate to \(requiredInstallationDate)", logger: sofaLog) LogManager.notice("SOFA Matched OS Version: \(osVersion.latest.productVersion)", logger: sofaLog) LogManager.notice("SOFA Assets: \(osVersion.latest.supportedDevices)", logger: sofaLog) + LogManager.notice("SOFA CVEs: \(osVersion.latest.cves)", logger: sofaLog) + if OptionalFeatureVariables.attemptToCheckForSupportedDevice { LogManager.notice("Assessed Model ID: \(Globals.hardwareModelID)", logger: sofaLog) let deviceMatchFound = osVersion.latest.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }) - print("CVEs: \(osVersion.latest.cves)") - print("Actively Exploited CVEs: \(osVersion.latest.activelyExploitedCVEs.count > 0)") + LogManager.notice("Assessed Model ID found in SOFA Entry: \(deviceMatchFound)", logger: sofaLog) nudgePrimaryState.deviceSupportedByOSVersion = deviceMatchFound // nudgePrimaryState.deviceSupportedByOSVersion = false From 0764e29e67f1e31f42f6145a997f61e0e69e8656 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 17 May 2024 14:57:13 -0500 Subject: [PATCH 033/141] add actively exploited UI --- Nudge/UI/Defaults.swift | 1 + Nudge/UI/Main.swift | 7 +++---- Nudge/UI/StandardMode/LeftSide.swift | 3 +++ 3 files changed, 7 insertions(+), 4 deletions(-) diff --git a/Nudge/UI/Defaults.swift b/Nudge/UI/Defaults.swift index e92079c4..e4f87795 100644 --- a/Nudge/UI/Defaults.swift +++ b/Nudge/UI/Defaults.swift @@ -65,6 +65,7 @@ struct UIConstants { } class AppState: ObservableObject { + @Published var activelyExploitedCVEs = false @Published var afterFirstStateChange = false @Published var allowButtons = true @Published var daysRemaining = DateManager().getNumberOfDaysBetween() diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 953123e1..409dcc37 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -175,7 +175,6 @@ class AppDelegate: NSObject, NSApplicationDelegate { func applicationWillFinishLaunching(_ notification: Notification) { // print("applicationWillFinishLaunching") // TODO: Implement SOFA Caching with key refreshSOFAFeedTime - // TODO: Implement Actively Exploited sidebar info on standardMode // TODO: Implement "latest-minor" or something for implementing all of the minor releases. // TODO: Add more logging to "unsupported devices" UI. // TODO: Add localization for "unsupported devices" text fields @@ -187,9 +186,9 @@ class AppDelegate: NSObject, NSApplicationDelegate { if PrefsWrapper.requiredMinimumOSVersion == "latest" { // Check if the specified device is in the supported devices of the matching asset nudgePrimaryState.requiredMinimumOSVersion = osVersion.latest.productVersion - let activelyExploited = osVersion.latest.activelyExploitedCVEs.count > 0 - LogManager.notice("SOFA Actively Exploited CVEs: \(activelyExploited)", logger: sofaLog) - let slaExtension = activelyExploited ? OSVersionRequirementVariables.activelyExploitedInstallationSLA * 86400 : OSVersionRequirementVariables.standardInstallationSLA * 86400 + nudgePrimaryState.activelyExploitedCVEs = osVersion.latest.activelyExploitedCVEs.count > 0 + LogManager.notice("SOFA Actively Exploited CVEs: \(nudgePrimaryState.activelyExploitedCVEs)", logger: sofaLog) + let slaExtension = nudgePrimaryState.activelyExploitedCVEs ? OSVersionRequirementVariables.activelyExploitedInstallationSLA * 86400 : OSVersionRequirementVariables.standardInstallationSLA * 86400 requiredInstallationDate = osVersion.latest.releaseDate?.addingTimeInterval(TimeInterval(slaExtension)) ?? DateManager().getCurrentDate().addingTimeInterval(1209600) LogManager.notice("Extending requiredInstallationDate to \(requiredInstallationDate)", logger: sofaLog) LogManager.notice("SOFA Matched OS Version: \(osVersion.latest.productVersion)", logger: sofaLog) diff --git a/Nudge/UI/StandardMode/LeftSide.swift b/Nudge/UI/StandardMode/LeftSide.swift index 16ad91f7..ea9ea8ff 100644 --- a/Nudge/UI/StandardMode/LeftSide.swift +++ b/Nudge/UI/StandardMode/LeftSide.swift @@ -37,6 +37,9 @@ struct StandardModeLeftSide: View { private var informationStack: some View { VStack(alignment: .center, spacing: interLineSpacing) { InfoRow(label: "Required OS Version:", value: String(appState.requiredMinimumOSVersion), boldText: true) + if OptionalFeatureVariables.utilizeSOFAFeed { + InfoRow(label: "Actively Exploited CVEs:", value: String(appState.activelyExploitedCVEs).capitalized, boldText: appState.activelyExploitedCVEs) + } InfoRow(label: "Current OS Version:", value: GlobalVariables.currentOSVersion) remainingTimeRow if UserInterfaceVariables.showDeferralCount { From 90974f41bce40fece1ae945fcfc3987f5428833b Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 17 May 2024 14:58:16 -0500 Subject: [PATCH 034/141] more todo --- Nudge/UI/Main.swift | 1 + 1 file changed, 1 insertion(+) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 409dcc37..ea59de8b 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -175,6 +175,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { func applicationWillFinishLaunching(_ notification: Notification) { // print("applicationWillFinishLaunching") // TODO: Implement SOFA Caching with key refreshSOFAFeedTime + // TODO: Reset sofa after x amount of time // TODO: Implement "latest-minor" or something for implementing all of the minor releases. // TODO: Add more logging to "unsupported devices" UI. // TODO: Add localization for "unsupported devices" text fields From 51184f71e5770b9f4ddf9e950d0ed772d2549e1a Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 17 May 2024 16:07:49 -0500 Subject: [PATCH 035/141] markdown provided unsupported device --- .../com.github.macadmins.Nudge.tester.json | 14 +++++++------- Nudge/Preferences/DefaultPreferencesNudge.swift | 16 ++++++++-------- Nudge/UI/Common/InformationButton.swift | 4 ++-- Nudge/UI/Main.swift | 2 +- Nudge/UI/StandardMode/RightSide.swift | 15 ++++++--------- Nudge/Utilities/Preferences.swift | 6 +++--- 6 files changed, 27 insertions(+), 30 deletions(-) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index 9bcfed7e..7196d164 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -46,23 +46,23 @@ "customDeferralButtonText": "customDeferralButtonText", "customDeferralDropdownText": "customDeferralDropdownText", "informationButtonText": "informationButtonText", - "informationButtonTextUnsupported": "informationButtonTextUnsupported", + "informationButtonTextUnsupported1": "informationButtonTextUnsupported", "mainContentHeader": "mainContentHeader", - "mainContentHeaderUnsupported": "mainContentHeaderUnsupported", + "mainContentHeaderUnsupported1": "mainContentHeaderUnsupported", "mainContentNote": "mainContentNote", - "mainContentNoteUnsupported": "mainContentNoteUnsupported", + "mainContentNoteUnsupported1": "mainContentNoteUnsupported", "mainContentSubHeader": "mainContentSubHeader", - "mainContentSubHeaderUnsupported": "mainContentSubHeaderUnsupported", + "mainContentSubHeaderUnsupported1": "mainContentSubHeaderUnsupported", "mainContentText": "mainContentText", - "mainContentTextUnsupported": "mainContentTextUnsupported", + "mainContentTextUnsupported1": "mainContentTextUnsupported", "mainHeader": "mainHeader", - "mainHeaderUnsupported": "mainHeaderUnsupported", + "mainHeaderUnsupported1": "mainHeaderUnsupported", "oneDayDeferralButtonText": "oneDayDeferralButtonText", "oneHourDeferralButtonText": "oneHourDeferralButtonText", "primaryQuitButtonText": "primaryQuitButtonText", "secondaryQuitButtonText": "secondaryQuitButtonText", "subHeader": "subHeader", - "subHeaderUnsupported": "subHeaderUnsupported", + "subHeaderUnsupported1": "subHeaderUnsupported", "screenShotAltText": "Click to zoom into screenshot" } ] diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index a5518eb5..1f3181d1 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -404,25 +404,25 @@ struct UserInterfaceVariables { static var mainContentHeader: String { userInterfaceUpdateElementsProfile?["mainContentHeader"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentHeader ?? - "Your device will restart during this update" + "**Your device will restart during this update**" } static var mainContentHeaderUnsupported: String { userInterfaceUpdateElementsProfile?["mainContentHeaderUnsupported"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentHeaderUnsupported ?? - "Your device will restart during this update" + "**Your device is no longer capable of receving critical security updates**" } static var mainContentNote: String { userInterfaceUpdateElementsProfile?["mainContentNote"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentNote ?? - "Important Notes" + "**Important Notes**" } static var mainContentNoteUnsupported: String { userInterfaceUpdateElementsProfile?["mainContentNoteUnsupported"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentNoteUnsupported ?? - "Important Notes" + "**Important Notes**" } static var mainContentSubHeader: String { @@ -434,7 +434,7 @@ struct UserInterfaceVariables { static var mainContentSubHeaderUnsupported: String { userInterfaceUpdateElementsProfile?["mainContentSubHeaderUnsupported"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentSubHeaderUnsupported ?? - "Updates can take around 30 minutes to complete" + "Please work with your local IT team to obtain a replacement device" } static var mainContentText: String { @@ -446,7 +446,7 @@ struct UserInterfaceVariables { static var mainContentTextUnsupported: String { userInterfaceUpdateElementsProfile?["mainContentTextUnsupported"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentTextUnsupported ?? - "A fully up-to-date device is required to ensure that IT can accurately protect your device.\n\nIf you do not update your device, you may lose access to some items necessary for your day-to-day tasks.\n\nTo begin the update, simply click on the Update Device button and follow the provided steps." + "A fully up-to-date device is required to ensure that IT can accurately protect your device.\n\nIf you do not obtain a replacement device, you will lose access to some items necessary for your day-to-day tasks.\n\nFor more information about this, please click on the **Replace Your Device** button." } static var primaryQuitButtonText: String { @@ -476,13 +476,13 @@ struct UserInterfaceVariables { static var subHeader: String { userInterfaceUpdateElementsProfile?["subHeader"] as? String ?? userInterfaceUpdateElementsJSON?.subHeader ?? - "A friendly reminder from your local IT team" + "**A friendly reminder from your local IT team**" } static var subHeaderUnsupported: String { userInterfaceUpdateElementsProfile?["subHeaderUnsupported"] as? String ?? userInterfaceUpdateElementsJSON?.subHeaderUnsupported ?? - "A friendly reminder from your local IT team" + "**A friendly reminder from your local IT team**" } diff --git a/Nudge/UI/Common/InformationButton.swift b/Nudge/UI/Common/InformationButton.swift index 41c742c8..04e7763d 100644 --- a/Nudge/UI/Common/InformationButton.swift +++ b/Nudge/UI/Common/InformationButton.swift @@ -24,7 +24,7 @@ struct InformationButton: View { return AnyView( Button(action: UIUtilities().openMoreInfo) { - Text(UserInterfaceVariables.informationButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(.init(UserInterfaceVariables.informationButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)))) .foregroundColor(dynamicTextColor) } .buttonStyle(.plain) @@ -36,7 +36,7 @@ struct InformationButton: View { return AnyView( Button(action: UIUtilities().openMoreInfoUnsupported) { - Text(UserInterfaceVariables.informationButtonTextUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(.init(UserInterfaceVariables.informationButtonTextUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)))) .foregroundColor(dynamicTextColor) } .buttonStyle(.plain) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index ea59de8b..6d60fb1a 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -202,7 +202,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { LogManager.notice("Assessed Model ID found in SOFA Entry: \(deviceMatchFound)", logger: sofaLog) nudgePrimaryState.deviceSupportedByOSVersion = deviceMatchFound - // nudgePrimaryState.deviceSupportedByOSVersion = false + nudgePrimaryState.deviceSupportedByOSVersion = false } foundMatch = true } else { diff --git a/Nudge/UI/StandardMode/RightSide.swift b/Nudge/UI/StandardMode/RightSide.swift index cdba747d..5ff5966a 100644 --- a/Nudge/UI/StandardMode/RightSide.swift +++ b/Nudge/UI/StandardMode/RightSide.swift @@ -38,7 +38,7 @@ struct StandardModeRightSide: View { HStack { VStack(alignment: .leading, spacing: 5) { HStack { - Text(appState.deviceSupportedByOSVersion ? getMainHeader().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : getMainHeaderUnsupported().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(.init(appState.deviceSupportedByOSVersion ? getMainHeader().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : getMainHeaderUnsupported().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)))) .font(.largeTitle) .minimumScaleFactor(0.5) .frame(maxHeight: 25) @@ -46,9 +46,8 @@ struct StandardModeRightSide: View { } HStack { - Text(appState.deviceSupportedByOSVersion ? UserInterfaceVariables.subHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.subHeaderUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(.init(appState.deviceSupportedByOSVersion ? UserInterfaceVariables.subHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.subHeaderUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)))) .font(.body) - .fontWeight(.bold) .lineLimit(1) } } @@ -64,9 +63,8 @@ struct StandardModeRightSide: View { HStack(alignment: .center) { VStack(alignment: .leading, spacing: 1) { HStack { - Text(appState.deviceSupportedByOSVersion ? UserInterfaceVariables.mainContentHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentHeaderUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(.init(appState.deviceSupportedByOSVersion ? UserInterfaceVariables.mainContentHeader.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentHeaderUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)))) .font(.callout) - .fontWeight(.bold) Spacer() } HStack { @@ -81,7 +79,7 @@ struct StandardModeRightSide: View { Button(action: { UIUtilities().updateDevice() }) { - Text(UserInterfaceVariables.actionButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(.init(UserInterfaceVariables.actionButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)))) } .keyboardShortcut(.defaultAction) } @@ -90,9 +88,8 @@ struct StandardModeRightSide: View { Divider() HStack { - Text(appState.deviceSupportedByOSVersion ? UserInterfaceVariables.mainContentNote.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentNoteUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(.init(appState.deviceSupportedByOSVersion ? UserInterfaceVariables.mainContentNote.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentNoteUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)))) .font(.callout) - .fontWeight(.bold) .foregroundColor(appState.differentiateWithoutColor ? .accessibleRed : .red) Spacer() } @@ -100,7 +97,7 @@ struct StandardModeRightSide: View { ScrollView(.vertical) { VStack { HStack { - Text(appState.deviceSupportedByOSVersion ? UserInterfaceVariables.mainContentText.replacingOccurrences(of: "\\n", with: "\n").localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentTextUnsupported.replacingOccurrences(of: "\\n", with: "\n").localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + Text(.init(appState.deviceSupportedByOSVersion ? UserInterfaceVariables.mainContentText.replacingOccurrences(of: "\\n", with: "\n").localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : UserInterfaceVariables.mainContentTextUnsupported.replacingOccurrences(of: "\\n", with: "\n").localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)))) .font(.callout) .multilineTextAlignment(.leading) Spacer() diff --git a/Nudge/Utilities/Preferences.swift b/Nudge/Utilities/Preferences.swift index 9a037ea5..bc2707e4 100644 --- a/Nudge/Utilities/Preferences.swift +++ b/Nudge/Utilities/Preferences.swift @@ -262,12 +262,12 @@ func getMainHeader() -> String { func getMainHeaderUnsupported() -> String { if CommandLineUtilities().demoModeEnabled() { - return "Your device is no longer capable of security updates (Demo Mode)" + return "Your device requires a security update (Demo Mode)" } else if CommandLineUtilities().unitTestingEnabled() { - return "Your device is no longer capable of security updates (Unit Testing Mode)" + return "Your device requires a security update (Unit Testing Mode)" } return UserInterfaceVariables.userInterfaceUpdateElementsProfile?["mainHeaderUnsupported"] as? String ?? - getUserInterfaceUpdateElementsJSON()?.mainHeaderUnsupported ?? "Your device is no longer capable of security updates" + getUserInterfaceUpdateElementsJSON()?.mainHeaderUnsupported ?? "Your device requires a security update" } func simpleMode() -> Bool { From 0d54d7c057de9428680c4bdbfaa92059cb9edf82 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 17 May 2024 16:34:39 -0500 Subject: [PATCH 036/141] add overlay --- Nudge/UI/Common/CompanyLogo.swift | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/Nudge/UI/Common/CompanyLogo.swift b/Nudge/UI/Common/CompanyLogo.swift index 751744d2..f9000375 100644 --- a/Nudge/UI/Common/CompanyLogo.swift +++ b/Nudge/UI/Common/CompanyLogo.swift @@ -19,6 +19,7 @@ struct CompanyLogo: View { Group { if shouldShowCompanyLogo() { companyImage + .overlay(companyImageOverlay, alignment: .topTrailing) } else if UIUtilities().showEasterEgg() { easterEggView } else { @@ -49,6 +50,16 @@ struct CompanyLogo: View { } } + private var companyImageOverlay: some View { + guard !appState.deviceSupportedByOSVersion else { return AnyView(EmptyView()) } + return AnyView( + Image(systemName: "exclamationmark.triangle") + .symbolRenderingMode(.hierarchical) + .foregroundStyle(Color.red) + .font(.title) + ) + } + private var defaultImage: some View { Image(systemName: "applelogo") .customResizable(width: uiConstants.logoWidth, height: uiConstants.logoHeight) From a010370b189444095c95eb033affa0bcca50e16b Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 21 May 2024 09:27:44 -0500 Subject: [PATCH 037/141] move to xcode 15.4 --- build_nudge.zsh | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/build_nudge.zsh b/build_nudge.zsh index bb9949fa..26c6d83d 100755 --- a/build_nudge.zsh +++ b/build_nudge.zsh @@ -3,7 +3,7 @@ # Build script for Nudge # Variables -XCODE_PATH="/Applications/Xcode_15.2.app" +XCODE_PATH="/Applications/Xcode_15.4.app" APP_SIGNING_IDENTITY="Developer ID Application: Mac Admins Open Source (T4SK8ZXCXG)" INSTALLER_SIGNING_IDENTITY="Developer ID Installer: Mac Admins Open Source (T4SK8ZXCXG)" MP_SHA="71c57fcfdf43692adcd41fa7305be08f66bae3e5" From a3e984d42195b06bcb7a0d18f669f500724d7144 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 21 May 2024 12:52:21 -0500 Subject: [PATCH 038/141] add logic for sofa caching --- .../Preferences/DefaultPreferencesNudge.swift | 6 +++ Nudge/Preferences/PreferencesStructure.swift | 6 ++- Nudge/UI/Main.swift | 2 - Nudge/Utilities/Utils.swift | 43 ++++++++++++++++--- 4 files changed, 49 insertions(+), 8 deletions(-) diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 1f3181d1..97a320e9 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -143,6 +143,12 @@ struct OptionalFeatureVariables { false } + static var refreshSOFAFeedTime: Int { + optionalFeaturesProfile?["refreshSOFAFeedTime"] as? Int ?? + optionalFeaturesJSON?.refreshSOFAFeedTime ?? + 86400 + } + static var terminateApplicationsOnLaunch: Bool { optionalFeaturesProfile?["terminateApplicationsOnLaunch"] as? Bool ?? optionalFeaturesJSON?.terminateApplicationsOnLaunch ?? diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index e9163eaf..e9403baf 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -53,7 +53,9 @@ struct OptionalFeatures: Codable { var acceptableApplicationBundleIDs, acceptableAssertionApplicationNames: [String]? var acceptableAssertionUsage, acceptableCameraUsage, acceptableScreenSharingUsage, aggressiveUserExperience, aggressiveUserFullScreenExperience, asynchronousSoftwareUpdate, attemptToBlockApplicationLaunches, attemptToCheckForSupportedDevice, attemptToFetchMajorUpgrade: Bool? var blockedApplicationBundleIDs: [String]? - var disableSoftwareUpdateWorkflow, enforceMinorUpdates, terminateApplicationsOnLaunch, utilizeSOFAFeed: Bool? + var disableSoftwareUpdateWorkflow, enforceMinorUpdates: Bool? + var refreshSOFAFeedTime: Int? + var terminateApplicationsOnLaunch, utilizeSOFAFeed: Bool? } // MARK: OptionalFeatures convenience initializers and mutators @@ -89,6 +91,7 @@ extension OptionalFeatures { blockedApplicationBundleIDs: [String]? = nil, disableSoftwareUpdateWorkflow: Bool? = nil, enforceMinorUpdates: Bool? = nil, + refreshSOFAFeedTime: Int? = nil, terminateApplicationsOnLaunch: Bool? = nil, utilizeSOFAFeed: Bool? = nil ) -> OptionalFeatures { @@ -107,6 +110,7 @@ extension OptionalFeatures { blockedApplicationBundleIDs: blockedApplicationBundleIDs ?? self.blockedApplicationBundleIDs, disableSoftwareUpdateWorkflow: disableSoftwareUpdateWorkflow ?? self.disableSoftwareUpdateWorkflow, enforceMinorUpdates: enforceMinorUpdates ?? self.enforceMinorUpdates, + refreshSOFAFeedTime: refreshSOFAFeedTime ?? self.refreshSOFAFeedTime, terminateApplicationsOnLaunch: terminateApplicationsOnLaunch ?? self.terminateApplicationsOnLaunch, utilizeSOFAFeed: utilizeSOFAFeed ?? self.utilizeSOFAFeed ) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 6d60fb1a..ae09ce48 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -174,8 +174,6 @@ class AppDelegate: NSObject, NSApplicationDelegate { // Pre-Launch Logic func applicationWillFinishLaunching(_ notification: Notification) { // print("applicationWillFinishLaunching") - // TODO: Implement SOFA Caching with key refreshSOFAFeedTime - // TODO: Reset sofa after x amount of time // TODO: Implement "latest-minor" or something for implementing all of the minor releases. // TODO: Add more logging to "unsupported devices" UI. // TODO: Add localization for "unsupported devices" text fields diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 9fa4a8fd..b1c07f7f 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -108,7 +108,7 @@ struct AppStateManager { exit(0) } - private func getCreationDateForPath(_ path: String, testFileDate: Date?) -> Date? { + func getCreationDateForPath(_ path: String, testFileDate: Date?) -> Date? { let attributes = try? FileManager.default.attributesOfItem(atPath: path) let creationDate = attributes?[.creationDate] as? Date return testFileDate ?? creationDate @@ -828,21 +828,54 @@ struct NetworkFileManager { if !OptionalFeatureVariables.utilizeSOFAFeed { return nil } + let fileManager = FileManager.default + let appSupportDirectory = fileManager.urls(for: .applicationSupportDirectory, in: .userDomainMask).first! + let appDirectory = appSupportDirectory.appendingPathComponent(Bundle.main.bundleIdentifier ?? "com.github.macadmins.Nudge") + let sofaFile = "sofa-macos_data_feed.json" + let sofaPath = appDirectory.appendingPathComponent(sofaFile) + if fileManager.fileExists(atPath: sofaPath.path) { + let sofaPathCreationDate = AppStateManager().getCreationDateForPath(sofaPath.path, testFileDate: nil) + // Use Cache as it is within time inverval + if TimeInterval(OptionalFeatureVariables.refreshSOFAFeedTime) >= Date().timeIntervalSince(sofaPathCreationDate!) { + LogManager.info("Utilizing previously cached SOFA json", logger: sofaLog) + do { + let sofaData = try Data(contentsOf: sofaPath) + let assetInfo = try MacOSDataFeed(data: sofaData) + return assetInfo + } catch { + LogManager.error("Failed to decode local sofa JSON: \(error.localizedDescription)", logger: sofaLog) + LogManager.error("Failed to decode sofa JSON: \(error)", logger: sofaLog) + } + } + } else { + // Ensure the Application Support directory exists + if !fileManager.fileExists(atPath: appDirectory.path) { + do { + try fileManager.createDirectory(at: appDirectory, withIntermediateDirectories: true, attributes: nil) + } catch { + LogManager.error("Failed to create Nudge's Application Support directory: \(error.localizedDescription)", logger: utilsLog) + } + } + } + if let url = URL(string: "https://sofa.macadmins.io/v1/macos_data_feed.json") { let sofaData = SOFA().URLSync(url: url) if (sofaData.error == nil) { do { + if fileManager.fileExists(atPath: appDirectory.path) { + try sofaData.data!.write(to: sofaPath) + } let assetInfo = try MacOSDataFeed(data: sofaData.data!) return assetInfo } catch { - LogManager.error("Failed to decode sofa JSON: \(error.localizedDescription)", logger: utilsLog) - LogManager.error("Failed to decode sofa JSON: \(error)", logger: utilsLog) + LogManager.error("Failed to decode sofa JSON: \(error.localizedDescription)", logger: sofaLog) + LogManager.error("Failed to decode sofa JSON: \(error)", logger: sofaLog) } } else { - LogManager.error("Failed to fetch sofa JSON: \(sofaData.error!.localizedDescription)", logger: utilsLog) + LogManager.error("Failed to fetch sofa JSON: \(sofaData.error!.localizedDescription)", logger: sofaLog) } } else { - LogManager.error("Failed to decode sofa JSON URL string", logger: utilsLog) + LogManager.error("Failed to decode sofa JSON URL string", logger: sofaLog) } return nil } From 93757a897cc658d1a964cf0baa3a3335496ba844 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 21 May 2024 16:03:03 -0500 Subject: [PATCH 039/141] 2.0.0 release and more notes --- CHANGELOG.md | 24 +++++++++++++++++++++--- Nudge.xcodeproj/project.pbxproj | 4 ++-- Nudge/Info.plist | 4 ++-- 3 files changed, 25 insertions(+), 7 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index d8781556..8876904f 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,16 +4,34 @@ All notable changes to this project will be documented in this file. The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). -## [1.2.0] - 2024-05-08 +## [2.0.0] - 2024-05-21 Requires macOS 12.0 and higher. Further releases and feature requests may make this macOS 13 and higher depending on code complexity. ### Changed -Upcoming +- You can now pass the string `latest` in the `requiredMinimumOSVersion` key + - This requires utilizing the SOFA feed features to properly work + - Nudge will then utilize two date integers to automatically calculate the `requiredInstallationDate` + - `activelyExploitedInstallationSLA` under the `osVersionRequirement` key will default to 14 days + - `standardInstallationSLA` under the `osVersionRequirement` key will default to 28 days + - These dates are calculated against the `ReleaseDate` key in the SOFA feed ### Fixed Upcoming + ### Added -Upcoming +- An admin can now allow users to move the Nudge window with `userExperience` key `allowMovableWindow` +- Basic SwiftUI support for Markdown text options + - Utilizing Apple's markdown features, you can now utilize, bold, italic, underline, subscript and url links directly into any of the text fields +- [SOFA](https://github.com/macadmins/sofa) feed support + - Set the `utilizeSOFAFeed` key `true` under `optionalFeatures` to enable this feature + - Nudge will by default check the feed every 24 hours. + - In order to change this, please configure the `refreshSOFAFeedTime` key under `optionalFeatures` in seconds +- "Unsupported" device UI in standard mode that utilizes the SOFA feed + - Set the `attemptToCheckForSupportedDevice` key `true` under `optionalFeatures` to enable this feature + - There are now keys to set all of text fields + - `informationButtonTextUnsupported`, `mainContentHeaderUnsupported`, `mainContentNoteUnsupported`, `mainContentSubHeaderUnsupported`, `mainContentTextUnsupported`, `subHeaderUnsupported` + - `unsupportedURL` and `unsupportedURLs` can change the information button itself, but it will remain in the `osVersionRequirement` key with `unsupportedURLs` and `unsupportedURLs`. + - An icon will appear as an overlay on top of the company image to further emphasize the device is no longer supported ## [1.1.16] - 2024-03-13 This will be the **final Nudge release** for macOS 11 and potentially other versions of macOS. diff --git a/Nudge.xcodeproj/project.pbxproj b/Nudge.xcodeproj/project.pbxproj index 9f563e03..783f770f 100644 --- a/Nudge.xcodeproj/project.pbxproj +++ b/Nudge.xcodeproj/project.pbxproj @@ -690,7 +690,7 @@ "@executable_path/../Frameworks", ); MACOSX_DEPLOYMENT_TARGET = 12.0; - MARKETING_VERSION = 1.2.0; + MARKETING_VERSION = 2.0.0; PRODUCT_BUNDLE_IDENTIFIER = com.github.macadmins.Nudge; PRODUCT_NAME = "$(TARGET_NAME)"; PROVISIONING_PROFILE_SPECIFIER = ""; @@ -721,7 +721,7 @@ "@executable_path/../Frameworks", ); MACOSX_DEPLOYMENT_TARGET = 12.0; - MARKETING_VERSION = 1.2.0; + MARKETING_VERSION = 2.0.0; PRODUCT_BUNDLE_IDENTIFIER = com.github.macadmins.Nudge; PRODUCT_NAME = "$(TARGET_NAME)"; PROVISIONING_PROFILE_SPECIFIER = ""; diff --git a/Nudge/Info.plist b/Nudge/Info.plist index 90c42a07..9eee977e 100644 --- a/Nudge/Info.plist +++ b/Nudge/Info.plist @@ -15,9 +15,9 @@ CFBundlePackageType $(PRODUCT_BUNDLE_PACKAGE_TYPE) CFBundleShortVersionString - 1.2.0 + 2.0.0 CFBundleVersion - 1.2.0 + 2.0.0 LSApplicationCategoryType public.app-category.utilities LSMinimumSystemVersion From 67936c404c643039e102bc77650d6715a8ce4d15 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 21 May 2024 16:07:59 -0500 Subject: [PATCH 040/141] macos13 to macos14 for new xcode --- .github/workflows/build_nudge_pr.yml | 2 +- .github/workflows/build_nudge_prerelease.yml | 2 +- .github/workflows/build_nudge_prerelease_manual.yml | 2 +- .github/workflows/build_nudge_release.yml | 2 +- .github/workflows/build_nudge_release_manual.yml | 2 +- 5 files changed, 5 insertions(+), 5 deletions(-) diff --git a/.github/workflows/build_nudge_pr.yml b/.github/workflows/build_nudge_pr.yml index f297cd0b..7954ea00 100644 --- a/.github/workflows/build_nudge_pr.yml +++ b/.github/workflows/build_nudge_pr.yml @@ -6,7 +6,7 @@ on: jobs: build: - runs-on: macos-13 + runs-on: macos-14-large if: contains(github.event.pull_request.labels.*.name, 'safe to test') steps: diff --git a/.github/workflows/build_nudge_prerelease.yml b/.github/workflows/build_nudge_prerelease.yml index 6a43d70c..a432b351 100644 --- a/.github/workflows/build_nudge_prerelease.yml +++ b/.github/workflows/build_nudge_prerelease.yml @@ -15,7 +15,7 @@ on: jobs: build: - runs-on: macos-13 + runs-on: macos-14-large steps: - name: Checkout nudge repo diff --git a/.github/workflows/build_nudge_prerelease_manual.yml b/.github/workflows/build_nudge_prerelease_manual.yml index ed3b64cb..3fb96e25 100644 --- a/.github/workflows/build_nudge_prerelease_manual.yml +++ b/.github/workflows/build_nudge_prerelease_manual.yml @@ -7,7 +7,7 @@ on: [workflow_dispatch] jobs: build: - runs-on: macos-13 + runs-on: macos-14-large steps: - name: Checkout nudge repo diff --git a/.github/workflows/build_nudge_release.yml b/.github/workflows/build_nudge_release.yml index 4a045ba7..c9a158b3 100644 --- a/.github/workflows/build_nudge_release.yml +++ b/.github/workflows/build_nudge_release.yml @@ -15,7 +15,7 @@ on: jobs: build: - runs-on: macos-13 + runs-on: macos-14-large steps: - name: Checkout nudge repo diff --git a/.github/workflows/build_nudge_release_manual.yml b/.github/workflows/build_nudge_release_manual.yml index 91bbddc2..a4d31400 100644 --- a/.github/workflows/build_nudge_release_manual.yml +++ b/.github/workflows/build_nudge_release_manual.yml @@ -7,7 +7,7 @@ on: [workflow_dispatch] jobs: build: - runs-on: macos-13 + runs-on: macos-14-large steps: - name: Checkout nudge repo From 0c6535f7b7bc76d21895cb85585469dafb7d5c56 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 21 May 2024 16:12:16 -0500 Subject: [PATCH 041/141] dont try -large --- .github/workflows/build_nudge_pr.yml | 2 +- .github/workflows/build_nudge_prerelease.yml | 2 +- .github/workflows/build_nudge_prerelease_manual.yml | 2 +- .github/workflows/build_nudge_release.yml | 2 +- .github/workflows/build_nudge_release_manual.yml | 2 +- 5 files changed, 5 insertions(+), 5 deletions(-) diff --git a/.github/workflows/build_nudge_pr.yml b/.github/workflows/build_nudge_pr.yml index 7954ea00..ff366fd1 100644 --- a/.github/workflows/build_nudge_pr.yml +++ b/.github/workflows/build_nudge_pr.yml @@ -6,7 +6,7 @@ on: jobs: build: - runs-on: macos-14-large + runs-on: macos-14 if: contains(github.event.pull_request.labels.*.name, 'safe to test') steps: diff --git a/.github/workflows/build_nudge_prerelease.yml b/.github/workflows/build_nudge_prerelease.yml index a432b351..56983b78 100644 --- a/.github/workflows/build_nudge_prerelease.yml +++ b/.github/workflows/build_nudge_prerelease.yml @@ -15,7 +15,7 @@ on: jobs: build: - runs-on: macos-14-large + runs-on: macos-14 steps: - name: Checkout nudge repo diff --git a/.github/workflows/build_nudge_prerelease_manual.yml b/.github/workflows/build_nudge_prerelease_manual.yml index 3fb96e25..141937db 100644 --- a/.github/workflows/build_nudge_prerelease_manual.yml +++ b/.github/workflows/build_nudge_prerelease_manual.yml @@ -7,7 +7,7 @@ on: [workflow_dispatch] jobs: build: - runs-on: macos-14-large + runs-on: macos-14 steps: - name: Checkout nudge repo diff --git a/.github/workflows/build_nudge_release.yml b/.github/workflows/build_nudge_release.yml index c9a158b3..d5ccbece 100644 --- a/.github/workflows/build_nudge_release.yml +++ b/.github/workflows/build_nudge_release.yml @@ -15,7 +15,7 @@ on: jobs: build: - runs-on: macos-14-large + runs-on: macos-14 steps: - name: Checkout nudge repo diff --git a/.github/workflows/build_nudge_release_manual.yml b/.github/workflows/build_nudge_release_manual.yml index a4d31400..97551cb5 100644 --- a/.github/workflows/build_nudge_release_manual.yml +++ b/.github/workflows/build_nudge_release_manual.yml @@ -7,7 +7,7 @@ on: [workflow_dispatch] jobs: build: - runs-on: macos-14-large + runs-on: macos-14 steps: - name: Checkout nudge repo From 5f4b89931d5f84a496b3f8ce94eabfc4c20307ae Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 21 May 2024 17:00:24 -0500 Subject: [PATCH 042/141] fix changelog --- CHANGELOG.md | 28 ++++++++++++++-------------- 1 file changed, 14 insertions(+), 14 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 8876904f..466ed191 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -9,11 +9,11 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t ### Changed - You can now pass the string `latest` in the `requiredMinimumOSVersion` key - - This requires utilizing the SOFA feed features to properly work - - Nudge will then utilize two date integers to automatically calculate the `requiredInstallationDate` - - `activelyExploitedInstallationSLA` under the `osVersionRequirement` key will default to 14 days - - `standardInstallationSLA` under the `osVersionRequirement` key will default to 28 days - - These dates are calculated against the `ReleaseDate` key in the SOFA feed + - This requires utilizing the SOFA feed features to properly work + - Nudge will then utilize two date integers to automatically calculate the `requiredInstallationDate` + - `activelyExploitedInstallationSLA` under the `osVersionRequirement` key will default to 14 days + - `standardInstallationSLA` under the `osVersionRequirement` key will default to 28 days + - These dates are calculated against the `ReleaseDate` key in the SOFA feed ### Fixed Upcoming @@ -21,17 +21,17 @@ Upcoming ### Added - An admin can now allow users to move the Nudge window with `userExperience` key `allowMovableWindow` - Basic SwiftUI support for Markdown text options - - Utilizing Apple's markdown features, you can now utilize, bold, italic, underline, subscript and url links directly into any of the text fields + - Utilizing Apple's markdown features, you can now utilize, bold, italic, underline, subscript and url links directly into any of the text fields - [SOFA](https://github.com/macadmins/sofa) feed support - - Set the `utilizeSOFAFeed` key `true` under `optionalFeatures` to enable this feature - - Nudge will by default check the feed every 24 hours. - - In order to change this, please configure the `refreshSOFAFeedTime` key under `optionalFeatures` in seconds + - Set the `utilizeSOFAFeed` key `true` under `optionalFeatures` to enable this feature + - Nudge will by default check the feed every 24 hours. + - In order to change this, please configure the `refreshSOFAFeedTime` key under `optionalFeatures` in seconds - "Unsupported" device UI in standard mode that utilizes the SOFA feed - - Set the `attemptToCheckForSupportedDevice` key `true` under `optionalFeatures` to enable this feature - - There are now keys to set all of text fields - - `informationButtonTextUnsupported`, `mainContentHeaderUnsupported`, `mainContentNoteUnsupported`, `mainContentSubHeaderUnsupported`, `mainContentTextUnsupported`, `subHeaderUnsupported` - - `unsupportedURL` and `unsupportedURLs` can change the information button itself, but it will remain in the `osVersionRequirement` key with `unsupportedURLs` and `unsupportedURLs`. - - An icon will appear as an overlay on top of the company image to further emphasize the device is no longer supported + - Set the `attemptToCheckForSupportedDevice` key `true` under `optionalFeatures` to enable this feature + - There are now keys to set all of text fields + - `informationButtonTextUnsupported`, `mainContentHeaderUnsupported`, `mainContentNoteUnsupported`, `mainContentSubHeaderUnsupported`, `mainContentTextUnsupported`, `subHeaderUnsupported` + - `unsupportedURL` and `unsupportedURLs` can change the information button itself, but it will remain in the `osVersionRequirement` key with `unsupportedURLs` and `unsupportedURLs`. + - An icon will appear as an overlay on top of the company image to further emphasize the device is no longer supported ## [1.1.16] - 2024-03-13 This will be the **final Nudge release** for macOS 11 and potentially other versions of macOS. From bc7ec7e0d6ae47b676e67ddcbc7ad3470bb6dbe0 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 21 May 2024 17:06:18 -0500 Subject: [PATCH 043/141] remove forced unsupported --- Nudge/UI/Main.swift | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index ae09ce48..af6a067d 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -178,6 +178,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { // TODO: Add more logging to "unsupported devices" UI. // TODO: Add localization for "unsupported devices" text fields // TODO: Get someone to update JAMF JSON schema for all the new keys and wiki + // TODO: add support for custom sofa feed url if OptionalFeatureVariables.utilizeSOFAFeed { if let macOSSOFAAssets = Globals.sofaAssets?.osVersions { var foundMatch = false @@ -185,6 +186,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { if PrefsWrapper.requiredMinimumOSVersion == "latest" { // Check if the specified device is in the supported devices of the matching asset nudgePrimaryState.requiredMinimumOSVersion = osVersion.latest.productVersion + // nudgePrimaryState.requiredMinimumOSVersion = "14.6" // TODO: remove when testing is done nudgePrimaryState.activelyExploitedCVEs = osVersion.latest.activelyExploitedCVEs.count > 0 LogManager.notice("SOFA Actively Exploited CVEs: \(nudgePrimaryState.activelyExploitedCVEs)", logger: sofaLog) let slaExtension = nudgePrimaryState.activelyExploitedCVEs ? OSVersionRequirementVariables.activelyExploitedInstallationSLA * 86400 : OSVersionRequirementVariables.standardInstallationSLA * 86400 @@ -200,7 +202,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { LogManager.notice("Assessed Model ID found in SOFA Entry: \(deviceMatchFound)", logger: sofaLog) nudgePrimaryState.deviceSupportedByOSVersion = deviceMatchFound - nudgePrimaryState.deviceSupportedByOSVersion = false + // nudgePrimaryState.deviceSupportedByOSVersion = false // TODO: remove when testing is done } foundMatch = true } else { From c5c1dd4e5c0924c9e7463e65419c33ad4bc5b855 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 21 May 2024 17:14:37 -0500 Subject: [PATCH 044/141] add another todo --- Nudge/UI/Main.swift | 1 + 1 file changed, 1 insertion(+) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index af6a067d..7c4a30a6 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -179,6 +179,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { // TODO: Add localization for "unsupported devices" text fields // TODO: Get someone to update JAMF JSON schema for all the new keys and wiki // TODO: add support for custom sofa feed url + // TODO: check the sofa json etag even after the timelimit age to reduce more bandwidth if OptionalFeatureVariables.utilizeSOFAFeed { if let macOSSOFAAssets = Globals.sofaAssets?.osVersions { var foundMatch = false From 45928ced8a4b31baf4091007624023858fe7b124 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 22 May 2024 08:37:01 -0500 Subject: [PATCH 045/141] remove unneccessary key for testing latest --- Example Assets/com.github.macadmins.Nudge.tester.json | 1 - 1 file changed, 1 deletion(-) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index 7196d164..970ba928 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -21,7 +21,6 @@ "osVersionRequirements": [ { "aboutUpdateURL": "https://apple.com", - "requiredInstallationDate": "2025-01-01T00:00:00Z", "requiredMinimumOSVersion": "latest", "unsupportedURL": "https://google.com", } From 94685906a485db4be28c7892e580bea7c5d92a35 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 22 May 2024 09:02:28 -0500 Subject: [PATCH 046/141] add support for "latest-supported" --- Example Assets/com.github.macadmins.Nudge.tester.json | 2 +- Nudge/Preferences/DefaultPreferencesNudge.swift | 2 +- Nudge/UI/Main.swift | 7 ++++++- 3 files changed, 8 insertions(+), 3 deletions(-) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index 970ba928..f56eeec5 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -21,7 +21,7 @@ "osVersionRequirements": [ { "aboutUpdateURL": "https://apple.com", - "requiredMinimumOSVersion": "latest", + "requiredMinimumOSVersion": "latest-supported", "unsupportedURL": "https://google.com", } ], diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 97a320e9..b3552a13 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -189,7 +189,7 @@ struct OSVersionRequirementVariables { } static var requiredMinimumOSVersion: String { - if PrefsWrapper.requiredMinimumOSVersion == "latest" { + if ["latest", "latest-supported", "latest-major"].contains(PrefsWrapper.requiredMinimumOSVersion) { PrefsWrapper.requiredMinimumOSVersion } else { try! OSVersion(PrefsWrapper.requiredMinimumOSVersion).description diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 7c4a30a6..afbb99fa 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -184,7 +184,12 @@ class AppDelegate: NSObject, NSApplicationDelegate { if let macOSSOFAAssets = Globals.sofaAssets?.osVersions { var foundMatch = false for osVersion in macOSSOFAAssets { - if PrefsWrapper.requiredMinimumOSVersion == "latest" { + if ["latest", "latest-supported", "latest-major"].contains(PrefsWrapper.requiredMinimumOSVersion) { + if PrefsWrapper.requiredMinimumOSVersion == "latest-supported" { + if VersionManager.getMajorOSVersion() != Int(osVersion.osVersion.split(separator: " ").last!) { + continue + } + } // Check if the specified device is in the supported devices of the matching asset nudgePrimaryState.requiredMinimumOSVersion = osVersion.latest.productVersion // nudgePrimaryState.requiredMinimumOSVersion = "14.6" // TODO: remove when testing is done From 3b364189d5f54f1f579db5af90e09342878a3de7 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 22 May 2024 09:29:03 -0500 Subject: [PATCH 047/141] add logic to test bundle profile --- .../com.github.macadmins.Nudge.tester.plist | 123 ++++++++++++++++++ Nudge.xcodeproj/project.pbxproj | 4 + ...udge - Debug (-bundle-mode-json).xcscheme} | 2 +- ...ge - Debug (-bundle-mode-profile).xcscheme | 104 +++++++++++++++ .../Nudge - Debug (-demo-mode).xcscheme | 2 +- .../xcschemes/xcschememanagement.plist | 13 +- Nudge/Utilities/Utils.swift | 22 +++- 7 files changed, 260 insertions(+), 10 deletions(-) create mode 100644 Example Assets/com.github.macadmins.Nudge.tester.plist rename Nudge.xcodeproj/xcshareddata/xcschemes/{Nudge - Debug (-bundle-mode).xcscheme => Nudge - Debug (-bundle-mode-json).xcscheme} (98%) create mode 100644 Nudge.xcodeproj/xcshareddata/xcschemes/Nudge - Debug (-bundle-mode-profile).xcscheme diff --git a/Example Assets/com.github.macadmins.Nudge.tester.plist b/Example Assets/com.github.macadmins.Nudge.tester.plist new file mode 100644 index 00000000..a58bd10d --- /dev/null +++ b/Example Assets/com.github.macadmins.Nudge.tester.plist @@ -0,0 +1,123 @@ + + + + + optionalFeatures + + acceptableAssertionApplicationNames + + zoom.us + Meeting Center + Google Chrome + Safari + + acceptableAssertionUsage + + acceptableCameraUsage + + acceptableScreenSharingUsage + + attemptToBlockApplicationLaunches + + attemptToCheckForSupportedDevice + + blockedApplicationBundleIDs + + com.microsoft.VSCode + us.zoom.xos + + terminateApplicationsOnLaunch + + utilizeSOFAFeed + + + osVersionRequirements + + + aboutUpdateURL + https://apple.com + requiredMinimumOSVersion + latest-supported + unsupportedURL + https://google.com + + + userExperience + + elapsedRefreshCycle + 10 + initialRefreshCycle + 10 + loadLaunchAgent + + nudgeRefreshCycle + 5 + randomDelay + + + userInterface + + iconDarkPath + https://github.com/macadmins/nudge/blob/main/assets/NudgeIconInverted.png?raw=true + iconLightPath + https://github.com/macadmins/nudge/blob/main/assets/NudgeIcon.png?raw=true + screenShotDarkPath + https://github.com/macadmins/nudge/blob/main/assets/standard_mode/demo_dark_1_icon.png?raw=true + screenShotLightPath + https://github.com/macadmins/nudge/blob/main/assets/standard_mode/demo_light_1_icon.png?raw=true + simpleMode + + updateElements + + + _language + en + actionButtonText + actionButtonText + customDeferralButtonText + customDeferralButtonText + customDeferralDropdownText + customDeferralDropdownText + informationButtonText + informationButtonText + informationButtonTextUnsupported1 + informationButtonTextUnsupported + mainContentHeader + mainContentHeader + mainContentHeaderUnsupported1 + mainContentHeaderUnsupported + mainContentNote + mainContentNote + mainContentNoteUnsupported1 + mainContentNoteUnsupported + mainContentSubHeader + mainContentSubHeader + mainContentSubHeaderUnsupported1 + mainContentSubHeaderUnsupported + mainContentText + mainContentText + mainContentTextUnsupported1 + mainContentTextUnsupported + mainHeader + mainHeader + mainHeaderUnsupported1 + mainHeaderUnsupported + oneDayDeferralButtonText + oneDayDeferralButtonText + oneHourDeferralButtonText + oneHourDeferralButtonText + primaryQuitButtonText + primaryQuitButtonText + screenShotAltText + Click to zoom into screenshot + secondaryQuitButtonText + secondaryQuitButtonText + subHeader + subHeader + subHeaderUnsupported1 + subHeaderUnsupported + + + + + diff --git a/Nudge.xcodeproj/project.pbxproj b/Nudge.xcodeproj/project.pbxproj index 783f770f..c51b9710 100644 --- a/Nudge.xcodeproj/project.pbxproj +++ b/Nudge.xcodeproj/project.pbxproj @@ -27,6 +27,7 @@ 636C4B4A25D1BECE0004A791 /* DefaultPreferencesNudge.swift in Sources */ = {isa = PBXBuildFile; fileRef = 636C4B4925D1BECE0004A791 /* DefaultPreferencesNudge.swift */; }; 636C4B7625D4306A0004A791 /* UILogic.swift in Sources */ = {isa = PBXBuildFile; fileRef = 636C4B7525D4306A0004A791 /* UILogic.swift */; }; 637CEBC12A30C9E700EFA3E9 /* Localizable.xcstrings in Resources */ = {isa = PBXBuildFile; fileRef = 637CEBC02A30C9E700EFA3E9 /* Localizable.xcstrings */; }; + 6388B6972BFE363B0094F26B /* com.github.macadmins.Nudge.tester.plist in Resources */ = {isa = PBXBuildFile; fileRef = 6388B6962BFE363B0094F26B /* com.github.macadmins.Nudge.tester.plist */; }; 639B6B0F25DC9ED300E38EC1 /* com.github.macadmins.Nudge.mobileconfig in Resources */ = {isa = PBXBuildFile; fileRef = 639B6B0E25DC9ED300E38EC1 /* com.github.macadmins.Nudge.mobileconfig */; }; 639B6B3B25DF200C00E38EC1 /* Preferences.swift in Sources */ = {isa = PBXBuildFile; fileRef = 639B6B3A25DF200C00E38EC1 /* Preferences.swift */; }; 639B6B5825DF377B00E38EC1 /* DeviceInfo.swift in Sources */ = {isa = PBXBuildFile; fileRef = 639B6B5725DF377B00E38EC1 /* DeviceInfo.swift */; }; @@ -101,6 +102,7 @@ 636C4B4925D1BECE0004A791 /* DefaultPreferencesNudge.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DefaultPreferencesNudge.swift; sourceTree = ""; }; 636C4B7525D4306A0004A791 /* UILogic.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UILogic.swift; sourceTree = ""; }; 637CEBC02A30C9E700EFA3E9 /* Localizable.xcstrings */ = {isa = PBXFileReference; lastKnownFileType = text.json.xcstrings; path = Localizable.xcstrings; sourceTree = ""; }; + 6388B6962BFE363B0094F26B /* com.github.macadmins.Nudge.tester.plist */ = {isa = PBXFileReference; lastKnownFileType = text.plist.xml; path = com.github.macadmins.Nudge.tester.plist; sourceTree = ""; }; 6397293C26CDBA4C00BDAF42 /* Nudge-Debug.entitlements */ = {isa = PBXFileReference; lastKnownFileType = text.plist.entitlements; path = "Nudge-Debug.entitlements"; sourceTree = ""; }; 639B6B0E25DC9ED300E38EC1 /* com.github.macadmins.Nudge.mobileconfig */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = text.xml; path = com.github.macadmins.Nudge.mobileconfig; sourceTree = ""; }; 639B6B3A25DF200C00E38EC1 /* Preferences.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = Preferences.swift; sourceTree = ""; }; @@ -164,6 +166,7 @@ 639B6B0E25DC9ED300E38EC1 /* com.github.macadmins.Nudge.mobileconfig */, 035C2AEB25D8ABC400429458 /* com.github.macadmins.Nudge.json */, 63C6A08D2833FB6500D5264A /* com.github.macadmins.Nudge.tester.json */, + 6388B6962BFE363B0094F26B /* com.github.macadmins.Nudge.tester.plist */, ); path = "Example Assets"; sourceTree = ""; @@ -451,6 +454,7 @@ 63D7D0E725C9E9A500236281 /* Assets.xcassets in Resources */, 639B6B0F25DC9ED300E38EC1 /* com.github.macadmins.Nudge.mobileconfig in Resources */, 73CC1D7829B81EE500FBF8E2 /* com.github.macadmins.Nudge.SMAppService.plist in Resources */, + 6388B6972BFE363B0094F26B /* com.github.macadmins.Nudge.tester.plist in Resources */, 63C6A08E2833FB6500D5264A /* com.github.macadmins.Nudge.tester.json in Resources */, 035C2AEC25D8ABC400429458 /* com.github.macadmins.Nudge.json in Resources */, 6316F0E72832CA0700E1354D /* Schema in Resources */, diff --git a/Nudge.xcodeproj/xcshareddata/xcschemes/Nudge - Debug (-bundle-mode).xcscheme b/Nudge.xcodeproj/xcshareddata/xcschemes/Nudge - Debug (-bundle-mode-json).xcscheme similarity index 98% rename from Nudge.xcodeproj/xcshareddata/xcschemes/Nudge - Debug (-bundle-mode).xcscheme rename to Nudge.xcodeproj/xcshareddata/xcschemes/Nudge - Debug (-bundle-mode-json).xcscheme index cbd5eef3..780f078a 100644 --- a/Nudge.xcodeproj/xcshareddata/xcschemes/Nudge - Debug (-bundle-mode).xcscheme +++ b/Nudge.xcodeproj/xcshareddata/xcschemes/Nudge - Debug (-bundle-mode-json).xcscheme @@ -72,7 +72,7 @@ diff --git a/Nudge.xcodeproj/xcshareddata/xcschemes/Nudge - Debug (-bundle-mode-profile).xcscheme b/Nudge.xcodeproj/xcshareddata/xcschemes/Nudge - Debug (-bundle-mode-profile).xcscheme new file mode 100644 index 00000000..2989bbfe --- /dev/null +++ b/Nudge.xcodeproj/xcshareddata/xcschemes/Nudge - Debug (-bundle-mode-profile).xcscheme @@ -0,0 +1,104 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/Nudge.xcodeproj/xcshareddata/xcschemes/Nudge - Debug (-demo-mode).xcscheme b/Nudge.xcodeproj/xcshareddata/xcschemes/Nudge - Debug (-demo-mode).xcscheme index cd3f835a..ecb73650 100644 --- a/Nudge.xcodeproj/xcshareddata/xcschemes/Nudge - Debug (-demo-mode).xcscheme +++ b/Nudge.xcodeproj/xcshareddata/xcschemes/Nudge - Debug (-demo-mode).xcscheme @@ -72,7 +72,7 @@ diff --git a/Nudge.xcodeproj/xcuserdata/erikg.xcuserdatad/xcschemes/xcschememanagement.plist b/Nudge.xcodeproj/xcuserdata/erikg.xcuserdatad/xcschemes/xcschememanagement.plist index a62fd3de..4ac0238e 100644 --- a/Nudge.xcodeproj/xcuserdata/erikg.xcuserdatad/xcschemes/xcschememanagement.plist +++ b/Nudge.xcodeproj/xcuserdata/erikg.xcuserdatad/xcschemes/xcschememanagement.plist @@ -4,11 +4,16 @@ SchemeUserState - Nudge - Debug (-bundle-mode).xcscheme_^#shared#^_ + Nudge - Debug (-bundle-mode-json).xcscheme_^#shared#^_ orderHint 4 + Nudge - Debug (-bundle-mode-profile).xcscheme_^#shared#^_ + + orderHint + 5 + Nudge - Debug (-demo-mode).xcscheme_^#shared#^_ orderHint @@ -22,12 +27,12 @@ Nudge - Debug (-demo-mode, -force-screenshot-icon).xcscheme_^#shared#^_ orderHint - 5 + 6 Nudge - Debug (-demo-mode, -simple-mode).xcscheme_^#shared#^_ orderHint - 6 + 7 Nudge - Debug (-simple-mode).xcscheme_^#shared#^_ @@ -42,7 +47,7 @@ Nudge - Release.xcscheme_^#shared#^_ orderHint - 7 + 8 Nudge Release.xcscheme_^#shared#^_ diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index b1c07f7f..33fb7e88 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -286,10 +286,19 @@ struct CameraUtilities { struct CommandLineUtilities { let arguments = Set(CommandLine.arguments) - func bundleModeEnabled() -> Bool { - let argumentPassed = arguments.contains("-bundle-mode") + func bundleModeJSONEnabled() -> Bool { + let argumentPassed = arguments.contains("-bundle-mode-json") if argumentPassed && !nudgeLogState.hasLoggedBundleMode { - LogManager.debug("-bundle-mode argument passed", logger: uiLog) + LogManager.debug("-bundle-mode-json argument passed", logger: uiLog) + nudgeLogState.hasLoggedBundleMode = true + } + return argumentPassed + } + + func bundleModeProfileEnabled() -> Bool { + let argumentPassed = arguments.contains("-bundle-mode-profile") + if argumentPassed && !nudgeLogState.hasLoggedBundleMode { + LogManager.debug("-bundle-mode-profile argument passed", logger: uiLog) nudgeLogState.hasLoggedBundleMode = true } return argumentPassed @@ -904,11 +913,16 @@ struct NetworkFileManager { return nil } - if CommandLineUtilities().bundleModeEnabled(), let bundleUrl = Globals.bundle.url(forResource: "com.github.macadmins.Nudge.tester", withExtension: "json") { + if CommandLineUtilities().bundleModeJSONEnabled(), let bundleUrl = Globals.bundle.url(forResource: "com.github.macadmins.Nudge.tester", withExtension: "json") { LogManager.debug("JSON url: \(bundleUrl)", logger: utilsLog) return decodeNudgePreferences(from: bundleUrl) } + if CommandLineUtilities().bundleModeProfileEnabled(), let bundleUrl = Globals.bundle.url(forResource: "com.github.macadmins.Nudge.tester", withExtension: "plist") { + LogManager.debug("Profile url: \(bundleUrl)", logger: utilsLog) + return decodeNudgePreferences(from: bundleUrl) + } + if let jsonUrl = URL(string: url) { LogManager.debug("JSON url: \(url)", logger: utilsLog) return decodeNudgePreferences(from: jsonUrl) From 1ba078ff3214b4de18bc5f35837b1bac1516e806 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 22 May 2024 12:27:26 -0500 Subject: [PATCH 048/141] finish all the wrapping around local plist --- Nudge/Utilities/Preferences.swift | 138 +++++++++++++++++++++++++----- Nudge/Utilities/Utils.swift | 39 +++++++-- 2 files changed, 146 insertions(+), 31 deletions(-) diff --git a/Nudge/Utilities/Preferences.swift b/Nudge/Utilities/Preferences.swift index bc2707e4..7bdd5d39 100644 --- a/Nudge/Utilities/Preferences.swift +++ b/Nudge/Utilities/Preferences.swift @@ -33,19 +33,37 @@ func getOptionalFeaturesJSON() -> OptionalFeatures? { } func getOptionalFeaturesProfile() -> [String: Any]? { - guard !CommandLineUtilities().demoModeEnabled(), - !CommandLineUtilities().unitTestingEnabled(), - let optionalFeatures = Globals.nudgeDefaults.dictionary(forKey: "optionalFeatures") else { + if CommandLineUtilities().demoModeEnabled() || CommandLineUtilities().unitTestingEnabled() { logEmptyKey("optionalFeatures", forJSON: false) return nil } - return optionalFeatures + + // Check if the Data is empty + if Globals.configProfile.isEmpty { + logEmptyKey("optionalFeatures", forJSON: false) + return nil + } + + do { + // Attempt to decode the plist Data into a dictionary + if let dictionary = try PropertyListSerialization.propertyList(from: Globals.configProfile, options: [], format: nil) as? [String: Any], + let optionalFeatures = dictionary["optionalFeatures"] as? [String: Any] { + return optionalFeatures + } else { + logEmptyKey("optionalFeatures", forJSON: false) + return nil + } + } catch { + print("Failed to decode plist: \(error)") + return nil + } } private func logEmptyKey(_ key: String, forJSON: Bool) { if !nudgeLogState.afterFirstLaunch { let log = forJSON ? prefsJSONLog : prefsProfileLog - LogManager.info("\(key) key is empty", logger: log) + let type = forJSON ? "json" : "profile" + LogManager.info("\(key) key is empty - \(type)", logger: log) } } @@ -83,13 +101,31 @@ func getOSVersionRequirementsJSON() -> OSVersionRequirement? { } func getOSVersionRequirementsProfile() -> OSVersionRequirement? { - guard let osRequirementsArray = Globals.nudgeDefaults.array(forKey: "osVersionRequirements") as? [[String: AnyObject]] else { + if CommandLineUtilities().demoModeEnabled() || CommandLineUtilities().unitTestingEnabled() { + logEmptyKey("osVersionRequirements", forJSON: false) + return nil + } + + // Check if the Data is empty + if Globals.configProfile.isEmpty { logEmptyKey("osVersionRequirements", forJSON: false) return nil } - let requirements = osRequirementsArray.map { OSVersionRequirement(fromDictionary: $0) } - return getOSVersionRequirements(from: requirements) + do { + // Attempt to decode the plist Data into a dictionary + if let dictionary = try PropertyListSerialization.propertyList(from: Globals.configProfile, options: [], format: nil) as? [String: Any], + let osVersionRequirements = dictionary["osVersionRequirements"] as? [[String: AnyObject]] { + let requirements = osVersionRequirements.map { OSVersionRequirement(fromDictionary: $0) } + return getOSVersionRequirements(from: requirements) + } else { + logEmptyKey("osVersionRequirements", forJSON: false) + return nil + } + } catch { + print("Failed to decode plist: \(error)") + return nil + } } private func getOSVersionRequirements(from requirements: [OSVersionRequirement]?) -> OSVersionRequirement? { @@ -156,13 +192,30 @@ func getUserExperienceJSON() -> UserExperience? { } func getUserExperienceProfile() -> [String: Any]? { - guard !CommandLineUtilities().demoModeEnabled(), - !CommandLineUtilities().unitTestingEnabled(), - let userExperience = Globals.nudgeDefaults.dictionary(forKey: "userExperience") else { + if CommandLineUtilities().demoModeEnabled() || CommandLineUtilities().unitTestingEnabled() { logEmptyKey("userExperience", forJSON: false) return nil } - return userExperience + + // Check if the Data is empty + if Globals.configProfile.isEmpty { + logEmptyKey("userExperience", forJSON: false) + return nil + } + + do { + // Attempt to decode the plist Data into a dictionary + if let dictionary = try PropertyListSerialization.propertyList(from: Globals.configProfile, options: [], format: nil) as? [String: Any], + let userExperience = dictionary["userExperience"] as? [String: Any] { + return userExperience + } else { + logEmptyKey("userExperience", forJSON: false) + return nil + } + } catch { + print("Failed to decode plist: \(error)") + return nil + } } // userInterface @@ -187,13 +240,30 @@ func getUserInterfaceJSON() -> UserInterface? { } func getUserInterfaceProfile() -> [String: Any]? { - guard !CommandLineUtilities().demoModeEnabled(), - !CommandLineUtilities().unitTestingEnabled(), - let userInterface = Globals.nudgeDefaults.dictionary(forKey: "userInterface") else { + if CommandLineUtilities().demoModeEnabled() || CommandLineUtilities().unitTestingEnabled() { logEmptyKey("userInterface", forJSON: false) return nil } - return userInterface + + // Check if the Data is empty + if Globals.configProfile.isEmpty { + logEmptyKey("userInterface", forJSON: false) + return nil + } + + do { + // Attempt to decode the plist Data into a dictionary + if let dictionary = try PropertyListSerialization.propertyList(from: Globals.configProfile, options: [], format: nil) as? [String: Any], + let userInterface = dictionary["userInterface"] as? [String: Any] { + return userInterface + } else { + logEmptyKey("userInterface", forJSON: false) + return nil + } + } catch { + print("Failed to decode plist: \(error)") + return nil + } } // Loop through JSON userInterface -> updateElements preferences and then compare language @@ -206,16 +276,38 @@ func getUserInterfaceUpdateElementsJSON() -> UpdateElement? { } func getUserInterfaceUpdateElementsProfile() -> [String: AnyObject]? { - // Mutate the profile into our required construct - guard let updateElementsArray = UserInterfaceVariables.userInterfaceProfile?["updateElements"] as? [[String: AnyObject]] else { + if CommandLineUtilities().demoModeEnabled() || CommandLineUtilities().unitTestingEnabled() { logEmptyKey("updateElements", forJSON: false) return nil } - return getMatchingUpdateElements( - updateElements: updateElementsArray, - languageKey: "_language", - logKey: "Profile" - ) + + // Check if the Data is empty + if Globals.configProfile.isEmpty { + logEmptyKey("updateElements", forJSON: false) + return nil + } + + do { + // Attempt to decode the plist Data into a dictionary + if let dictionary = try PropertyListSerialization.propertyList(from: Globals.configProfile, options: [], format: nil) as? [String: Any], + let userInterface = dictionary["userInterface"] as? [String: Any] { + guard let updateElementsArray = userInterface["updateElements"] as? [[String: AnyObject]] else { + logEmptyKey("updateElements", forJSON: false) + return nil + } + return getMatchingUpdateElements( + updateElements: updateElementsArray, + languageKey: "_language", + logKey: "Profile" + ) + } else { + logEmptyKey("updateElements", forJSON: false) + return nil + } + } catch { + print("Failed to decode plist: \(error)") + return nil + } } private func getMatchingUpdateElements(updateElements: [T]?, languageKey: String, logKey: String) -> T? { diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 33fb7e88..717c7467 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -396,10 +396,33 @@ struct ConfigurationManager { func getConfigurationAsProfile() -> Data { var nudgeProfileConfig = [String: Any]() - nudgeProfileConfig["optionalFeatures"] = Globals.nudgeDefaults.dictionary(forKey: "optionalFeatures") - nudgeProfileConfig["osVersionRequirements"] = Globals.nudgeDefaults.array(forKey: "osVersionRequirements") - nudgeProfileConfig["userExperience"] = Globals.nudgeDefaults.dictionary(forKey: "userExperience") - nudgeProfileConfig["userInterface"] = Globals.nudgeDefaults.dictionary(forKey: "userInterface") + if CommandLineUtilities().bundleModeJSONEnabled() { + return Data() + } + if CommandLineUtilities().bundleModeProfileEnabled(), let bundleUrl = Globals.bundle.url(forResource: "com.github.macadmins.Nudge.tester", withExtension: "plist") { + LogManager.debug("Profile url: \(bundleUrl)", logger: utilsLog) + guard let data = try? Data(contentsOf: bundleUrl) else { + LogManager.error("Failed to load profile data from URL.", logger: uiLog) + return Data() + } + do { + let plist = try PropertyListSerialization.propertyList(from: data, options: [], format: nil) + if let dictionary = plist as? [String: AnyObject] { + nudgeProfileConfig = dictionary + } else { + LogManager.error("Plist is not a dictionary.", logger: uiLog) + return Data() + } + } catch { + LogManager.error("Error reading plist: \(error)", logger: uiLog) + return Data() + } + } else { + nudgeProfileConfig["optionalFeatures"] = Globals.nudgeDefaults.dictionary(forKey: "optionalFeatures") + nudgeProfileConfig["osVersionRequirements"] = Globals.nudgeDefaults.array(forKey: "osVersionRequirements") + nudgeProfileConfig["userExperience"] = Globals.nudgeDefaults.dictionary(forKey: "userExperience") + nudgeProfileConfig["userInterface"] = Globals.nudgeDefaults.dictionary(forKey: "userInterface") + } guard !nudgeProfileConfig.isEmpty, let plistData = try? PropertyListSerialization.data(fromPropertyList: nudgeProfileConfig, format: .xml, options: 0), @@ -783,7 +806,7 @@ struct NetworkFileManager { private func decodeNudgePreferences(from url: URL) -> NudgePreferences? { guard let data = try? Data(contentsOf: url) else { if Globals.configProfile.isEmpty { - LogManager.error("Failed to load data from URL: \(url)", logger: prefsJSONLog) + LogManager.error("Failed to load data from URL: \(url)", logger: prefsProfileLog) } return nil } @@ -919,8 +942,8 @@ struct NetworkFileManager { } if CommandLineUtilities().bundleModeProfileEnabled(), let bundleUrl = Globals.bundle.url(forResource: "com.github.macadmins.Nudge.tester", withExtension: "plist") { - LogManager.debug("Profile url: \(bundleUrl)", logger: utilsLog) - return decodeNudgePreferences(from: bundleUrl) + LogManager.debug("Using embedded plist url: \(bundleUrl)", logger: utilsLog) + return nil } if let jsonUrl = URL(string: url) { @@ -1198,7 +1221,7 @@ struct VersionManager { static func getMajorRequiredNudgeOSVersion() -> Int { guard let majorVersion = Int(nudgePrimaryState.requiredMinimumOSVersion.split(separator: ".").first ?? "") else { - LogManager.error("Invalid format for requiredMinimumOSVersion", logger: utilsLog) + LogManager.error("Invalid format for requiredMinimumOSVersion - value is \(nudgePrimaryState.requiredMinimumOSVersion)", logger: utilsLog) return 0 } logOSVersion(majorVersion, for: "Major required OS version") From 4b33f1d1a61454bf3598a79263d87d5867843901 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 22 May 2024 12:27:53 -0500 Subject: [PATCH 049/141] fix key typo for utilizeSOFAFeed on mdm profiles --- Nudge/Preferences/DefaultPreferencesNudge.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index b3552a13..85de33af 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -156,7 +156,7 @@ struct OptionalFeatureVariables { } static var utilizeSOFAFeed: Bool { - optionalFeaturesProfile?["utilizeSOFAFeedh"] as? Bool ?? + optionalFeaturesProfile?["utilizeSOFAFeed"] as? Bool ?? optionalFeaturesJSON?.utilizeSOFAFeed ?? false } From d7f041aea169c78d228a85e6d970ed219927e9c2 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Thu, 23 May 2024 11:13:55 -0500 Subject: [PATCH 050/141] simplify the sofa logic and add latest-* supported --- CHANGELOG.md | 14 ++- .../com.github.macadmins.Nudge.tester.json | 3 +- .../com.github.macadmins.Nudge.tester.plist | 2 - Nudge/3rd Party Assets/sofa.swift | 23 +++- .../Preferences/DefaultPreferencesNudge.swift | 36 ++++--- Nudge/Preferences/PreferencesStructure.swift | 16 ++- Nudge/UI/Main.swift | 101 ++++++++++-------- 7 files changed, 127 insertions(+), 68 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 466ed191..03066044 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -8,17 +8,23 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 Requires macOS 12.0 and higher. Further releases and feature requests may make this macOS 13 and higher depending on code complexity. ### Changed -- You can now pass the string `latest` in the `requiredMinimumOSVersion` key +- You can now pass the strings `latest`, `latest-supported` and `latest-minor` in the `requiredMinimumOSVersion` key + - `latest`: always force latest release and if the machine can't this version, show the new "unsupported device" user interface + - `latest-supported`: always get the latest version sofa shows that is supported by this device + - `latest-minor`: stay in the current major release and get the latest minor updates available - This requires utilizing the SOFA feed features to properly work - Nudge will then utilize two date integers to automatically calculate the `requiredInstallationDate` - - `activelyExploitedInstallationSLA` under the `osVersionRequirement` key will default to 14 days + - `activelyExploitedCVEsInstallationSLA` under the `osVersionRequirement` key will default to 14 days + - `nonActivelyExploitedCVEsSLA` under the `osVersionRequirement` key will default to 21 days - `standardInstallationSLA` under the `osVersionRequirement` key will default to 28 days - These dates are calculated against the `ReleaseDate` key in the SOFA feed + - If you'd like to not have nudge events for releases without any known CVEs, please configure the `disableNudgeForStandardInstalls` key under `optionalFeatures` to true ### Fixed Upcoming ### Added +- Remote URLs can now be used on `iconDarkPath`, `iconLightPath`, `screenShotDarkPath` and `screenShotLightPath` - An admin can now allow users to move the Nudge window with `userExperience` key `allowMovableWindow` - Basic SwiftUI support for Markdown text options - Utilizing Apple's markdown features, you can now utilize, bold, italic, underline, subscript and url links directly into any of the text fields @@ -26,8 +32,8 @@ Upcoming - Set the `utilizeSOFAFeed` key `true` under `optionalFeatures` to enable this feature - Nudge will by default check the feed every 24 hours. - In order to change this, please configure the `refreshSOFAFeedTime` key under `optionalFeatures` in seconds -- "Unsupported" device UI in standard mode that utilizes the SOFA feed - - Set the `attemptToCheckForSupportedDevice` key `true` under `optionalFeatures` to enable this feature +- "Unsupported device" UI in standard mode that utilizes the SOFA feed + - Set the `attemptToCheckForSupportedDevice` key `false` under `optionalFeatures` to disable this feature - There are now keys to set all of text fields - `informationButtonTextUnsupported`, `mainContentHeaderUnsupported`, `mainContentNoteUnsupported`, `mainContentSubHeaderUnsupported`, `mainContentTextUnsupported`, `subHeaderUnsupported` - `unsupportedURL` and `unsupportedURLs` can change the information button itself, but it will remain in the `osVersionRequirement` key with `unsupportedURLs` and `unsupportedURLs`. diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index f56eeec5..8c921705 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -10,7 +10,6 @@ "acceptableCameraUsage": true, "acceptableScreenSharingUsage": true, "attemptToBlockApplicationLaunches": true, - "attemptToCheckForSupportedDevice": true, "blockedApplicationBundleIDs": [ "com.microsoft.VSCode", "us.zoom.xos" @@ -21,7 +20,7 @@ "osVersionRequirements": [ { "aboutUpdateURL": "https://apple.com", - "requiredMinimumOSVersion": "latest-supported", + "requiredMinimumOSVersion": "latest", "unsupportedURL": "https://google.com", } ], diff --git a/Example Assets/com.github.macadmins.Nudge.tester.plist b/Example Assets/com.github.macadmins.Nudge.tester.plist index a58bd10d..9654d0a5 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.plist +++ b/Example Assets/com.github.macadmins.Nudge.tester.plist @@ -19,8 +19,6 @@ attemptToBlockApplicationLaunches - attemptToCheckForSupportedDevice - blockedApplicationBundleIDs com.microsoft.VSCode diff --git a/Nudge/3rd Party Assets/sofa.swift b/Nudge/3rd Party Assets/sofa.swift index 8c066a0b..bc4ca3fd 100644 --- a/Nudge/3rd Party Assets/sofa.swift +++ b/Nudge/3rd Party Assets/sofa.swift @@ -39,6 +39,17 @@ struct SofaOSVersion: Codable { } } +protocol OSInformation { + var productVersion: String { get } + var build: String { get } + var releaseDate: Date? { get } + var securityInfo: String { get } + var supportedDevices: [String] { get } + var cves: [String: Bool] { get } + var activelyExploitedCVEs: [String] { get } + var uniqueCVEsCount: Int { get } +} + struct LatestOS: Codable { let productVersion, build: String let releaseDate: Date? @@ -62,9 +73,13 @@ struct LatestOS: Codable { } } +extension LatestOS: OSInformation { + // All required properties are already implemented +} + struct SecurityRelease: Codable { let updateName, productVersion: String - let releaseDate: Date + let releaseDate: Date? let securityInfo: String let supportedDevices: [String] let cves: [String: Bool] @@ -84,6 +99,12 @@ struct SecurityRelease: Codable { } } +extension SecurityRelease: OSInformation { + var build: String { + "" + } // fake out build for now +} + struct SupportedModel: Codable { let model: String let url: String diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 85de33af..2ac6fd18 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -116,7 +116,7 @@ struct OptionalFeatureVariables { static var attemptToCheckForSupportedDevice: Bool { optionalFeaturesProfile?["attemptToCheckForSupportedDevice"] as? Bool ?? optionalFeaturesJSON?.attemptToCheckForSupportedDevice ?? - false + true } static var attemptToFetchMajorUpgrade: Bool { @@ -131,18 +131,24 @@ struct OptionalFeatureVariables { [String]() } - static var enforceMinorUpdates: Bool { - optionalFeaturesProfile?["enforceMinorUpdates"] as? Bool ?? - optionalFeaturesJSON?.enforceMinorUpdates ?? - true - } - static var disableSoftwareUpdateWorkflow: Bool { optionalFeaturesProfile?["disableSoftwareUpdateWorkflow"] as? Bool ?? optionalFeaturesJSON?.disableSoftwareUpdateWorkflow ?? false } + static var disableNudgeForStandardInstalls: Bool { + optionalFeaturesProfile?["disableNudgeForStandardInstalls"] as? Bool ?? + optionalFeaturesJSON?.disableNudgeForStandardInstalls ?? + false + } + + static var enforceMinorUpdates: Bool { + optionalFeaturesProfile?["enforceMinorUpdates"] as? Bool ?? + optionalFeaturesJSON?.enforceMinorUpdates ?? + true + } + static var refreshSOFAFeedTime: Int { optionalFeaturesProfile?["refreshSOFAFeedTime"] as? Int ?? optionalFeaturesJSON?.refreshSOFAFeedTime ?? @@ -176,9 +182,9 @@ struct OSVersionRequirementVariables { "" } - static var activelyExploitedInstallationSLA: Int { - osVersionRequirementsProfile?.activelyExploitedInstallationSLA ?? - osVersionRequirementsJSON?.activelyExploitedInstallationSLA ?? + static var activelyExploitedCVEsInstallationSLA: Int { + osVersionRequirementsProfile?.activelyExploitedCVEsInstallationSLA ?? + osVersionRequirementsJSON?.activelyExploitedCVEsInstallationSLA ?? 14 } @@ -187,9 +193,15 @@ struct OSVersionRequirementVariables { osVersionRequirementsJSON?.majorUpgradeAppPath ?? "" } - + + static var nonActivelyExploitedCVEsSLA: Int { + osVersionRequirementsProfile?.nonActivelyExploitedCVEsSLA ?? + osVersionRequirementsJSON?.nonActivelyExploitedCVEsSLA ?? + 21 + } + static var requiredMinimumOSVersion: String { - if ["latest", "latest-supported", "latest-major"].contains(PrefsWrapper.requiredMinimumOSVersion) { + if ["latest", "latest-supported", "latest-minor"].contains(PrefsWrapper.requiredMinimumOSVersion) { PrefsWrapper.requiredMinimumOSVersion } else { try! OSVersion(PrefsWrapper.requiredMinimumOSVersion).description diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index e9403baf..906d1b84 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -53,7 +53,7 @@ struct OptionalFeatures: Codable { var acceptableApplicationBundleIDs, acceptableAssertionApplicationNames: [String]? var acceptableAssertionUsage, acceptableCameraUsage, acceptableScreenSharingUsage, aggressiveUserExperience, aggressiveUserFullScreenExperience, asynchronousSoftwareUpdate, attemptToBlockApplicationLaunches, attemptToCheckForSupportedDevice, attemptToFetchMajorUpgrade: Bool? var blockedApplicationBundleIDs: [String]? - var disableSoftwareUpdateWorkflow, enforceMinorUpdates: Bool? + var disableNudgeForStandardInstalls, disableSoftwareUpdateWorkflow, enforceMinorUpdates: Bool? var refreshSOFAFeedTime: Int? var terminateApplicationsOnLaunch, utilizeSOFAFeed: Bool? } @@ -89,6 +89,7 @@ extension OptionalFeatures { attemptToCheckForSupportedDevice: Bool? = nil, attemptToFetchMajorUpgrade: Bool? = nil, blockedApplicationBundleIDs: [String]? = nil, + disableNudgeForStandardInstalls: Bool? = nil, disableSoftwareUpdateWorkflow: Bool? = nil, enforceMinorUpdates: Bool? = nil, refreshSOFAFeedTime: Int? = nil, @@ -108,6 +109,7 @@ extension OptionalFeatures { attemptToCheckForSupportedDevice: attemptToCheckForSupportedDevice ?? self.attemptToCheckForSupportedDevice, attemptToFetchMajorUpgrade: attemptToFetchMajorUpgrade ?? self.attemptToFetchMajorUpgrade, blockedApplicationBundleIDs: blockedApplicationBundleIDs ?? self.blockedApplicationBundleIDs, + disableNudgeForStandardInstalls: disableNudgeForStandardInstalls ?? self.disableNudgeForStandardInstalls, disableSoftwareUpdateWorkflow: disableSoftwareUpdateWorkflow ?? self.disableSoftwareUpdateWorkflow, enforceMinorUpdates: enforceMinorUpdates ?? self.enforceMinorUpdates, refreshSOFAFeedTime: refreshSOFAFeedTime ?? self.refreshSOFAFeedTime, @@ -122,8 +124,9 @@ struct OSVersionRequirement: Codable { var aboutUpdateURL: String? var aboutUpdateURLs: [AboutUpdateURL]? var actionButtonPath: String? - var activelyExploitedInstallationSLA: Int? + var activelyExploitedCVEsInstallationSLA: Int? var majorUpgradeAppPath: String? + var nonActivelyExploitedCVEsSLA: Int? var requiredInstallationDate: Date? var requiredMinimumOSVersion: String? var standardInstallationSLA: Int? @@ -137,8 +140,9 @@ extension OSVersionRequirement { init(fromDictionary: [String: AnyObject]) { self.aboutUpdateURL = fromDictionary["aboutUpdateURL"] as? String self.actionButtonPath = fromDictionary["actionButtonPath"] as? String - self.activelyExploitedInstallationSLA = fromDictionary["activelyExploitedInstallationSLA"] as? Int + self.activelyExploitedCVEsInstallationSLA = fromDictionary["activelyExploitedCVEsInstallationSLA"] as? Int self.majorUpgradeAppPath = fromDictionary["majorUpgradeAppPath"] as? String + self.nonActivelyExploitedCVEsSLA = fromDictionary["nonActivelyExploitedCVEsSLA"] as? Int self.requiredMinimumOSVersion = fromDictionary["requiredMinimumOSVersion"] as? String self.standardInstallationSLA = fromDictionary["standardInstallationSLA"] as? Int self.targetedOSVersionsRule = fromDictionary["targetedOSVersionsRule"] as? String @@ -196,8 +200,9 @@ extension OSVersionRequirement { aboutUpdateURL: String? = nil, aboutUpdateURLs: [AboutUpdateURL]? = nil, actionButtonPath: String? = nil, - activelyExploitedInstallationSLA: Int? = nil, + activelyExploitedCVEsInstallationSLA: Int? = nil, majorUpgradeAppPath: String? = nil, + nonActivelyExploitedCVEsSLA: Int? = nil, requiredInstallationDate: Date? = nil, requiredMinimumOSVersion: String? = nil, standardInstallationSLA: Int? = nil, @@ -209,8 +214,9 @@ extension OSVersionRequirement { aboutUpdateURL: aboutUpdateURL ?? self.aboutUpdateURL, aboutUpdateURLs: aboutUpdateURLs ?? self.aboutUpdateURLs, actionButtonPath: actionButtonPath ?? self.actionButtonPath, - activelyExploitedInstallationSLA: activelyExploitedInstallationSLA ?? self.activelyExploitedInstallationSLA, + activelyExploitedCVEsInstallationSLA: activelyExploitedCVEsInstallationSLA ?? self.activelyExploitedCVEsInstallationSLA, majorUpgradeAppPath: majorUpgradeAppPath ?? self.majorUpgradeAppPath, + nonActivelyExploitedCVEsSLA: nonActivelyExploitedCVEsSLA ?? self.nonActivelyExploitedCVEsSLA, requiredInstallationDate: requiredInstallationDate ?? self.requiredInstallationDate, requiredMinimumOSVersion: requiredMinimumOSVersion ?? self.requiredMinimumOSVersion, standardInstallationSLA: standardInstallationSLA ?? self.standardInstallationSLA, diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index afbb99fa..b6fb3314 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -181,63 +181,80 @@ class AppDelegate: NSObject, NSApplicationDelegate { // TODO: add support for custom sofa feed url // TODO: check the sofa json etag even after the timelimit age to reduce more bandwidth if OptionalFeatureVariables.utilizeSOFAFeed { + var selectedOS: OSInformation? + var foundMatch = false if let macOSSOFAAssets = Globals.sofaAssets?.osVersions { - var foundMatch = false for osVersion in macOSSOFAAssets { - if ["latest", "latest-supported", "latest-major"].contains(PrefsWrapper.requiredMinimumOSVersion) { - if PrefsWrapper.requiredMinimumOSVersion == "latest-supported" { - if VersionManager.getMajorOSVersion() != Int(osVersion.osVersion.split(separator: " ").last!) { - continue - } + if PrefsWrapper.requiredMinimumOSVersion == "latest" { + selectedOS = osVersion.latest + } else if PrefsWrapper.requiredMinimumOSVersion == "latest-minor" { + if VersionManager.getMajorOSVersion() == Int(osVersion.osVersion.split(separator: " ").last!) { + selectedOS = osVersion.latest + } else { + continue } - // Check if the specified device is in the supported devices of the matching asset - nudgePrimaryState.requiredMinimumOSVersion = osVersion.latest.productVersion - // nudgePrimaryState.requiredMinimumOSVersion = "14.6" // TODO: remove when testing is done - nudgePrimaryState.activelyExploitedCVEs = osVersion.latest.activelyExploitedCVEs.count > 0 - LogManager.notice("SOFA Actively Exploited CVEs: \(nudgePrimaryState.activelyExploitedCVEs)", logger: sofaLog) - let slaExtension = nudgePrimaryState.activelyExploitedCVEs ? OSVersionRequirementVariables.activelyExploitedInstallationSLA * 86400 : OSVersionRequirementVariables.standardInstallationSLA * 86400 - requiredInstallationDate = osVersion.latest.releaseDate?.addingTimeInterval(TimeInterval(slaExtension)) ?? DateManager().getCurrentDate().addingTimeInterval(1209600) - LogManager.notice("Extending requiredInstallationDate to \(requiredInstallationDate)", logger: sofaLog) - LogManager.notice("SOFA Matched OS Version: \(osVersion.latest.productVersion)", logger: sofaLog) - LogManager.notice("SOFA Assets: \(osVersion.latest.supportedDevices)", logger: sofaLog) - LogManager.notice("SOFA CVEs: \(osVersion.latest.cves)", logger: sofaLog) - + } else if PrefsWrapper.requiredMinimumOSVersion == "latest-supported" { if OptionalFeatureVariables.attemptToCheckForSupportedDevice { - LogManager.notice("Assessed Model ID: \(Globals.hardwareModelID)", logger: sofaLog) - let deviceMatchFound = osVersion.latest.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }) - - LogManager.notice("Assessed Model ID found in SOFA Entry: \(deviceMatchFound)", logger: sofaLog) - nudgePrimaryState.deviceSupportedByOSVersion = deviceMatchFound - // nudgePrimaryState.deviceSupportedByOSVersion = false // TODO: remove when testing is done + selectedOS = osVersion.securityReleases.first + if !selectedOS!.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }) { + continue + } + } else { + LogManager.notice("Attempting to use latest-supported without supported device UI features. Please set attemptToCheckForSupportedDevice to true", logger: sofaLog) + break } - foundMatch = true } else { - if let matchingAsset = osVersion.securityReleases.first(where: { $0.productVersion == nudgePrimaryState.requiredMinimumOSVersion }) { - // Check if the specified device is in the supported devices of the matching asset - LogManager.notice("SOFA Matched OS Version: \(matchingAsset.productVersion)", logger: sofaLog) - LogManager.notice("SOFA Assets: \(matchingAsset.supportedDevices)", logger: sofaLog) - if OptionalFeatureVariables.attemptToCheckForSupportedDevice { - LogManager.notice("Assessed Model ID: \(Globals.hardwareModelID)", logger: sofaLog) - let deviceMatchFound = matchingAsset.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }) - print("CVEs: \(matchingAsset.cves)") - print("Actively Exploited CVEs: \(matchingAsset.activelyExploitedCVEs.count > 0)") - LogManager.notice("Assessed Model ID found in SOFA Entry: \(deviceMatchFound)", logger: sofaLog) - nudgePrimaryState.deviceSupportedByOSVersion = deviceMatchFound - // nudgePrimaryState.deviceSupportedByOSVersion = false - } - foundMatch = true + if osVersion.securityReleases.first(where: { $0.productVersion == nudgePrimaryState.requiredMinimumOSVersion }) != nil { + selectedOS = osVersion.securityReleases.first + } else { + continue } } - if foundMatch { - break + let activelyExploitedCVEs = selectedOS!.activelyExploitedCVEs.count > 0 + let presentCVEs = selectedOS!.cves.count > 0 + let slaExtension: TimeInterval + switch (activelyExploitedCVEs, presentCVEs) { + case (false, true): + slaExtension = TimeInterval(OSVersionRequirementVariables.nonActivelyExploitedCVEsSLA * 86400) + case (true, true): + slaExtension = TimeInterval(OSVersionRequirementVariables.activelyExploitedCVEsInstallationSLA * 86400) + case (false, false): + slaExtension = TimeInterval(OSVersionRequirementVariables.standardInstallationSLA * 86400) + default: + slaExtension = TimeInterval(OSVersionRequirementVariables.standardInstallationSLA * 86400) + } + + if OptionalFeatureVariables.disableNudgeForStandardInstalls && !presentCVEs { + LogManager.notice("No known CVEs for \(selectedOS!.productVersion) and disableNudgeForStandardInstalls is set to true", logger: sofaLog) + AppStateManager().exitNudge() + } + LogManager.notice("SOFA Actively Exploited CVEs: \(activelyExploitedCVEs)", logger: sofaLog) + + // Start setting UI fields + nudgePrimaryState.requiredMinimumOSVersion = osVersion.latest.productVersion + nudgePrimaryState.activelyExploitedCVEs = activelyExploitedCVEs + requiredInstallationDate = selectedOS!.releaseDate?.addingTimeInterval(slaExtension) ?? DateManager().getCurrentDate().addingTimeInterval(TimeInterval(OSVersionRequirementVariables.standardInstallationSLA * 86400)) + + LogManager.notice("Extending requiredInstallationDate to \(requiredInstallationDate)", logger: sofaLog) + LogManager.notice("SOFA Matched OS Version: \(selectedOS!.productVersion)", logger: sofaLog) + LogManager.notice("SOFA Assets: \(selectedOS!.supportedDevices)", logger: sofaLog) + LogManager.notice("SOFA CVEs: \(selectedOS!.cves)", logger: sofaLog) + + if OptionalFeatureVariables.attemptToCheckForSupportedDevice { + LogManager.notice("Assessed Model ID: \(Globals.hardwareModelID)", logger: sofaLog) + let deviceMatchFound = selectedOS!.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }) + LogManager.notice("Assessed Model ID found in SOFA Entry: \(deviceMatchFound)", logger: sofaLog) + nudgePrimaryState.deviceSupportedByOSVersion = deviceMatchFound } + foundMatch = true + break } if !foundMatch { // If no matching product version found or the device is not supported, return false LogManager.notice("Could not find requiredMinimumOSVersion \(nudgePrimaryState.requiredMinimumOSVersion) in SOFA feed", logger: sofaLog) } } else { - LogManager.notice("Could not fetch SOFA feed", logger: sofaLog) + LogManager.error("Could not fetch SOFA feed", logger: sofaLog) } } handleSMAppService() From 5fbd5178854f78edec99ce5eef653e9faab41257 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Thu, 23 May 2024 11:19:34 -0500 Subject: [PATCH 051/141] fix bug when matching requiredMinimumOSVersion with OS version values --- Nudge/UI/Main.swift | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index b6fb3314..b0898472 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -174,7 +174,6 @@ class AppDelegate: NSObject, NSApplicationDelegate { // Pre-Launch Logic func applicationWillFinishLaunching(_ notification: Notification) { // print("applicationWillFinishLaunching") - // TODO: Implement "latest-minor" or something for implementing all of the minor releases. // TODO: Add more logging to "unsupported devices" UI. // TODO: Add localization for "unsupported devices" text fields // TODO: Get someone to update JAMF JSON schema for all the new keys and wiki @@ -205,7 +204,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { } } else { if osVersion.securityReleases.first(where: { $0.productVersion == nudgePrimaryState.requiredMinimumOSVersion }) != nil { - selectedOS = osVersion.securityReleases.first + selectedOS = osVersion.securityReleases.first(where: { $0.productVersion == nudgePrimaryState.requiredMinimumOSVersion }) } else { continue } From 1e0fa3d8d862e45219353d3ba4244879d40d93ec Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Thu, 23 May 2024 11:19:52 -0500 Subject: [PATCH 052/141] test the skip macOS versions no CVE logic --- Example Assets/com.github.macadmins.Nudge.tester.json | 1 + 1 file changed, 1 insertion(+) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index 8c921705..26200cda 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -14,6 +14,7 @@ "com.microsoft.VSCode", "us.zoom.xos" ], + "disableNudgeForStandardInstalls": true, "terminateApplicationsOnLaunch": false, "utilizeSOFAFeed": true }, From 863477a2d64d1fbce98e814f63e54196a8eed76a Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Thu, 23 May 2024 11:32:00 -0500 Subject: [PATCH 053/141] add showActivelyExploitedCVEs --- CHANGELOG.md | 2 ++ Nudge/Preferences/DefaultPreferencesNudge.swift | 6 ++++++ Nudge/Preferences/PreferencesStructure.swift | 4 +++- Nudge/UI/StandardMode/LeftSide.swift | 2 +- 4 files changed, 12 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 03066044..b352d1a7 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -25,6 +25,8 @@ Upcoming ### Added - Remote URLs can now be used on `iconDarkPath`, `iconLightPath`, `screenShotDarkPath` and `screenShotLightPath` +- Actively Exploited CVEs in the left sidebar + - To disable this item, please configure the `showActivelyExploitedCVEs` key under `userInterface` to false - An admin can now allow users to move the Nudge window with `userExperience` key `allowMovableWindow` - Basic SwiftUI support for Markdown text options - Utilizing Apple's markdown features, you can now utilize, bold, italic, underline, subscript and url links directly into any of the text fields diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 2ac6fd18..f25f280c 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -479,6 +479,12 @@ struct UserInterfaceVariables { "I understand" } + static var showActivelyExploitedCVEs: Bool { + userInterfaceProfile?["showActivelyExploitedCVEs"] as? Bool ?? + userInterfaceJSON?.showActivelyExploitedCVEs ?? + true + } + static var showDeferralCount: Bool { userInterfaceProfile?["showDeferralCount"] as? Bool ?? userInterfaceJSON?.showDeferralCount ?? diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index 906d1b84..2cc64cab 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -397,7 +397,7 @@ struct UserInterface: Codable { var actionButtonPath, fallbackLanguage: String? var forceFallbackLanguage, forceScreenShotIcon: Bool? var iconDarkPath, iconLightPath, screenShotDarkPath, screenShotLightPath: String? - var showDeferralCount, simpleMode, singleQuitButton: Bool? + var showActivelyExploitedCVEs, showDeferralCount, simpleMode, singleQuitButton: Bool? var updateElements: [UpdateElement]? } @@ -428,6 +428,7 @@ extension UserInterface { iconLightPath: String? = nil, screenShotDarkPath: String? = nil, screenShotLightPath: String? = nil, + showActivelyExploitedCVEs: Bool? = nil, showDeferralCount: Bool? = nil, simpleMode: Bool? = nil, singleQuitButton: Bool? = nil, @@ -442,6 +443,7 @@ extension UserInterface { iconLightPath: iconLightPath ?? self.iconLightPath, screenShotDarkPath: screenShotDarkPath ?? self.screenShotDarkPath, screenShotLightPath: screenShotLightPath ?? self.screenShotLightPath, + showActivelyExploitedCVEs: showActivelyExploitedCVEs ?? self.showActivelyExploitedCVEs, showDeferralCount: showDeferralCount ?? self.showDeferralCount, simpleMode: simpleMode ?? self.simpleMode, singleQuitButton: singleQuitButton ?? self.singleQuitButton, diff --git a/Nudge/UI/StandardMode/LeftSide.swift b/Nudge/UI/StandardMode/LeftSide.swift index ea9ea8ff..b57efc0f 100644 --- a/Nudge/UI/StandardMode/LeftSide.swift +++ b/Nudge/UI/StandardMode/LeftSide.swift @@ -37,7 +37,7 @@ struct StandardModeLeftSide: View { private var informationStack: some View { VStack(alignment: .center, spacing: interLineSpacing) { InfoRow(label: "Required OS Version:", value: String(appState.requiredMinimumOSVersion), boldText: true) - if OptionalFeatureVariables.utilizeSOFAFeed { + if OptionalFeatureVariables.utilizeSOFAFeed && UserInterfaceVariables.showActivelyExploitedCVEs { InfoRow(label: "Actively Exploited CVEs:", value: String(appState.activelyExploitedCVEs).capitalized, boldText: appState.activelyExploitedCVEs) } InfoRow(label: "Current OS Version:", value: GlobalVariables.currentOSVersion) From 8a490797412404341a7adf58f5982603000551f5 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 24 May 2024 08:17:42 -0500 Subject: [PATCH 054/141] active exploit cve test to red when true --- Nudge/UI/StandardMode/LeftSide.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Nudge/UI/StandardMode/LeftSide.swift b/Nudge/UI/StandardMode/LeftSide.swift index b57efc0f..0d5d858f 100644 --- a/Nudge/UI/StandardMode/LeftSide.swift +++ b/Nudge/UI/StandardMode/LeftSide.swift @@ -38,7 +38,7 @@ struct StandardModeLeftSide: View { VStack(alignment: .center, spacing: interLineSpacing) { InfoRow(label: "Required OS Version:", value: String(appState.requiredMinimumOSVersion), boldText: true) if OptionalFeatureVariables.utilizeSOFAFeed && UserInterfaceVariables.showActivelyExploitedCVEs { - InfoRow(label: "Actively Exploited CVEs:", value: String(appState.activelyExploitedCVEs).capitalized, boldText: appState.activelyExploitedCVEs) + InfoRow(label: "Actively Exploited CVEs:", value: String(appState.activelyExploitedCVEs).capitalized, isHighlighted: appState.activelyExploitedCVEs ? true : false, boldText: appState.activelyExploitedCVEs) } InfoRow(label: "Current OS Version:", value: GlobalVariables.currentOSVersion) remainingTimeRow From 818503632ff5bbb6b927d1b69081c6c6dc82517e Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 24 May 2024 08:31:01 -0500 Subject: [PATCH 055/141] add user-agent in sofa request --- Nudge/3rd Party Assets/sofa.swift | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/Nudge/3rd Party Assets/sofa.swift b/Nudge/3rd Party Assets/sofa.swift index bc4ca3fd..31a3deac 100644 --- a/Nudge/3rd Party Assets/sofa.swift +++ b/Nudge/3rd Party Assets/sofa.swift @@ -224,7 +224,8 @@ extension MacOSDataFeed { class SOFA: NSObject, URLSessionDelegate { func URLSync(url: URL, maxRetries: Int = 3) -> (data: Data?, response: URLResponse?, error: Error?) { let semaphore = DispatchSemaphore(value: 0) - let request = URLRequest(url: url) + var request = URLRequest(url: url) + request.setValue("\(Globals.bundleID)/\(VersionManager.getNudgeVersion())", forHTTPHeaderField: "User-Agent") let session = URLSession(configuration: .default, delegate: self, delegateQueue: nil) var attempts = 0 From 31dd94b6b1223b23fa6d4a2c05ec2ea55a00445f Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 24 May 2024 08:31:12 -0500 Subject: [PATCH 056/141] move to globals.bundleid for all bundleid calls --- Nudge/Utilities/Logger.swift | 2 +- Nudge/Utilities/Utils.swift | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/Nudge/Utilities/Logger.swift b/Nudge/Utilities/Logger.swift index eab199a6..d58294b4 100644 --- a/Nudge/Utilities/Logger.swift +++ b/Nudge/Utilities/Logger.swift @@ -10,7 +10,7 @@ import os // Logger Manager struct LogManager { - static private let bundleID = Bundle.main.bundleIdentifier ?? "com.github.macadmins.Nudge" + static private let bundleID = Globals.bundleID static func createLogger(category: String) -> Logger { return Logger(subsystem: bundleID, category: category) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 717c7467..f5b32883 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -862,7 +862,7 @@ struct NetworkFileManager { } let fileManager = FileManager.default let appSupportDirectory = fileManager.urls(for: .applicationSupportDirectory, in: .userDomainMask).first! - let appDirectory = appSupportDirectory.appendingPathComponent(Bundle.main.bundleIdentifier ?? "com.github.macadmins.Nudge") + let appDirectory = appSupportDirectory.appendingPathComponent(Globals.bundleID) let sofaFile = "sofa-macos_data_feed.json" let sofaPath = appDirectory.appendingPathComponent(sofaFile) if fileManager.fileExists(atPath: sofaPath.path) { From ead34ff95e1c594f94f09c7f02e1eded9257f7e4 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 24 May 2024 08:37:53 -0500 Subject: [PATCH 057/141] add customSOFAFeedURL --- CHANGELOG.md | 1 + Nudge/Preferences/DefaultPreferencesNudge.swift | 6 ++++++ Nudge/Preferences/PreferencesStructure.swift | 3 +++ Nudge/UI/Main.swift | 1 - Nudge/Utilities/Utils.swift | 2 +- 5 files changed, 11 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index b352d1a7..7e25cd83 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -34,6 +34,7 @@ Upcoming - Set the `utilizeSOFAFeed` key `true` under `optionalFeatures` to enable this feature - Nudge will by default check the feed every 24 hours. - In order to change this, please configure the `refreshSOFAFeedTime` key under `optionalFeatures` in seconds + - If you are utilizing a custom sofa feed, please configure the `customSOFAFeedURL` key under `optionalFeatures` - "Unsupported device" UI in standard mode that utilizes the SOFA feed - Set the `attemptToCheckForSupportedDevice` key `false` under `optionalFeatures` to disable this feature - There are now keys to set all of text fields diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index f25f280c..b3ecd703 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -131,6 +131,12 @@ struct OptionalFeatureVariables { [String]() } + static var customSOFAFeedURL: String { + optionalFeaturesProfile?["customSOFAFeedURL"] as? String ?? + optionalFeaturesJSON?.customSOFAFeedURL ?? + "https://sofa.macadmins.io/v1/macos_data_feed.json" + } + static var disableSoftwareUpdateWorkflow: Bool { optionalFeaturesProfile?["disableSoftwareUpdateWorkflow"] as? Bool ?? optionalFeaturesJSON?.disableSoftwareUpdateWorkflow ?? diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index 2cc64cab..b7d91243 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -53,6 +53,7 @@ struct OptionalFeatures: Codable { var acceptableApplicationBundleIDs, acceptableAssertionApplicationNames: [String]? var acceptableAssertionUsage, acceptableCameraUsage, acceptableScreenSharingUsage, aggressiveUserExperience, aggressiveUserFullScreenExperience, asynchronousSoftwareUpdate, attemptToBlockApplicationLaunches, attemptToCheckForSupportedDevice, attemptToFetchMajorUpgrade: Bool? var blockedApplicationBundleIDs: [String]? + var customSOFAFeedURL: String? var disableNudgeForStandardInstalls, disableSoftwareUpdateWorkflow, enforceMinorUpdates: Bool? var refreshSOFAFeedTime: Int? var terminateApplicationsOnLaunch, utilizeSOFAFeed: Bool? @@ -89,6 +90,7 @@ extension OptionalFeatures { attemptToCheckForSupportedDevice: Bool? = nil, attemptToFetchMajorUpgrade: Bool? = nil, blockedApplicationBundleIDs: [String]? = nil, + customSOFAFeedURL: String? = nil, disableNudgeForStandardInstalls: Bool? = nil, disableSoftwareUpdateWorkflow: Bool? = nil, enforceMinorUpdates: Bool? = nil, @@ -109,6 +111,7 @@ extension OptionalFeatures { attemptToCheckForSupportedDevice: attemptToCheckForSupportedDevice ?? self.attemptToCheckForSupportedDevice, attemptToFetchMajorUpgrade: attemptToFetchMajorUpgrade ?? self.attemptToFetchMajorUpgrade, blockedApplicationBundleIDs: blockedApplicationBundleIDs ?? self.blockedApplicationBundleIDs, + customSOFAFeedURL: customSOFAFeedURL ?? self.customSOFAFeedURL, disableNudgeForStandardInstalls: disableNudgeForStandardInstalls ?? self.disableNudgeForStandardInstalls, disableSoftwareUpdateWorkflow: disableSoftwareUpdateWorkflow ?? self.disableSoftwareUpdateWorkflow, enforceMinorUpdates: enforceMinorUpdates ?? self.enforceMinorUpdates, diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index b0898472..e2a585aa 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -177,7 +177,6 @@ class AppDelegate: NSObject, NSApplicationDelegate { // TODO: Add more logging to "unsupported devices" UI. // TODO: Add localization for "unsupported devices" text fields // TODO: Get someone to update JAMF JSON schema for all the new keys and wiki - // TODO: add support for custom sofa feed url // TODO: check the sofa json etag even after the timelimit age to reduce more bandwidth if OptionalFeatureVariables.utilizeSOFAFeed { var selectedOS: OSInformation? diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index f5b32883..4f600404 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -890,7 +890,7 @@ struct NetworkFileManager { } } - if let url = URL(string: "https://sofa.macadmins.io/v1/macos_data_feed.json") { + if let url = URL(string: OptionalFeatureVariables.customSOFAFeedURL) { let sofaData = SOFA().URLSync(url: url) if (sofaData.error == nil) { do { From c5bf7cc43e7597c8143b75f1f7f6c578ad1d788d Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 24 May 2024 12:02:58 -0500 Subject: [PATCH 058/141] add etag logic even though it doesn't work in syncronous calls --- Nudge/3rd Party Assets/sofa.swift | 35 +++++++++++++++++++-------- Nudge/Utilities/Utils.swift | 39 ++++++++++++++++++++++++------- 2 files changed, 55 insertions(+), 19 deletions(-) diff --git a/Nudge/3rd Party Assets/sofa.swift b/Nudge/3rd Party Assets/sofa.swift index 31a3deac..4e00a315 100644 --- a/Nudge/3rd Party Assets/sofa.swift +++ b/Nudge/3rd Party Assets/sofa.swift @@ -222,39 +222,54 @@ extension MacOSDataFeed { } class SOFA: NSObject, URLSessionDelegate { - func URLSync(url: URL, maxRetries: Int = 3) -> (data: Data?, response: URLResponse?, error: Error?) { + func URLSync(url: URL, maxRetries: Int = 3) -> (data: Data?, response: URLResponse?, error: Error?, responseCode: Int?) { let semaphore = DispatchSemaphore(value: 0) + let lastEtag = Globals.nudgeDefaults.string(forKey: "LastEtag") ?? "" var request = URLRequest(url: url) - request.setValue("\(Globals.bundleID)/\(VersionManager.getNudgeVersion())", forHTTPHeaderField: "User-Agent") let session = URLSession(configuration: .default, delegate: self, delegateQueue: nil) + request.addValue("\(Globals.bundleID)/\(VersionManager.getNudgeVersion())", forHTTPHeaderField: "User-Agent") + request.setValue(lastEtag, forHTTPHeaderField: "If-None-Match") var attempts = 0 var responseData: Data? var response: URLResponse? var responseError: Error? + var responseCode: Int? // Retry loop while attempts < maxRetries { attempts += 1 let task = session.dataTask(with: request) { data, resp, error in + guard let httpResponse = resp as? HTTPURLResponse else { + print("Error receiving response: \(error?.localizedDescription ?? "No error information")") + semaphore.signal() + return + } + + responseCode = httpResponse.statusCode + if responseCode == 200 { + if let etag = httpResponse.allHeaderFields["Etag"] as? String { + Globals.nudgeDefaults.set(etag, forKey: "LastEtag") + } + } + responseData = data response = resp responseError = error semaphore.signal() } task.resume() - semaphore.wait() - // Break the loop if the task succeeded or return an error other than a timeout - if responseError == nil || (responseError! as NSError).code != NSURLErrorTimedOut { - break - } else if attempts < maxRetries { - // Reset the error to try again - responseError = nil + // Check if we should retry the request + if let error = responseError as NSError? { + if error.code == NSURLErrorTimedOut && attempts < maxRetries { + continue // Retry only if it's a timeout error + } + break // Break for all other errors or no errors } } - return (responseData, response, responseError) + return (responseData, response, responseError, responseCode) } } diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 4f600404..6f0f3c1d 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -110,6 +110,9 @@ struct AppStateManager { func getCreationDateForPath(_ path: String, testFileDate: Date?) -> Date? { let attributes = try? FileManager.default.attributesOfItem(atPath: path) + if attributes?[.size] as? Int == 0 { + return DateManager().coerceStringToDate(dateString: "2020-08-06T00:00:00Z") + } let creationDate = attributes?[.creationDate] as? Date return testFileDate ?? creationDate } @@ -865,7 +868,8 @@ struct NetworkFileManager { let appDirectory = appSupportDirectory.appendingPathComponent(Globals.bundleID) let sofaFile = "sofa-macos_data_feed.json" let sofaPath = appDirectory.appendingPathComponent(sofaFile) - if fileManager.fileExists(atPath: sofaPath.path) { + let sofaJSONExists = fileManager.fileExists(atPath: sofaPath.path) + if sofaJSONExists { let sofaPathCreationDate = AppStateManager().getCreationDateForPath(sofaPath.path, testFileDate: nil) // Use Cache as it is within time inverval if TimeInterval(OptionalFeatureVariables.refreshSOFAFeedTime) >= Date().timeIntervalSince(sofaPathCreationDate!) { @@ -877,7 +881,10 @@ struct NetworkFileManager { } catch { LogManager.error("Failed to decode local sofa JSON: \(error.localizedDescription)", logger: sofaLog) LogManager.error("Failed to decode sofa JSON: \(error)", logger: sofaLog) + return nil } + } else { + LogManager.info("Previously cached SOFA json has expired", logger: sofaLog) } } else { // Ensure the Application Support directory exists @@ -893,15 +900,29 @@ struct NetworkFileManager { if let url = URL(string: OptionalFeatureVariables.customSOFAFeedURL) { let sofaData = SOFA().URLSync(url: url) if (sofaData.error == nil) { - do { - if fileManager.fileExists(atPath: appDirectory.path) { - try sofaData.data!.write(to: sofaPath) + if sofaData.responseCode == 304 && sofaJSONExists { + LogManager.info("Utilizing previously cached SOFA json due to Etag not changing", logger: sofaLog) + do { + let sofaData = try Data(contentsOf: sofaPath) + let assetInfo = try MacOSDataFeed(data: sofaData) + return assetInfo + } catch { + LogManager.error("Failed to decode local sofa JSON: \(error.localizedDescription)", logger: sofaLog) + LogManager.error("Failed to decode sofa JSON: \(error)", logger: sofaLog) + return nil + } + } else { + do { + if fileManager.fileExists(atPath: appDirectory.path) { + try sofaData.data!.write(to: sofaPath) + } + let assetInfo = try MacOSDataFeed(data: sofaData.data!) + return assetInfo + } catch { + LogManager.error("Failed to decode sofa JSON: \(error.localizedDescription)", logger: sofaLog) + LogManager.error("Failed to decode sofa JSON: \(error)", logger: sofaLog) + return nil } - let assetInfo = try MacOSDataFeed(data: sofaData.data!) - return assetInfo - } catch { - LogManager.error("Failed to decode sofa JSON: \(error.localizedDescription)", logger: sofaLog) - LogManager.error("Failed to decode sofa JSON: \(error)", logger: sofaLog) } } else { LogManager.error("Failed to fetch sofa JSON: \(sofaData.error!.localizedDescription)", logger: sofaLog) From 8a355358a084f94af1bd9bab90065401b16dc61d Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 24 May 2024 12:10:10 -0500 Subject: [PATCH 059/141] move to modified date not creation date --- Nudge/Utilities/Utils.swift | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 6f0f3c1d..13b7561d 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -117,6 +117,15 @@ struct AppStateManager { return testFileDate ?? creationDate } + func getModifiedDateForPath(_ path: String, testFileDate: Date?) -> Date? { + let attributes = try? FileManager.default.attributesOfItem(atPath: path) + if attributes?[.size] as? Int == 0 { + return DateManager().coerceStringToDate(dateString: "2020-08-06T00:00:00Z") + } + let creationDate = attributes?[.modificationDate] as? Date + return testFileDate ?? creationDate + } + // Adapted from https://github.com/ProfileCreator/ProfileCreator/blob/master/ProfileCreator/ProfileCreator/Extensions/ExtensionBundle.swift func getSigningInfo() -> String? { var osStatus = noErr @@ -870,7 +879,7 @@ struct NetworkFileManager { let sofaPath = appDirectory.appendingPathComponent(sofaFile) let sofaJSONExists = fileManager.fileExists(atPath: sofaPath.path) if sofaJSONExists { - let sofaPathCreationDate = AppStateManager().getCreationDateForPath(sofaPath.path, testFileDate: nil) + let sofaPathCreationDate = AppStateManager().getModifiedDateForPath(sofaPath.path, testFileDate: nil) // Use Cache as it is within time inverval if TimeInterval(OptionalFeatureVariables.refreshSOFAFeedTime) >= Date().timeIntervalSince(sofaPathCreationDate!) { LogManager.info("Utilizing previously cached SOFA json", logger: sofaLog) From 2a32c1243054cdb28cc99917e7138406fc9295a2 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 24 May 2024 12:16:06 -0500 Subject: [PATCH 060/141] break the loop after one query --- Nudge/3rd Party Assets/sofa.swift | 12 +++++++++++- 1 file changed, 11 insertions(+), 1 deletion(-) diff --git a/Nudge/3rd Party Assets/sofa.swift b/Nudge/3rd Party Assets/sofa.swift index 4e00a315..22f82afc 100644 --- a/Nudge/3rd Party Assets/sofa.swift +++ b/Nudge/3rd Party Assets/sofa.swift @@ -226,7 +226,9 @@ class SOFA: NSObject, URLSessionDelegate { let semaphore = DispatchSemaphore(value: 0) let lastEtag = Globals.nudgeDefaults.string(forKey: "LastEtag") ?? "" var request = URLRequest(url: url) - let session = URLSession(configuration: .default, delegate: self, delegateQueue: nil) + let config = URLSessionConfiguration.default + config.requestCachePolicy = .useProtocolCachePolicy + let session = URLSession(configuration: config, delegate: self, delegateQueue: nil) request.addValue("\(Globals.bundleID)/\(VersionManager.getNudgeVersion())", forHTTPHeaderField: "User-Agent") request.setValue(lastEtag, forHTTPHeaderField: "If-None-Match") var attempts = 0 @@ -235,6 +237,7 @@ class SOFA: NSObject, URLSessionDelegate { var response: URLResponse? var responseError: Error? var responseCode: Int? + var successfulQuery = false // Retry loop while attempts < maxRetries { @@ -251,6 +254,9 @@ class SOFA: NSObject, URLSessionDelegate { if let etag = httpResponse.allHeaderFields["Etag"] as? String { Globals.nudgeDefaults.set(etag, forKey: "LastEtag") } + successfulQuery = true + } else if responseCode == 304 { + successfulQuery = true } responseData = data @@ -261,6 +267,10 @@ class SOFA: NSObject, URLSessionDelegate { task.resume() semaphore.wait() + if successfulQuery { + break + } + // Check if we should retry the request if let error = responseError as NSError? { if error.code == NSURLErrorTimedOut && attempts < maxRetries { From b536468a2c0ed351e18a55f42cb51b7ef9cfbedf Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 24 May 2024 12:31:32 -0500 Subject: [PATCH 061/141] move sofa logic to a function so it can be called after other logc --- Nudge/UI/Main.swift | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index e2a585aa..2becefb7 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -171,9 +171,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { // print("applicationWillBecomeActive") } - // Pre-Launch Logic - func applicationWillFinishLaunching(_ notification: Notification) { - // print("applicationWillFinishLaunching") + func sofaPreLaunchLogic() { // TODO: Add more logging to "unsupported devices" UI. // TODO: Add localization for "unsupported devices" text fields // TODO: Get someone to update JAMF JSON schema for all the new keys and wiki @@ -255,11 +253,17 @@ class AppDelegate: NSObject, NSApplicationDelegate { LogManager.error("Could not fetch SOFA feed", logger: sofaLog) } } + } + + // Pre-Launch Logic + func applicationWillFinishLaunching(_ notification: Notification) { + // print("applicationWillFinishLaunching") handleSMAppService() checkForBadProfilePath() handleCommandLineArguments() applyGracePeriodLogic() applyRandomDelayIfNecessary() + sofaPreLaunchLogic() updateNudgeState() handleSoftwareUpdateRequirements() } From 34cd108700ca34b3c62a18b40be33b33ac03735e Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 24 May 2024 13:14:29 -0500 Subject: [PATCH 062/141] fix probable issue with xcode tests failing --- Nudge/Utilities/Utils.swift | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 13b7561d..92833ec2 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -110,7 +110,7 @@ struct AppStateManager { func getCreationDateForPath(_ path: String, testFileDate: Date?) -> Date? { let attributes = try? FileManager.default.attributesOfItem(atPath: path) - if attributes?[.size] as? Int == 0 { + if attributes?[.size] as? Int == 0 && testFileDate == nil { return DateManager().coerceStringToDate(dateString: "2020-08-06T00:00:00Z") } let creationDate = attributes?[.creationDate] as? Date @@ -119,7 +119,7 @@ struct AppStateManager { func getModifiedDateForPath(_ path: String, testFileDate: Date?) -> Date? { let attributes = try? FileManager.default.attributesOfItem(atPath: path) - if attributes?[.size] as? Int == 0 { + if attributes?[.size] as? Int == 0 && testFileDate == nil { return DateManager().coerceStringToDate(dateString: "2020-08-06T00:00:00Z") } let creationDate = attributes?[.modificationDate] as? Date From 585712feeb19c03cd2f8fa78ca25af1ccad912a8 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 24 May 2024 13:14:53 -0500 Subject: [PATCH 063/141] move todos --- Nudge/3rd Party Assets/sofa.swift | 1 + Nudge/UI/Main.swift | 1 - 2 files changed, 1 insertion(+), 1 deletion(-) diff --git a/Nudge/3rd Party Assets/sofa.swift b/Nudge/3rd Party Assets/sofa.swift index 22f82afc..cee37fde 100644 --- a/Nudge/3rd Party Assets/sofa.swift +++ b/Nudge/3rd Party Assets/sofa.swift @@ -231,6 +231,7 @@ class SOFA: NSObject, URLSessionDelegate { let session = URLSession(configuration: config, delegate: self, delegateQueue: nil) request.addValue("\(Globals.bundleID)/\(VersionManager.getNudgeVersion())", forHTTPHeaderField: "User-Agent") request.setValue(lastEtag, forHTTPHeaderField: "If-None-Match") + // TODO: I'm saving the Etag and sending it, but due to forcing this into a syncronous call, it is always returning a 200 code. When using this in an asycronous method, it eventually returns the 304 response. I'm not sure how to fix this bug. var attempts = 0 var responseData: Data? diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 2becefb7..aaba5f0e 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -175,7 +175,6 @@ class AppDelegate: NSObject, NSApplicationDelegate { // TODO: Add more logging to "unsupported devices" UI. // TODO: Add localization for "unsupported devices" text fields // TODO: Get someone to update JAMF JSON schema for all the new keys and wiki - // TODO: check the sofa json etag even after the timelimit age to reduce more bandwidth if OptionalFeatureVariables.utilizeSOFAFeed { var selectedOS: OSInformation? var foundMatch = false From 0ba119adde85102fff29d64b237070782f7696e9 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 24 May 2024 13:48:38 -0500 Subject: [PATCH 064/141] add logic for sofaPeriodLaunchDelay --- CHANGELOG.md | 1 + Nudge/Preferences/DefaultPreferencesNudge.swift | 7 +++++++ Nudge/Preferences/PreferencesStructure.swift | 7 +++++-- Nudge/UI/Main.swift | 11 ++++++++++- Nudge/Utilities/Utils.swift | 14 ++++++++++++++ 5 files changed, 37 insertions(+), 3 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 7e25cd83..4e559d35 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -18,6 +18,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - `nonActivelyExploitedCVEsSLA` under the `osVersionRequirement` key will default to 21 days - `standardInstallationSLA` under the `osVersionRequirement` key will default to 28 days - These dates are calculated against the `ReleaseDate` key in the SOFA feed + - To artificially change the `ReleaseDate` thereby giving your users a default grace period for all SOFA OS updates, please configure the `sofaPeriodLaunchDelay` key under `userExperience` - If you'd like to not have nudge events for releases without any known CVEs, please configure the `disableNudgeForStandardInstalls` key under `optionalFeatures` to true ### Fixed diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index b3ecd703..79fc3554 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -178,6 +178,7 @@ struct OptionalFeatureVariables { var majorUpgradeAppPathExists = FileManager.default.fileExists(atPath: OSVersionRequirementVariables.majorUpgradeAppPath) var majorUpgradeBackupAppPathExists = FileManager.default.fileExists(atPath: NetworkFileManager().getBackupMajorUpgradeAppPath()) var requiredInstallationDate = DateManager().getFormattedDate(date: PrefsWrapper.requiredInstallationDate) +var releaseDate = Date() struct OSVersionRequirementVariables { static var osVersionRequirementsProfile: OSVersionRequirement? = getOSVersionRequirementsProfile() static var osVersionRequirementsJSON: OSVersionRequirement? = getOSVersionRequirementsJSON() @@ -362,6 +363,12 @@ struct UserExperienceVariables { userExperienceJSON?.randomDelay ?? false } + + static var sofaPeriodLaunchDelay: Int { + userExperienceProfile?["sofaPeriodLaunchDelay"] as? Int ?? + userExperienceJSON?.sofaPeriodLaunchDelay ?? + 0 + } } // User Interface diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index b7d91243..5dffecb1 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -324,6 +324,7 @@ struct UserExperience: Codable { var noTimers: Bool? var nudgeRefreshCycle: Int? var randomDelay: Bool? + var sofaPeriodLaunchDelay: Int? } // MARK: UserExperience convenience initializers and mutators @@ -366,7 +367,8 @@ extension UserExperience { maxRandomDelayInSeconds: Int? = nil, noTimers: Bool? = nil, nudgeRefreshCycle: Int? = nil, - randomDelay: Bool? = nil + randomDelay: Bool? = nil, + sofaPeriodLaunchDelay: Int? = nil ) -> UserExperience { return UserExperience( allowGracePeriods: allowGracePeriods ?? self.allowGracePeriods, @@ -390,7 +392,8 @@ extension UserExperience { maxRandomDelayInSeconds: maxRandomDelayInSeconds ?? self.maxRandomDelayInSeconds, noTimers: noTimers ?? self.noTimers, nudgeRefreshCycle: nudgeRefreshCycle ?? self.nudgeRefreshCycle, - randomDelay: randomDelay ?? self.randomDelay + randomDelay: randomDelay ?? self.randomDelay, + sofaPeriodLaunchDelay: sofaPeriodLaunchDelay ?? self.sofaPeriodLaunchDelay ) } } diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index aaba5f0e..aa988820 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -228,6 +228,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { // Start setting UI fields nudgePrimaryState.requiredMinimumOSVersion = osVersion.latest.productVersion nudgePrimaryState.activelyExploitedCVEs = activelyExploitedCVEs + releaseDate = selectedOS!.releaseDate ?? Date() requiredInstallationDate = selectedOS!.releaseDate?.addingTimeInterval(slaExtension) ?? DateManager().getCurrentDate().addingTimeInterval(TimeInterval(OSVersionRequirementVariables.standardInstallationSLA * 86400)) LogManager.notice("Extending requiredInstallationDate to \(requiredInstallationDate)", logger: sofaLog) @@ -261,8 +262,9 @@ class AppDelegate: NSObject, NSApplicationDelegate { checkForBadProfilePath() handleCommandLineArguments() applyGracePeriodLogic() - applyRandomDelayIfNecessary() sofaPreLaunchLogic() + applySOFAPeriodLogic() + applyRandomDelayIfNecessary() updateNudgeState() handleSoftwareUpdateRequirements() } @@ -337,6 +339,13 @@ class AppDelegate: NSObject, NSApplicationDelegate { } } + private func applySOFAPeriodLogic() { + _ = AppStateManager().sofaPeriodLogic() + if nudgePrimaryState.shouldExit { + exit(0) + } + } + private func checkForBadProfilePath() { let badProfilePath = "/Library/Managed Preferences/com.github.macadmins.Nudge.json.plist" if FileManager.default.fileExists(atPath: badProfilePath) { diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 92833ec2..04820447 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -176,6 +176,20 @@ struct AppStateManager { return calculateNewRequiredInstallationDateIfNeeded(currentDate: currentDate, gracePeriodPathCreationDate: gracePeriodPathCreationDate) } + func sofaPeriodLogic(currentDate: Date = DateManager().getCurrentDate(), testFileDate: Date? = nil) -> Date { + if OptionalFeatureVariables.utilizeSOFAFeed { + if releaseDate.addingTimeInterval(TimeInterval(UserExperienceVariables.sofaPeriodLaunchDelay * 86400)) > currentDate { + LogManager.info("Device within sofaPeriodLaunchDelay, exiting Nudge", logger: uiLog) + nudgePrimaryState.shouldExit = true + return currentDate + } else { + LogManager.info("Device outside sofaPeriodLaunchDelay", logger: uiLog) + return PrefsWrapper.requiredInstallationDate + } + } + return PrefsWrapper.requiredInstallationDate + } + private func isDeferralAllowed(threshold: Int, logMessage: String) -> Bool { if CommandLineUtilities().demoModeEnabled() { return true From 34fd8862cb983f121c36d78b4833bfcb0e1ceda3 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 24 May 2024 16:10:26 -0500 Subject: [PATCH 065/141] Update changelog --- CHANGELOG.md | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4e559d35..26022eb2 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -8,6 +8,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 Requires macOS 12.0 and higher. Further releases and feature requests may make this macOS 13 and higher depending on code complexity. ### Changed +- Now built on Swift 5.10, Xcode 15.4 and macOS 14 +- New Xcode Scheme `-bundle-mode-profile` to test profile logic + - `-bundle-mode` has been renamed to `-bundle-mode-json` - You can now pass the strings `latest`, `latest-supported` and `latest-minor` in the `requiredMinimumOSVersion` key - `latest`: always force latest release and if the machine can't this version, show the new "unsupported device" user interface - `latest-supported`: always get the latest version sofa shows that is supported by this device @@ -22,10 +25,15 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - If you'd like to not have nudge events for releases without any known CVEs, please configure the `disableNudgeForStandardInstalls` key under `optionalFeatures` to true ### Fixed -Upcoming +- `screenshotDisplay` view had a bug that may result in the screenshot being partially cut off or zoomable +- More descriptive logs when loading json/mdm profile keys +- Refactor portions of the `softwareupdate` logic to reduce potential errors +- Fixed errors when moving to Swift 5.10 ### Added - Remote URLs can now be used on `iconDarkPath`, `iconLightPath`, `screenShotDarkPath` and `screenShotLightPath` + - Please note that these files will be downloaded each time Nudge is ran and there is currently not a way to cache these objects. + - If these files fail to download, a default company logo will be shown. - Actively Exploited CVEs in the left sidebar - To disable this item, please configure the `showActivelyExploitedCVEs` key under `userInterface` to false - An admin can now allow users to move the Nudge window with `userExperience` key `allowMovableWindow` From eb0088ea81d1f86dec93ec62bfdae5e5472b304e Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 24 May 2024 16:22:21 -0500 Subject: [PATCH 066/141] -disable-randomDelay --- CHANGELOG.md | 1 + Nudge/UI/Main.swift | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 26022eb2..2ab40c4d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -37,6 +37,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - Actively Exploited CVEs in the left sidebar - To disable this item, please configure the `showActivelyExploitedCVEs` key under `userInterface` to false - An admin can now allow users to move the Nudge window with `userExperience` key `allowMovableWindow` +- To ease testing, you can now pass `-disable-randomDelay` as an argument to ignore the `randomDelay` key if it is set by a JSON or mobileconfig - Basic SwiftUI support for Markdown text options - Utilizing Apple's markdown features, you can now utilize, bold, italic, underline, subscript and url links directly into any of the text fields - [SOFA](https://github.com/macadmins/sofa) feed support diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index aa988820..27ee53a6 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -332,7 +332,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { } private func applyRandomDelayIfNecessary() { - if UserExperienceVariables.randomDelay { + if UserExperienceVariables.randomDelay && !CommandLine.arguments.contains("-disable-randomDelay") { let delaySeconds = Int.random(in: 1...UserExperienceVariables.maxRandomDelayInSeconds) LogManager.notice("Delaying initial run (in seconds) by: \(delaySeconds)", logger: uiLog) sleep(UInt32(delaySeconds)) From c1aaec229ac662a3c2865b9ad84196510de06ce0 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 24 May 2024 16:42:26 -0500 Subject: [PATCH 067/141] activate nudge if blocked application is terminated --- Nudge/UI/Main.swift | 9 +++++++-- 1 file changed, 7 insertions(+), 2 deletions(-) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 27ee53a6..5442052f 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -321,7 +321,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { @objc func terminateApplicationSender(_ notification: Notification) { LogManager.info("Application launched - checking if application should be terminated", logger: utilsLog) - terminateApplications() + terminateApplications(afterInitialLaunch: true) } private func applyGracePeriodLogic() { @@ -611,10 +611,11 @@ class AppDelegate: NSObject, NSApplicationDelegate { LogManager.info("Successfully terminated application: \(application.bundleIdentifier ?? "")", logger: utilsLog) } - private func terminateApplications() { + private func terminateApplications(afterInitialLaunch: Bool = false) { guard DateManager().pastRequiredInstallationDate() else { return } + var hasTerminatedAnApplication = false let runningApplications = NSWorkspace.shared.runningApplications for runningApplication in runningApplications { let appBundleID = runningApplication.bundleIdentifier ?? "" @@ -625,8 +626,12 @@ class AppDelegate: NSObject, NSApplicationDelegate { LogManager.info("Found \(appBundleID), terminating application", logger: utilsLog) scheduleLocal(applicationIdentifier: appBundleID) terminateApplication(runningApplication) + hasTerminatedAnApplication = true } } + if hasTerminatedAnApplication && afterInitialLaunch { + AppStateManager().activateNudge() + } } func runSoftwareUpdate() { From 4f3d71367eade1e6589209731b4d2230b62674d2 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 24 May 2024 17:57:34 -0500 Subject: [PATCH 068/141] Update CHANGELOG.md --- CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 2ab40c4d..05f134e9 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -47,7 +47,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - If you are utilizing a custom sofa feed, please configure the `customSOFAFeedURL` key under `optionalFeatures` - "Unsupported device" UI in standard mode that utilizes the SOFA feed - Set the `attemptToCheckForSupportedDevice` key `false` under `optionalFeatures` to disable this feature - - There are now keys to set all of text fields + - There are new keys to set all of text fields - `informationButtonTextUnsupported`, `mainContentHeaderUnsupported`, `mainContentNoteUnsupported`, `mainContentSubHeaderUnsupported`, `mainContentTextUnsupported`, `subHeaderUnsupported` - `unsupportedURL` and `unsupportedURLs` can change the information button itself, but it will remain in the `osVersionRequirement` key with `unsupportedURLs` and `unsupportedURLs`. - An icon will appear as an overlay on top of the company image to further emphasize the device is no longer supported From b3bce47d1940913b91c5237732741ac3c9284023 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 28 May 2024 09:41:25 -0500 Subject: [PATCH 069/141] add all of the localizations for english --- Localizable.xcstrings | 593 ++++++++++-------- .../Preferences/DefaultPreferencesNudge.swift | 2 +- 2 files changed, 330 insertions(+), 265 deletions(-) diff --git a/Localizable.xcstrings b/Localizable.xcstrings index 798d0ac7..a9b6fef8 100644 --- a/Localizable.xcstrings +++ b/Localizable.xcstrings @@ -4,222 +4,464 @@ "" : { }, - "A friendly reminder from your local IT team" : { + "**A friendly reminder from your local IT team**" : { + "comment" : "subHeader / subHeaderUnsupported", "extractionState" : "manual", "localizations" : { "da" : { "stringUnit" : { "state" : "translated", - "value" : "En venligt ment påmindelse fra din lokale IT afdeling" + "value" : "**En venligt ment påmindelse fra din lokale IT afdeling**" } }, "de" : { "stringUnit" : { "state" : "translated", - "value" : "Eine freundliche Erinnerung deines IT-Teams" + "value" : "**Eine freundliche Erinnerung deines IT-Teams**" } }, "en" : { "stringUnit" : { "state" : "translated", - "value" : "A friendly reminder from your local IT team" + "value" : "**A friendly reminder from your local IT team**" } }, "es" : { "stringUnit" : { "state" : "translated", - "value" : "Un recordatorio amistoso de tu equipo IT local" + "value" : "**Un recordatorio amistoso de tu equipo IT local**" } }, "fr" : { "stringUnit" : { "state" : "translated", - "value" : "Un rappel amical de votre équipe informatique" + "value" : "**Un rappel amical de votre équipe informatique**" } }, "hi" : { "stringUnit" : { "state" : "translated", - "value" : "आपके स्थानीय IT से मित्रवत अनुस्मारक" + "value" : "**आपके स्थानीय IT से मित्रवत अनुस्मारक**" } }, "it" : { "stringUnit" : { "state" : "translated", - "value" : "Un cortese promemoria dal tuo team IT locale" + "value" : "**Un cortese promemoria dal tuo team IT locale**" } }, "ja" : { "stringUnit" : { "state" : "translated", - "value" : "ローカルITチームからのリマインドです" + "value" : "**ローカルITチームからのリマインドです**" } }, "ko" : { "stringUnit" : { "state" : "translated", - "value" : "로컬 IT 팀에서 알려드립니다" + "value" : "**로컬 IT 팀에서 알려드립니다**" } }, "nb" : { "stringUnit" : { "state" : "translated", - "value" : "En påminnelse fra IT" + "value" : "**En påminnelse fra IT**" } }, "nl" : { "stringUnit" : { "state" : "translated", - "value" : "Een herinnering van uw IT-team" + "value" : "**Een herinnering van uw IT-team**" } }, "pl" : { "stringUnit" : { "state" : "translated", - "value" : "Przypomnienie IT" + "value" : "**Przypomnienie IT**" } }, "pt" : { "stringUnit" : { "state" : "translated", - "value" : "Um lembrete rápido da sua equipe de TI local" + "value" : "**Um lembrete rápido da sua equipe de TI local**" } }, "ru" : { "stringUnit" : { "state" : "translated", - "value" : "Напоминание от команды IT" + "value" : "**Напоминание от команды IT**" } }, "sv" : { "stringUnit" : { "state" : "translated", - "value" : "En påminnelse från IT" + "value" : "**En påminnelse från IT**" } }, "uk" : { "stringUnit" : { "state" : "translated", - "value" : "Дружнє нагадування від вашої IT-команди" + "value" : "**Дружнє нагадування від вашої IT-команди**" } }, "zh" : { "stringUnit" : { "state" : "translated", - "value" : "IT团队的友好提醒" + "value" : "**IT团队的友好提醒**" } } } }, - "A fully up-to-date device is required to ensure that IT can accurately protect your device.\n\nIf you do not update your device, you may lose access to some items necessary for your day-to-day tasks.\n\nTo begin the update, simply click on the Update Device button and follow the provided steps." : { + "**Important Notes**" : { + "comment" : "mainContentNote / mainContentNoteUnsupported", "extractionState" : "manual", "localizations" : { "da" : { "stringUnit" : { "state" : "translated", - "value" : "En fuldt opdateret Mac er påkrævet for at IT afdelingen kan beskytte dine data.\n\nHvis du ikke opdaterer din Mac, kan du risikere at miste adgang til muligheder, som er nødvendige i dagligdagen.\n\nFor at påbegynde opdateringen skal du klikke på \"Opdater Mac\"-knappen og følge instruktionerne på skærmen." + "value" : "**Vigtig information**" } }, "de" : { "stringUnit" : { "state" : "translated", - "value" : "Ein vollständig aktualisiertes Gerät ist erforderlich, um sicherzustellen, dass die IT-Abteilung dein Gerät effektiv schützen kann.\n\nWenn du dein Gerät nicht aktualisierst, verlierst du möglicherweise den Zugriff auf einige Werkzeuge, die du für deine täglichen Aufgaben benötigst.\n\nUm das Update zu starten, klicke auf die Schaltfläche Gerät Aktualisieren und befolge die angegebenen Schritte." + "value" : "**Wichtige Hinweise**" } }, "en" : { "stringUnit" : { "state" : "translated", - "value" : "A fully up-to-date device is required to ensure that IT can accurately protect your device.\n\nIf you do not update your device, you may lose access to some items necessary for your day-to-day tasks.\n\nTo begin the update, simply click on the Update Device button and follow the provided steps." + "value" : "**Important Notes**" } }, "es" : { "stringUnit" : { "state" : "translated", - "value" : "Se requiere un dispositivo completamente actualizado para garantizar que IT pueda proteger su dispositivo con precisión.\n\nSi no actualiza su dispositivo, es posible que pierda el acceso a algunos elementos necesarios para sus tareas diarias.\n\nPara comenzar la actualización, simplemente haga clic en el botón Actualizar dispositivo y siga los pasos proporcionados." + "value" : "**Notas importantes**" } }, "fr" : { "stringUnit" : { "state" : "translated", - "value" : "Un système entièrement à jour est nécessaire pour garantir que le service informatique puisse protéger votre appareil efficacement.\n\n Si vous ne mettez pas à jour votre appareil, vous risquez de perdre l'accès à certains outils nécessaires à vos tâches quotidiennes.\n\nPour commencer la mise à jour, cliquez simplement sur le bouton Mettre à jour l'appareil et suivez les étapes fournies." + "value" : "**Informations importantes**" } }, "hi" : { "stringUnit" : { "state" : "translated", - "value" : "पुरी तरह अपडेटेड उपकरण आवश्यक है IT द्वारा आपके उपकरण को सुरक्षित रखने के लिए \n\n यदि उपकरण को अपडेट नही किया तो आप अपने कुछ मूल्यवान नित्य उपयोगी वस्तुएँ खो सकते हैं जो दैन दिन उपयोग के लिए आवश्यक होते है \n\n अपडेट करने के लिए केवल अपडेट बटन दबाये और निर्देशों का पालन करें।" + "value" : "**आवश्यक जानकारी**" } }, "it" : { "stringUnit" : { "state" : "translated", - "value" : "È necessario un dispositivo completamente aggiornato per garantire che l'IT possa proteggere accuratamente il tuo dispositivo.\n\nSe non aggiorni il dispositivo, potresti perdere l'accesso ad alcuni elementi necessari per le tue attività quotidiane.\n\nPer iniziare l'aggiornamento, fai semplicemente clic sul pulsante Aggiorna dispositivo e segui i passaggi forniti." + "value" : "**Note importanti**" } }, "ja" : { "stringUnit" : { "state" : "translated", - "value" : "ITが正確にあなたのデバイスを保護するためには、デバイスを完全に最新にすることが必要です。\n\nデバイスを更新しない場合、日々のタスクに必要なアイテムへのアクセスができなくなる可能性があります。\n\n更新を開始するにはデバイスの更新ボタンをクリックし、指示されたステップに従ってください。" + "value" : "**重要**" } }, "ko" : { "stringUnit" : { "state" : "translated", - "value" : "IT 팀이 귀하의 기기를 정밀하게 보호하기 위하여 최신 업데이트가 기기에 항시 탑재될 것을 요청드립니다.\n\n귀하께서 기기를 업데이트 하지 않는다면, 일상의 업무에 필요한 아이템들의 접근 권한이 상실될 수 있습니다.\n\n기기를 업데이트 하시려면, 기기 업데이트 버튼을 클릭하시고 제공되는 안내를 따르시면 됩니다." + "value" : "**중요 사항**" } }, "nb" : { "stringUnit" : { "state" : "translated", - "value" : "Datamaskinen må være oppdatert med de siste sikkerhetsoppdateringene for å være beskyttet.\n\nHvis du ikke oppdaterer datamaskinen, kan det hende at du ikke får tilgang til de nødvendige ressursene.\n\nFor å starte oppdateringen, klikk på Oppdater nå og følg instruksene." + "value" : "**Viktig informasjon**" } }, "nl" : { "stringUnit" : { "state" : "translated", - "value" : "Een volledig up-to-date Mac is vereist om ervoor te zorgen dat uw Mac goed beschermd blijft.\n\nAls u uw Mac niet bijwerkt kunnen er restricties op de Mac komen.\n\nOm met de update te beginnen, klikt u op de knop Update Mac en volgt u de stappen." + "value" : "**Belangrijke informatie**" } }, "pl" : { "stringUnit" : { "state" : "translated", - "value" : "Wymagane jest uaktualnienie urządzenia, aby mieć pewność, że dział IT może w pełni chronić Twoje urządzenie.\n\nJeśli nie zaktualizujesz urządzenia, możesz utracić dostęp do niektórych elementów niezbędnych do wykonywania codziennych zadań.\n\nAby rozpocząć aktualizację, po prostu kliknij przycisk Uaktualnij system i postępuj zgodnie z podanymi krokami." + "value" : "**Ważne informacje**" } }, "pt" : { "stringUnit" : { "state" : "translated", - "value" : "Um dispositivo totalmente atualizado é necessário para garantir que IT possa proteger seu dispositivo com precisão.\n\nSe não atualizar seu dispositivo, é possível que você perca acesso a itens necessários para performar suas tarefas do dia-a-dia.\n\nPara começar a atualização, apenas clique no botão Atualizar Dispositivo e siga as instruções apresentadas." + "value" : "**Notas Importantes**" } }, "ru" : { "stringUnit" : { "state" : "translated", - "value" : "Обновления нужны, чтобы IT-команда могла защитить Mac и данные.\n\nЕсли вы не обновите устройство, есть риск потерять доступ к важным для работы сервисам.\n\nЧтобы начать обновление, просто нажмите кнопку установки." + "value" : "**Важные примечания**" } }, "sv" : { "stringUnit" : { "state" : "translated", - "value" : "Ett system med de senaste uppdateringarna krävs för att IT ska kunna säkerställa att din dator är skyddad.\n\nOm du inte uppdaterar systemet kommer du kanske inte kunna komma åt nödvändinga resurser.\n\nFör att påbörja uppdateringen, klicka på Uppdatera och följ stegen." + "value" : "**Viktig information**" } }, "uk" : { "stringUnit" : { "state" : "translated", - "value" : "Потрібен повністю оновлений пристрій щоб бути впевненим що IT зможе його надійно захистити.\n\nЯкщо ви не оновите пристрій, ви можете втратити доступ до деяких речей, необхідних для виконання повсякденних завдань.\n\nЩоб почати оновлення, просто клацніть кнопку Оновити пристрій та виконайте запропоновані кроки." + "value" : "**Important Notes**" } }, "zh" : { "stringUnit" : { "state" : "translated", + "value" : "**重要信息**" + } + } + } + }, + "**Your device is no longer capable of receving critical security updates**" : { + "comment" : "mainContentHeaderUnsupported", + "extractionState" : "manual", + "localizations" : { + "en" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Your device is no longer capable of receving critical security updates**" + } + } + } + }, + "**Your device will restart during this update**" : { + "comment" : "mainContentHeader", + "extractionState" : "manual", + "localizations" : { + "da" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Din Mac vil genstarte under opdateringen**" + } + }, + "de" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Dein Gerät wird während dieses Updates neu gestartet**" + } + }, + "en" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Your device will restart during this update**" + } + }, + "es" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Su dispositivo se reiniciará durante esta actualización**" + } + }, + "fr" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Votre appareil redémarrera pendant cette mise à jour**" + } + }, + "hi" : { + "stringUnit" : { + "state" : "translated", + "value" : "**आपका उपकरण इस अपडेट के समय पुनः शुरु होगा**" + } + }, + "it" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Il dispositivo si riavvierà durante l'aggiornamento**" + } + }, + "ja" : { + "stringUnit" : { + "state" : "translated", + "value" : "**デバイスは更新中に再起動します**" + } + }, + "ko" : { + "stringUnit" : { + "state" : "translated", + "value" : "**업데이트 도중에 귀하의 기기는 재시작 될 것입니다**" + } + }, + "nb" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Du vil bli spurt om å starte på nytt underveis i oppdateringen**" + } + }, + "nl" : { + "stringUnit" : { + "state" : "translated", + "value" : "**De Mac zal herstarten tijdens het updaten**" + } + }, + "pl" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Podczas tej aktualizacji Mac uruchomi się ponownie**" + } + }, + "pt" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Seu dispositivo irá reiniciar durante essa atualização**" + } + }, + "ru" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Ваше устройство перезагрузится во время обновления**" + } + }, + "sv" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Din dator kommer att startas om under uppdateringen**" + } + }, + "uk" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Ваш пристрій буде перезавантажено під час цього оновлення**" + } + }, + "zh" : { + "stringUnit" : { + "state" : "translated", + "value" : "**在更新过程中,你的Mac将会重新启动。**" + } + } + } + }, + "A fully up-to-date device is required to ensure that IT can accurately protect your device.\n\nIf you do not obtain a replacement device, you will lose access to some items necessary for your day-to-day tasks.\n\nFor more information about this, please click on the **Replace Your Device** button." : { + "comment" : "mainContentTextUnsupported", + "extractionState" : "manual", + "localizations" : { + "en" : { + "stringUnit" : { + "state" : "translated", + "value" : "A fully up-to-date device is required to ensure that IT can accurately protect your device.\n\nIf you do not obtain a replacement device, you will lose access to some items necessary for your day-to-day tasks.\n\nFor more information about this, please click on the **Replace Your Device** button." + } + } + } + }, + "A fully up-to-date device is required to ensure that IT can accurately protect your device.\n\nIf you do not update your device, you may lose access to some items necessary for your day-to-day tasks.\n\nTo begin the update, simply click on the **Update Device** button and follow the provided steps." : { + "comment" : "mainContentText", + "extractionState" : "manual", + "localizations" : { + "da" : { + "stringUnit" : { + "state" : "needs_review", + "value" : "En fuldt opdateret Mac er påkrævet for at IT afdelingen kan beskytte dine data.\n\nHvis du ikke opdaterer din Mac, kan du risikere at miste adgang til muligheder, som er nødvendige i dagligdagen.\n\nFor at påbegynde opdateringen skal du klikke på \"Opdater Mac\"-knappen og følge instruktionerne på skærmen." + } + }, + "de" : { + "stringUnit" : { + "state" : "needs_review", + "value" : "Ein vollständig aktualisiertes Gerät ist erforderlich, um sicherzustellen, dass die IT-Abteilung dein Gerät effektiv schützen kann.\n\nWenn du dein Gerät nicht aktualisierst, verlierst du möglicherweise den Zugriff auf einige Werkzeuge, die du für deine täglichen Aufgaben benötigst.\n\nUm das Update zu starten, klicke auf die Schaltfläche Gerät Aktualisieren und befolge die angegebenen Schritte." + } + }, + "en" : { + "stringUnit" : { + "state" : "translated", + "value" : "A fully up-to-date device is required to ensure that IT can accurately protect your device.\n\nIf you do not update your device, you may lose access to some items necessary for your day-to-day tasks.\n\nTo begin the update, simply click on the **Update Device** button and follow the provided steps." + } + }, + "es" : { + "stringUnit" : { + "state" : "needs_review", + "value" : "Se requiere un dispositivo completamente actualizado para garantizar que IT pueda proteger su dispositivo con precisión.\n\nSi no actualiza su dispositivo, es posible que pierda el acceso a algunos elementos necesarios para sus tareas diarias.\n\nPara comenzar la actualización, simplemente haga clic en el botón Actualizar dispositivo y siga los pasos proporcionados." + } + }, + "fr" : { + "stringUnit" : { + "state" : "needs_review", + "value" : "Un système entièrement à jour est nécessaire pour garantir que le service informatique puisse protéger votre appareil efficacement.\n\n Si vous ne mettez pas à jour votre appareil, vous risquez de perdre l'accès à certains outils nécessaires à vos tâches quotidiennes.\n\nPour commencer la mise à jour, cliquez simplement sur le bouton Mettre à jour l'appareil et suivez les étapes fournies." + } + }, + "hi" : { + "stringUnit" : { + "state" : "needs_review", + "value" : "पुरी तरह अपडेटेड उपकरण आवश्यक है IT द्वारा आपके उपकरण को सुरक्षित रखने के लिए \n\n यदि उपकरण को अपडेट नही किया तो आप अपने कुछ मूल्यवान नित्य उपयोगी वस्तुएँ खो सकते हैं जो दैन दिन उपयोग के लिए आवश्यक होते है \n\n अपडेट करने के लिए केवल अपडेट बटन दबाये और निर्देशों का पालन करें।" + } + }, + "it" : { + "stringUnit" : { + "state" : "needs_review", + "value" : "È necessario un dispositivo completamente aggiornato per garantire che l'IT possa proteggere accuratamente il tuo dispositivo.\n\nSe non aggiorni il dispositivo, potresti perdere l'accesso ad alcuni elementi necessari per le tue attività quotidiane.\n\nPer iniziare l'aggiornamento, fai semplicemente clic sul pulsante Aggiorna dispositivo e segui i passaggi forniti." + } + }, + "ja" : { + "stringUnit" : { + "state" : "needs_review", + "value" : "ITが正確にあなたのデバイスを保護するためには、デバイスを完全に最新にすることが必要です。\n\nデバイスを更新しない場合、日々のタスクに必要なアイテムへのアクセスができなくなる可能性があります。\n\n更新を開始するにはデバイスの更新ボタンをクリックし、指示されたステップに従ってください。" + } + }, + "ko" : { + "stringUnit" : { + "state" : "needs_review", + "value" : "IT 팀이 귀하의 기기를 정밀하게 보호하기 위하여 최신 업데이트가 기기에 항시 탑재될 것을 요청드립니다.\n\n귀하께서 기기를 업데이트 하지 않는다면, 일상의 업무에 필요한 아이템들의 접근 권한이 상실될 수 있습니다.\n\n기기를 업데이트 하시려면, 기기 업데이트 버튼을 클릭하시고 제공되는 안내를 따르시면 됩니다." + } + }, + "nb" : { + "stringUnit" : { + "state" : "needs_review", + "value" : "Datamaskinen må være oppdatert med de siste sikkerhetsoppdateringene for å være beskyttet.\n\nHvis du ikke oppdaterer datamaskinen, kan det hende at du ikke får tilgang til de nødvendige ressursene.\n\nFor å starte oppdateringen, klikk på Oppdater nå og følg instruksene." + } + }, + "nl" : { + "stringUnit" : { + "state" : "needs_review", + "value" : "Een volledig up-to-date Mac is vereist om ervoor te zorgen dat uw Mac goed beschermd blijft.\n\nAls u uw Mac niet bijwerkt kunnen er restricties op de Mac komen.\n\nOm met de update te beginnen, klikt u op de knop Update Mac en volgt u de stappen." + } + }, + "pl" : { + "stringUnit" : { + "state" : "needs_review", + "value" : "Wymagane jest uaktualnienie urządzenia, aby mieć pewność, że dział IT może w pełni chronić Twoje urządzenie.\n\nJeśli nie zaktualizujesz urządzenia, możesz utracić dostęp do niektórych elementów niezbędnych do wykonywania codziennych zadań.\n\nAby rozpocząć aktualizację, po prostu kliknij przycisk Uaktualnij system i postępuj zgodnie z podanymi krokami." + } + }, + "pt" : { + "stringUnit" : { + "state" : "needs_review", + "value" : "Um dispositivo totalmente atualizado é necessário para garantir que IT possa proteger seu dispositivo com precisão.\n\nSe não atualizar seu dispositivo, é possível que você perca acesso a itens necessários para performar suas tarefas do dia-a-dia.\n\nPara começar a atualização, apenas clique no botão Atualizar Dispositivo e siga as instruções apresentadas." + } + }, + "ru" : { + "stringUnit" : { + "state" : "needs_review", + "value" : "Обновления нужны, чтобы IT-команда могла защитить Mac и данные.\n\nЕсли вы не обновите устройство, есть риск потерять доступ к важным для работы сервисам.\n\nЧтобы начать обновление, просто нажмите кнопку установки." + } + }, + "sv" : { + "stringUnit" : { + "state" : "needs_review", + "value" : "Ett system med de senaste uppdateringarna krävs för att IT ska kunna säkerställa att din dator är skyddad.\n\nOm du inte uppdaterar systemet kommer du kanske inte kunna komma åt nödvändinga resurser.\n\nFör att påbörja uppdateringen, klicka på Uppdatera och följ stegen." + } + }, + "uk" : { + "stringUnit" : { + "state" : "needs_review", + "value" : "Потрібен повністю оновлений пристрій щоб бути впевненим що IT зможе його надійно захистити.\n\nЯкщо ви не оновите пристрій, ви можете втратити доступ до деяких речей, необхідних для виконання повсякденних завдань.\n\nЩоб почати оновлення, просто клацніть кнопку Оновити пристрій та виконайте запропоновані кроки." + } + }, + "zh" : { + "stringUnit" : { + "state" : "needs_review", "value" : "更新至最新的软件以确保 IT 可以精准的保护设备安全。\n\n如未及时完成更新,您将会无法访问日常工作所需的网页或程序。\n \n立刻更新,请点击【更新设备】并根据提供的步骤完成更新。" } } } }, "Additional Device Information" : { - "comment" : "// Additional Device Information", + "comment" : "Additional Device Information", "extractionState" : "manual", "localizations" : { "da" : { @@ -327,7 +569,7 @@ } }, "Application terminated" : { - "comment" : "User Notification", + "comment" : "Application terminated", "extractionState" : "manual", "localizations" : { "da" : { @@ -435,7 +677,7 @@ } }, "Architecture:" : { - "comment" : "Architecture", + "comment" : "Architecture:", "extractionState" : "manual", "localizations" : { "da" : { @@ -543,6 +785,7 @@ } }, "Click for additional device information" : { + "comment" : "Click for additional device information", "extractionState" : "manual", "localizations" : { "da" : { @@ -650,6 +893,7 @@ } }, "Click for more information about the security update" : { + "comment" : "Click for more information about the security update", "extractionState" : "manual", "localizations" : { "da" : { @@ -757,7 +1001,7 @@ } }, "Click to close" : { - "comment" : "Help buttons", + "comment" : "Click to close", "extractionState" : "manual", "localizations" : { "da" : { @@ -865,6 +1109,7 @@ } }, "Click to zoom into screenshot" : { + "comment" : "screenShotAltText", "extractionState" : "manual", "localizations" : { "da" : { @@ -972,7 +1217,7 @@ } }, "Current OS Version:" : { - "comment" : "Current OS Version", + "comment" : "Current OS Version:", "extractionState" : "manual", "localizations" : { "da" : { @@ -1080,6 +1325,7 @@ } }, "Custom" : { + "comment" : "customDeferralButtonText", "extractionState" : "manual", "localizations" : { "da" : { @@ -1187,7 +1433,7 @@ } }, "Days Remaining To Update:" : { - "comment" : "Days Remaining To Update", + "comment" : "Days Remaining To Update:", "extractionState" : "manual", "localizations" : { "da" : { @@ -1295,6 +1541,7 @@ } }, "Defer" : { + "comment" : "customDeferralDropdownText", "extractionState" : "manual", "localizations" : { "da" : { @@ -1402,7 +1649,7 @@ } }, "Deferred Count:" : { - "comment" : "Deferred Count", + "comment" : "Deferred Count:", "extractionState" : "manual", "localizations" : { "da" : { @@ -1510,7 +1757,7 @@ } }, "Hours Remaining To Update:" : { - "comment" : "Hours Remaining To Update", + "comment" : "Hours Remaining To Update:", "extractionState" : "manual", "localizations" : { "da" : { @@ -1618,6 +1865,7 @@ } }, "I understand" : { + "comment" : "secondaryQuitButtonText", "extractionState" : "manual", "localizations" : { "da" : { @@ -1724,115 +1972,8 @@ } } }, - "Important Notes" : { - "extractionState" : "manual", - "localizations" : { - "da" : { - "stringUnit" : { - "state" : "translated", - "value" : "Vigtig information" - } - }, - "de" : { - "stringUnit" : { - "state" : "translated", - "value" : "Wichtige Hinweise" - } - }, - "en" : { - "stringUnit" : { - "state" : "translated", - "value" : "Important Notes" - } - }, - "es" : { - "stringUnit" : { - "state" : "translated", - "value" : "Notas importantes" - } - }, - "fr" : { - "stringUnit" : { - "state" : "translated", - "value" : "Informations importantes" - } - }, - "hi" : { - "stringUnit" : { - "state" : "translated", - "value" : "आवश्यक जानकारी" - } - }, - "it" : { - "stringUnit" : { - "state" : "translated", - "value" : "Note importanti" - } - }, - "ja" : { - "stringUnit" : { - "state" : "translated", - "value" : "重要" - } - }, - "ko" : { - "stringUnit" : { - "state" : "translated", - "value" : "중요 사항" - } - }, - "nb" : { - "stringUnit" : { - "state" : "translated", - "value" : "Viktig informasjon" - } - }, - "nl" : { - "stringUnit" : { - "state" : "translated", - "value" : "Belangrijke informatie" - } - }, - "pl" : { - "stringUnit" : { - "state" : "translated", - "value" : "Ważne informacje" - } - }, - "pt" : { - "stringUnit" : { - "state" : "translated", - "value" : "Notas Importantes" - } - }, - "ru" : { - "stringUnit" : { - "state" : "translated", - "value" : "Важные примечания" - } - }, - "sv" : { - "stringUnit" : { - "state" : "translated", - "value" : "Viktig information" - } - }, - "uk" : { - "stringUnit" : { - "state" : "translated", - "value" : "Important Notes" - } - }, - "zh" : { - "stringUnit" : { - "state" : "translated", - "value" : "重要信息" - } - } - } - }, "Language:" : { - "comment" : "Language", + "comment" : "Language:", "extractionState" : "manual", "localizations" : { "da" : { @@ -1940,6 +2081,7 @@ } }, "Later" : { + "comment" : "primaryQuitButtonText", "extractionState" : "manual", "localizations" : { "da" : { @@ -2047,7 +2189,7 @@ } }, "More Info" : { - "comment" : "More Info", + "comment" : "informationButtonText", "extractionState" : "manual", "localizations" : { "da" : { @@ -2155,6 +2297,7 @@ } }, "One Day" : { + "comment" : "oneDayDeferralButtonText", "extractionState" : "manual", "localizations" : { "da" : { @@ -2262,6 +2405,7 @@ } }, "One Hour" : { + "comment" : "oneHourDeferralButtonText", "extractionState" : "manual", "localizations" : { "da" : { @@ -2369,6 +2513,7 @@ } }, "Please update your device to use this application" : { + "comment" : "Please update your device to use this application", "extractionState" : "manual", "localizations" : { "da" : { @@ -2475,8 +2620,32 @@ } } }, + "Please work with your local IT team to obtain a replacement device" : { + "comment" : "mainContentSubHeaderUnsupported", + "extractionState" : "manual", + "localizations" : { + "en" : { + "stringUnit" : { + "state" : "translated", + "value" : "Please work with your local IT team to obtain a replacement device" + } + } + } + }, + "Replace Your Device" : { + "comment" : "informationButtonTextUnsupported", + "extractionState" : "manual", + "localizations" : { + "en" : { + "stringUnit" : { + "state" : "translated", + "value" : "Replace Your Device" + } + } + } + }, "Required OS Version:" : { - "comment" : "Required OS Version", + "comment" : "Required OS Version:", "extractionState" : "manual", "localizations" : { "da" : { @@ -2584,7 +2753,7 @@ } }, "Serial Number:" : { - "comment" : "Serial Number", + "comment" : "Serial Number:", "extractionState" : "manual", "localizations" : { "da" : { @@ -2692,7 +2861,7 @@ } }, "Update Device" : { - "comment" : "// Right side of Nudge", + "comment" : "actionButtonText", "extractionState" : "manual", "localizations" : { "da" : { @@ -2800,6 +2969,7 @@ } }, "Updates can take around 30 minutes to complete" : { + "comment" : "mainContentSubHeader", "extractionState" : "manual", "localizations" : { "da" : { @@ -2907,7 +3077,7 @@ } }, "Username:" : { - "comment" : "Username", + "comment" : "Username:", "extractionState" : "manual", "localizations" : { "da" : { @@ -3015,7 +3185,7 @@ } }, "Version:" : { - "comment" : "Nudge Version", + "comment" : "Version:", "extractionState" : "manual", "localizations" : { "da" : { @@ -3123,6 +3293,7 @@ } }, "Your device requires a security update" : { + "comment" : "mainHeader", "extractionState" : "manual", "localizations" : { "da" : { @@ -3230,6 +3401,7 @@ } }, "Your device requires a security update (Demo Mode)" : { + "comment" : "mainHeader (Demo Mode)", "extractionState" : "manual", "localizations" : { "da" : { @@ -3335,113 +3507,6 @@ } } } - }, - "Your device will restart during this update" : { - "extractionState" : "manual", - "localizations" : { - "da" : { - "stringUnit" : { - "state" : "translated", - "value" : "Din Mac vil genstarte under opdateringen" - } - }, - "de" : { - "stringUnit" : { - "state" : "translated", - "value" : "Dein Gerät wird während dieses Updates neu gestartet" - } - }, - "en" : { - "stringUnit" : { - "state" : "translated", - "value" : "Your device will restart during this update" - } - }, - "es" : { - "stringUnit" : { - "state" : "translated", - "value" : "Su dispositivo se reiniciará durante esta actualización" - } - }, - "fr" : { - "stringUnit" : { - "state" : "translated", - "value" : "Votre appareil redémarrera pendant cette mise à jour" - } - }, - "hi" : { - "stringUnit" : { - "state" : "translated", - "value" : "आपका उपकरण इस अपडेट के समय पुनः शुरु होगा" - } - }, - "it" : { - "stringUnit" : { - "state" : "translated", - "value" : "Il dispositivo si riavvierà durante l'aggiornamento" - } - }, - "ja" : { - "stringUnit" : { - "state" : "translated", - "value" : "デバイスは更新中に再起動します" - } - }, - "ko" : { - "stringUnit" : { - "state" : "translated", - "value" : "업데이트 도중에 귀하의 기기는 재시작 될 것입니다" - } - }, - "nb" : { - "stringUnit" : { - "state" : "translated", - "value" : "Du vil bli spurt om å starte på nytt underveis i oppdateringen" - } - }, - "nl" : { - "stringUnit" : { - "state" : "translated", - "value" : "De Mac zal herstarten tijdens het updaten" - } - }, - "pl" : { - "stringUnit" : { - "state" : "translated", - "value" : "Podczas tej aktualizacji Mac uruchomi się ponownie" - } - }, - "pt" : { - "stringUnit" : { - "state" : "translated", - "value" : "Seu dispositivo irá reiniciar durante essa atualização" - } - }, - "ru" : { - "stringUnit" : { - "state" : "translated", - "value" : "Ваше устройство перезагрузится во время обновления" - } - }, - "sv" : { - "stringUnit" : { - "state" : "translated", - "value" : "Din dator kommer att startas om under uppdateringen" - } - }, - "uk" : { - "stringUnit" : { - "state" : "translated", - "value" : "Ваш пристрій буде перезавантажено під час цього оновлення" - } - }, - "zh" : { - "stringUnit" : { - "state" : "translated", - "value" : "在更新过程中,你的Mac将会重新启动。" - } - } - } } }, "version" : "1.0" diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 79fc3554..8ebe3ea1 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -471,7 +471,7 @@ struct UserInterfaceVariables { static var mainContentText: String { userInterfaceUpdateElementsProfile?["mainContentText"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentText ?? - "A fully up-to-date device is required to ensure that IT can accurately protect your device.\n\nIf you do not update your device, you may lose access to some items necessary for your day-to-day tasks.\n\nTo begin the update, simply click on the Update Device button and follow the provided steps." + "A fully up-to-date device is required to ensure that IT can accurately protect your device.\n\nIf you do not update your device, you may lose access to some items necessary for your day-to-day tasks.\n\nTo begin the update, simply click on the **Update Device** button and follow the provided steps." } static var mainContentTextUnsupported: String { From 0ce73f5f03317e4d06db489831f85d30bd36fbf8 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 28 May 2024 10:15:44 -0500 Subject: [PATCH 070/141] google translate as much as I can with what's left --- Localizable.xcstrings | 535 +++++++++++++++++++++++++++++++++++++++--- 1 file changed, 508 insertions(+), 27 deletions(-) diff --git a/Localizable.xcstrings b/Localizable.xcstrings index a9b6fef8..a31290c6 100644 --- a/Localizable.xcstrings +++ b/Localizable.xcstrings @@ -2,7 +2,104 @@ "sourceLanguage" : "en", "strings" : { "" : { - + "localizations" : { + "da" : { + "stringUnit" : { + "state" : "translated", + "value" : "" + } + }, + "de" : { + "stringUnit" : { + "state" : "translated", + "value" : "" + } + }, + "es" : { + "stringUnit" : { + "state" : "translated", + "value" : "" + } + }, + "fr" : { + "stringUnit" : { + "state" : "translated", + "value" : "" + } + }, + "hi" : { + "stringUnit" : { + "state" : "translated", + "value" : "" + } + }, + "it" : { + "stringUnit" : { + "state" : "translated", + "value" : "" + } + }, + "ja" : { + "stringUnit" : { + "state" : "translated", + "value" : "" + } + }, + "ko" : { + "stringUnit" : { + "state" : "translated", + "value" : "" + } + }, + "nb" : { + "stringUnit" : { + "state" : "translated", + "value" : "" + } + }, + "nl" : { + "stringUnit" : { + "state" : "translated", + "value" : "" + } + }, + "pl" : { + "stringUnit" : { + "state" : "translated", + "value" : "" + } + }, + "pt" : { + "stringUnit" : { + "state" : "translated", + "value" : "" + } + }, + "ru" : { + "stringUnit" : { + "state" : "translated", + "value" : "" + } + }, + "sv" : { + "stringUnit" : { + "state" : "translated", + "value" : "" + } + }, + "uk" : { + "stringUnit" : { + "state" : "translated", + "value" : "" + } + }, + "zh" : { + "stringUnit" : { + "state" : "translated", + "value" : "" + } + } + } }, "**A friendly reminder from your local IT team**" : { "comment" : "subHeader / subHeaderUnsupported", @@ -224,11 +321,107 @@ "comment" : "mainContentHeaderUnsupported", "extractionState" : "manual", "localizations" : { + "da" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Din enhed er ikke længere i stand til at modtage kritiske sikkerhedsopdateringer**" + } + }, + "de" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Ihr Gerät ist nicht mehr in der Lage, wichtige Sicherheitsupdates zu empfangen**" + } + }, "en" : { "stringUnit" : { "state" : "translated", "value" : "**Your device is no longer capable of receving critical security updates**" } + }, + "es" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Su dispositivo ya no es capaz de recibir actualizaciones de seguridad críticas**" + } + }, + "fr" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Votre appareil n'est plus capable de recevoir des mises à jour de sécurité critiques**" + } + }, + "hi" : { + "stringUnit" : { + "state" : "translated", + "value" : "**आपका डिवाइस अब महत्वपूर्ण सुरक्षा अपडेट प्राप्त करने में सक्षम नहीं है**" + } + }, + "it" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Il tuo dispositivo non è più in grado di ricevere aggiornamenti di sicurezza critici**" + } + }, + "ja" : { + "stringUnit" : { + "state" : "translated", + "value" : "**お使いのデバイスは重要なセキュリティ アップデートを受信できなくなりました**" + } + }, + "ko" : { + "stringUnit" : { + "state" : "translated", + "value" : "**귀하의 장치는 더 이상 중요한 보안 업데이트를 받을 수 없습니다**" + } + }, + "nb" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Enheten din er ikke lenger i stand til å motta kritiske sikkerhetsoppdateringer**" + } + }, + "nl" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Uw apparaat kan geen kritieke beveiligingsupdates meer ontvangen**" + } + }, + "pl" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Twoje urządzenie nie może już otrzymywać krytycznych aktualizacji zabezpieczeń**" + } + }, + "pt" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Seu dispositivo não é mais capaz de receber atualizações críticas de segurança**" + } + }, + "ru" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Ваше устройство больше не может получать критические обновления безопасности.**" + } + }, + "sv" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Din enhet kan inte längre ta emot viktiga säkerhetsuppdateringar**" + } + }, + "uk" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Ваш пристрій більше не може отримувати критичні оновлення безпеки**" + } + }, + "zh" : { + "stringUnit" : { + "state" : "translated", + "value" : "**您的设备不再能够接收关键安全更新**" + } } } }, @@ -344,11 +537,107 @@ "comment" : "mainContentTextUnsupported", "extractionState" : "manual", "localizations" : { + "da" : { + "stringUnit" : { + "state" : "translated", + "value" : "En fuldt opdateret enhed er påkrævet for at sikre, at IT kan beskytte din enhed nøjagtigt.\n\nHvis du ikke anskaffer dig en erstatningsenhed, mister du adgangen til nogle elementer, der er nødvendige til dine daglige opgaver.\n\nFor mere information om dette, klik venligst på knappen **Erstat din enhed**." + } + }, + "de" : { + "stringUnit" : { + "state" : "translated", + "value" : "Um sicherzustellen, dass die IT Ihr Gerät ordnungsgemäß schützen kann, ist ein vollständig aktuelles Gerät erforderlich.\n\nWenn Sie kein Ersatzgerät erhalten, verlieren Sie den Zugriff auf einige Dinge, die Sie für Ihre täglichen Aufgaben benötigen.\n\nFür weitere Informationen hierzu klicken Sie bitte auf die Schaltfläche **Gerät ersetzen**." + } + }, "en" : { "stringUnit" : { "state" : "translated", "value" : "A fully up-to-date device is required to ensure that IT can accurately protect your device.\n\nIf you do not obtain a replacement device, you will lose access to some items necessary for your day-to-day tasks.\n\nFor more information about this, please click on the **Replace Your Device** button." } + }, + "es" : { + "stringUnit" : { + "state" : "translated", + "value" : "Se requiere un dispositivo completamente actualizado para garantizar que TI pueda proteger con precisión su dispositivo.\n\nSi no obtiene un dispositivo de reemplazo, perderá el acceso a algunos elementos necesarios para sus tareas diarias.\n\nPara obtener más información sobre esto, haga clic en el botón **Reemplazar su dispositivo**." + } + }, + "fr" : { + "stringUnit" : { + "state" : "translated", + "value" : "Un appareil entièrement à jour est nécessaire pour garantir que le service informatique puisse protéger votre appareil avec précision.\n\nSi vous n'obtenez pas d'appareil de remplacement, vous perdrez l'accès à certains éléments nécessaires à vos tâches quotidiennes.\n\nPour plus d'informations à ce sujet, veuillez cliquer sur le bouton **Remplacer votre appareil**." + } + }, + "hi" : { + "stringUnit" : { + "state" : "translated", + "value" : "यह सुनिश्चित करने के लिए कि आईटी आपके डिवाइस की सटीक सुरक्षा कर सके, एक पूरी तरह से अद्यतित डिवाइस की आवश्यकता है।\n\nयदि आपको प्रतिस्थापन उपकरण नहीं मिलता है, तो आप अपने दैनिक कार्यों के लिए आवश्यक कुछ वस्तुओं तक पहुंच खो देंगे।\n\nइसके बारे में अधिक जानकारी के लिए कृपया **अपना डिवाइस बदलें** बटन पर क्लिक करें।" + } + }, + "it" : { + "stringUnit" : { + "state" : "translated", + "value" : "È necessario un dispositivo completamente aggiornato per garantire che l'IT possa proteggere accuratamente il tuo dispositivo.\n\nSe non ottieni un dispositivo sostitutivo, perderai l'accesso ad alcuni elementi necessari per le tue attività quotidiane.\n\nPer ulteriori informazioni a riguardo, fare clic sul pulsante **Sostituisci il dispositivo**." + } + }, + "ja" : { + "stringUnit" : { + "state" : "translated", + "value" : "IT 部門がデバイスを正確に保護するには、完全に最新のデバイスが必要です。\n\n交換用デバイスを入手しない場合、日常業務に必要な一部のアイテムにアクセスできなくなります。\n\n詳細については、[**デバイスを交換する**] ボタンをクリックしてください。" + } + }, + "ko" : { + "stringUnit" : { + "state" : "translated", + "value" : "IT가 귀하의 장치를 정확하게 보호하려면 완전히 최신 장치가 필요합니다.\n\n교체 장치를 구입하지 않으면 일상적인 작업에 필요한 일부 항목에 액세스할 수 없게 됩니다.\n\n이에 대한 자세한 내용을 보려면 **장치 교체** 버튼을 클릭하세요." + } + }, + "nb" : { + "stringUnit" : { + "state" : "translated", + "value" : "En fullstendig oppdatert enhet er nødvendig for å sikre at IT-enheten kan beskytte enheten din nøyaktig.\n\nHvis du ikke skaffer deg en erstatningsenhet, vil du miste tilgangen til enkelte elementer som er nødvendige for dine daglige oppgaver.\n\nFor mer informasjon om dette, klikk på knappen **Erstatt enheten din**." + } + }, + "nl" : { + "stringUnit" : { + "state" : "translated", + "value" : "Een volledig up-to-date apparaat is vereist om ervoor te zorgen dat IT uw apparaat nauwkeurig kan beschermen.\n\nAls u geen vervangend apparaat krijgt, verliest u de toegang tot bepaalde items die nodig zijn voor uw dagelijkse taken.\n\nVoor meer informatie hierover klikt u op de knop **Vervang uw apparaat**." + } + }, + "pl" : { + "stringUnit" : { + "state" : "translated", + "value" : "Aby dział IT mógł dokładnie chronić Twoje urządzenie, wymagane jest w pełni aktualne urządzenie.\n\nJeśli nie uzyskasz urządzenia zastępczego, utracisz dostęp do niektórych rzeczy niezbędnych do codziennych zadań.\n\nAby uzyskać więcej informacji na ten temat, kliknij przycisk **Wymień urządzenie**." + } + }, + "pt" : { + "stringUnit" : { + "state" : "translated", + "value" : "Um dispositivo totalmente atualizado para garantir que a TI possa proteger seu dispositivo com precisão.\n\nSe você não adquirir um dispositivo substituto, perderá o acesso a alguns itens necessários para suas tarefas diárias.\n\nPara obter mais informações sobre isso, clique no botão **Substitua seu dispositivo**." + } + }, + "ru" : { + "stringUnit" : { + "state" : "translated", + "value" : "Требуется полностью обновленное устройство, чтобы ИТ-специалисты могли точно защитить ваше устройство.\n\nЕсли вы не получите устройство на замену, вы потеряете доступ к некоторым элементам, необходимым для выполнения повседневных задач.\n\nДля получения дополнительной информации об этом нажмите кнопку **Заменить устройство**." + } + }, + "sv" : { + "stringUnit" : { + "state" : "translated", + "value" : "En helt uppdaterad enhet krävs för att säkerställa att IT kan skydda din enhet korrekt.\n\nOm du inte skaffar en ersättningsenhet kommer du att förlora tillgången till vissa artiklar som behövs för dina dagliga uppgifter.\n\nFör mer information om detta, klicka på knappen **Byt ut din enhet**." + } + }, + "uk" : { + "stringUnit" : { + "state" : "translated", + "value" : "Потрібен повністю оновлений пристрій, щоб ІТ-спеціалісти могли точно захистити ваш пристрій.\n\nЯкщо ви не отримаєте пристрій на заміну, ви втратите доступ до деяких предметів, необхідних для виконання повсякденних завдань.\n\nЩоб дізнатися більше про це, натисніть кнопку **Замінити свій пристрій**." + } + }, + "zh" : { + "stringUnit" : { + "state" : "translated", + "value" : "需要完全更新的设备才能确保 IT 能够准确保护您的设备。\n\n如果您没有获得替换设备,您将无法访问日常任务所需的某些物品。\n\n有关详细信息,请单击\"**更换您的设备**\"按钮。" + } } } }, @@ -358,14 +647,14 @@ "localizations" : { "da" : { "stringUnit" : { - "state" : "needs_review", - "value" : "En fuldt opdateret Mac er påkrævet for at IT afdelingen kan beskytte dine data.\n\nHvis du ikke opdaterer din Mac, kan du risikere at miste adgang til muligheder, som er nødvendige i dagligdagen.\n\nFor at påbegynde opdateringen skal du klikke på \"Opdater Mac\"-knappen og følge instruktionerne på skærmen." + "state" : "translated", + "value" : "En fuldt opdateret Mac er påkrævet for at IT afdelingen kan beskytte dine data.\n\nHvis du ikke opdaterer din Mac, kan du risikere at miste adgang til muligheder, som er nødvendige i dagligdagen.\n\nFor at påbegynde opdateringen skal du klikke på **Opdater Mac**-knappen og følge instruktionerne på skærmen." } }, "de" : { "stringUnit" : { - "state" : "needs_review", - "value" : "Ein vollständig aktualisiertes Gerät ist erforderlich, um sicherzustellen, dass die IT-Abteilung dein Gerät effektiv schützen kann.\n\nWenn du dein Gerät nicht aktualisierst, verlierst du möglicherweise den Zugriff auf einige Werkzeuge, die du für deine täglichen Aufgaben benötigst.\n\nUm das Update zu starten, klicke auf die Schaltfläche Gerät Aktualisieren und befolge die angegebenen Schritte." + "state" : "translated", + "value" : "Ein vollständig aktualisiertes Gerät ist erforderlich, um sicherzustellen, dass die IT-Abteilung dein Gerät effektiv schützen kann.\n\nWenn du dein Gerät nicht aktualisierst, verlierst du möglicherweise den Zugriff auf einige Werkzeuge, die du für deine täglichen Aufgaben benötigst.\n\nUm das Update zu starten, klicke auf die Schaltfläche **Gerät Aktualisieren** und befolge die angegebenen Schritte." } }, "en" : { @@ -376,14 +665,14 @@ }, "es" : { "stringUnit" : { - "state" : "needs_review", - "value" : "Se requiere un dispositivo completamente actualizado para garantizar que IT pueda proteger su dispositivo con precisión.\n\nSi no actualiza su dispositivo, es posible que pierda el acceso a algunos elementos necesarios para sus tareas diarias.\n\nPara comenzar la actualización, simplemente haga clic en el botón Actualizar dispositivo y siga los pasos proporcionados." + "state" : "translated", + "value" : "Se requiere un dispositivo completamente actualizado para garantizar que IT pueda proteger su dispositivo con precisión.\n\nSi no actualiza su dispositivo, es posible que pierda el acceso a algunos elementos necesarios para sus tareas diarias.\n\nPara comenzar la actualización, simplemente haga clic en el botón **Actualizar dispositivo** y siga los pasos proporcionados." } }, "fr" : { "stringUnit" : { - "state" : "needs_review", - "value" : "Un système entièrement à jour est nécessaire pour garantir que le service informatique puisse protéger votre appareil efficacement.\n\n Si vous ne mettez pas à jour votre appareil, vous risquez de perdre l'accès à certains outils nécessaires à vos tâches quotidiennes.\n\nPour commencer la mise à jour, cliquez simplement sur le bouton Mettre à jour l'appareil et suivez les étapes fournies." + "state" : "translated", + "value" : "Un système entièrement à jour est nécessaire pour garantir que le service informatique puisse protéger votre appareil efficacement.\n\n Si vous ne mettez pas à jour votre appareil, vous risquez de perdre l'accès à certains outils nécessaires à vos tâches quotidiennes.\n\nPour commencer la mise à jour, cliquez simplement sur le bouton **Mettre à jour l'appareil** et suivez les étapes fournies." } }, "hi" : { @@ -394,8 +683,8 @@ }, "it" : { "stringUnit" : { - "state" : "needs_review", - "value" : "È necessario un dispositivo completamente aggiornato per garantire che l'IT possa proteggere accuratamente il tuo dispositivo.\n\nSe non aggiorni il dispositivo, potresti perdere l'accesso ad alcuni elementi necessari per le tue attività quotidiane.\n\nPer iniziare l'aggiornamento, fai semplicemente clic sul pulsante Aggiorna dispositivo e segui i passaggi forniti." + "state" : "translated", + "value" : "È necessario un dispositivo completamente aggiornato per garantire che l'IT possa proteggere accuratamente il tuo dispositivo.\n\nSe non aggiorni il dispositivo, potresti perdere l'accesso ad alcuni elementi necessari per le tue attività quotidiane.\n\nPer iniziare l'aggiornamento, fai semplicemente clic sul pulsante **Aggiorna dispositivo** e segui i passaggi forniti." } }, "ja" : { @@ -406,32 +695,32 @@ }, "ko" : { "stringUnit" : { - "state" : "needs_review", - "value" : "IT 팀이 귀하의 기기를 정밀하게 보호하기 위하여 최신 업데이트가 기기에 항시 탑재될 것을 요청드립니다.\n\n귀하께서 기기를 업데이트 하지 않는다면, 일상의 업무에 필요한 아이템들의 접근 권한이 상실될 수 있습니다.\n\n기기를 업데이트 하시려면, 기기 업데이트 버튼을 클릭하시고 제공되는 안내를 따르시면 됩니다." + "state" : "translated", + "value" : "IT 팀이 귀하의 기기를 정밀하게 보호하기 위하여 최신 업데이트가 기기에 항시 탑재될 것을 요청드립니다.\n\n귀하께서 기기를 업데이트 하지 않는다면, 일상의 업무에 필요한 아이템들의 접근 권한이 상실될 수 있습니다.\n\n기기를 업데이트 하시려면, **기기 업데이트** 버튼을 클릭하시고 제공되는 안내를 따르시면 됩니다." } }, "nb" : { "stringUnit" : { - "state" : "needs_review", - "value" : "Datamaskinen må være oppdatert med de siste sikkerhetsoppdateringene for å være beskyttet.\n\nHvis du ikke oppdaterer datamaskinen, kan det hende at du ikke får tilgang til de nødvendige ressursene.\n\nFor å starte oppdateringen, klikk på Oppdater nå og følg instruksene." + "state" : "translated", + "value" : "Datamaskinen må være oppdatert med de siste sikkerhetsoppdateringene for å være beskyttet.\n\nHvis du ikke oppdaterer datamaskinen, kan det hende at du ikke får tilgang til de nødvendige ressursene.\n\nFor å starte oppdateringen, klikk på **Oppdater nå** og følg instruksene." } }, "nl" : { "stringUnit" : { - "state" : "needs_review", - "value" : "Een volledig up-to-date Mac is vereist om ervoor te zorgen dat uw Mac goed beschermd blijft.\n\nAls u uw Mac niet bijwerkt kunnen er restricties op de Mac komen.\n\nOm met de update te beginnen, klikt u op de knop Update Mac en volgt u de stappen." + "state" : "translated", + "value" : "Een volledig up-to-date Mac is vereist om ervoor te zorgen dat uw Mac goed beschermd blijft.\n\nAls u uw Mac niet bijwerkt kunnen er restricties op de Mac komen.\n\nOm met de update te beginnen, klikt u op de knop **Update Mac** en volgt u de stappen." } }, "pl" : { "stringUnit" : { - "state" : "needs_review", - "value" : "Wymagane jest uaktualnienie urządzenia, aby mieć pewność, że dział IT może w pełni chronić Twoje urządzenie.\n\nJeśli nie zaktualizujesz urządzenia, możesz utracić dostęp do niektórych elementów niezbędnych do wykonywania codziennych zadań.\n\nAby rozpocząć aktualizację, po prostu kliknij przycisk Uaktualnij system i postępuj zgodnie z podanymi krokami." + "state" : "translated", + "value" : "Wymagane jest uaktualnienie urządzenia, aby mieć pewność, że dział IT może w pełni chronić Twoje urządzenie.\n\nJeśli nie zaktualizujesz urządzenia, możesz utracić dostęp do niektórych elementów niezbędnych do wykonywania codziennych zadań.\n\nAby rozpocząć aktualizację, po prostu kliknij przycisk **Aktualizuj urządzenie** i postępuj zgodnie z podanymi krokami." } }, "pt" : { "stringUnit" : { - "state" : "needs_review", - "value" : "Um dispositivo totalmente atualizado é necessário para garantir que IT possa proteger seu dispositivo com precisão.\n\nSe não atualizar seu dispositivo, é possível que você perca acesso a itens necessários para performar suas tarefas do dia-a-dia.\n\nPara começar a atualização, apenas clique no botão Atualizar Dispositivo e siga as instruções apresentadas." + "state" : "translated", + "value" : "Um dispositivo totalmente atualizado é necessário para garantir que IT possa proteger seu dispositivo com precisão.\n\nSe não atualizar seu dispositivo, é possível que você perca acesso a itens necessários para performar suas tarefas do dia-a-dia.\n\nPara começar a atualização, apenas clique no botão **Atualizar Dispositivo** e siga as instruções apresentadas." } }, "ru" : { @@ -442,20 +731,20 @@ }, "sv" : { "stringUnit" : { - "state" : "needs_review", - "value" : "Ett system med de senaste uppdateringarna krävs för att IT ska kunna säkerställa att din dator är skyddad.\n\nOm du inte uppdaterar systemet kommer du kanske inte kunna komma åt nödvändinga resurser.\n\nFör att påbörja uppdateringen, klicka på Uppdatera och följ stegen." + "state" : "translated", + "value" : "Ett system med de senaste uppdateringarna krävs för att IT ska kunna säkerställa att din dator är skyddad.\n\nOm du inte uppdaterar systemet kommer du kanske inte kunna komma åt nödvändinga resurser.\n\nFör att påbörja uppdateringen, klicka på **Uppdatera** och följ stegen." } }, "uk" : { "stringUnit" : { - "state" : "needs_review", - "value" : "Потрібен повністю оновлений пристрій щоб бути впевненим що IT зможе його надійно захистити.\n\nЯкщо ви не оновите пристрій, ви можете втратити доступ до деяких речей, необхідних для виконання повсякденних завдань.\n\nЩоб почати оновлення, просто клацніть кнопку Оновити пристрій та виконайте запропоновані кроки." + "state" : "translated", + "value" : "Потрібен повністю оновлений пристрій щоб бути впевненим що IT зможе його надійно захистити.\n\nЯкщо ви не оновите пристрій, ви можете втратити доступ до деяких речей, необхідних для виконання повсякденних завдань.\n\nЩоб почати оновлення, просто клацніть кнопку **Оновити пристрій** та виконайте запропоновані кроки." } }, "zh" : { "stringUnit" : { - "state" : "needs_review", - "value" : "更新至最新的软件以确保 IT 可以精准的保护设备安全。\n\n如未及时完成更新,您将会无法访问日常工作所需的网页或程序。\n \n立刻更新,请点击【更新设备】并根据提供的步骤完成更新。" + "state" : "translated", + "value" : "更新至最新的软件以确保 IT 可以精准的保护设备安全。\n\n如未及时完成更新,您将会无法访问日常工作所需的网页或程序。\n \n立刻更新,请点击\"**更新设备**\"并根据提供的步骤完成更新。" } } } @@ -2624,11 +2913,107 @@ "comment" : "mainContentSubHeaderUnsupported", "extractionState" : "manual", "localizations" : { + "da" : { + "stringUnit" : { + "state" : "translated", + "value" : "Arbejd med dit lokale it-team for at få en erstatningsenhed" + } + }, + "de" : { + "stringUnit" : { + "state" : "translated", + "value" : "Bitte arbeiten Sie mit Ihrem lokalen IT-Team zusammen, um ein Ersatzgerät zu erhalten" + } + }, "en" : { "stringUnit" : { "state" : "translated", "value" : "Please work with your local IT team to obtain a replacement device" } + }, + "es" : { + "stringUnit" : { + "state" : "translated", + "value" : "Trabaje con su equipo de TI local para obtener un dispositivo de reemplazo." + } + }, + "fr" : { + "stringUnit" : { + "state" : "translated", + "value" : "Veuillez travailler avec votre équipe informatique locale pour obtenir un appareil de remplacement." + } + }, + "hi" : { + "stringUnit" : { + "state" : "translated", + "value" : "प्रतिस्थापन उपकरण प्राप्त करने के लिए कृपया अपनी स्थानीय आईटी टीम के साथ काम करें" + } + }, + "it" : { + "stringUnit" : { + "state" : "translated", + "value" : "Collabora con il team IT locale per ottenere un dispositivo sostitutivo" + } + }, + "ja" : { + "stringUnit" : { + "state" : "translated", + "value" : "地元の IT チームと協力して交換用デバイスを入手してください" + } + }, + "ko" : { + "stringUnit" : { + "state" : "translated", + "value" : "교체 장치를 얻으려면 현지 IT 팀과 협력하십시오." + } + }, + "nb" : { + "stringUnit" : { + "state" : "translated", + "value" : "Samarbeid med ditt lokale IT-team for å få en erstatningsenhet" + } + }, + "nl" : { + "stringUnit" : { + "state" : "translated", + "value" : "Werk samen met uw lokale IT-team om een vervangend apparaat te verkrijgen" + } + }, + "pl" : { + "stringUnit" : { + "state" : "translated", + "value" : "Skontaktuj się z lokalnym zespołem IT, aby uzyskać urządzenie zastępcze" + } + }, + "pt" : { + "stringUnit" : { + "state" : "translated", + "value" : "Trabalhe com sua equipe de TI local para obter um dispositivo de substituição" + } + }, + "ru" : { + "stringUnit" : { + "state" : "translated", + "value" : "Свяжитесь с местной ИТ-отделом, чтобы получить устройство на замену." + } + }, + "sv" : { + "stringUnit" : { + "state" : "translated", + "value" : "Vänligen samarbeta med ditt lokala IT-team för att skaffa en ersättningsenhet" + } + }, + "uk" : { + "stringUnit" : { + "state" : "translated", + "value" : "Зверніться до місцевої ІТ-служби, щоб отримати пристрій на заміну" + } + }, + "zh" : { + "stringUnit" : { + "state" : "translated", + "value" : "请与您当地的 IT 团队合作获取更换设备" + } } } }, @@ -2636,11 +3021,107 @@ "comment" : "informationButtonTextUnsupported", "extractionState" : "manual", "localizations" : { + "da" : { + "stringUnit" : { + "state" : "translated", + "value" : "Udskift din enhed" + } + }, + "de" : { + "stringUnit" : { + "state" : "translated", + "value" : "Ersetzen Sie Ihr Gerät" + } + }, "en" : { "stringUnit" : { "state" : "translated", "value" : "Replace Your Device" } + }, + "es" : { + "stringUnit" : { + "state" : "translated", + "value" : "Reemplace su dispositivo" + } + }, + "fr" : { + "stringUnit" : { + "state" : "translated", + "value" : "Remplacez votre appareil" + } + }, + "hi" : { + "stringUnit" : { + "state" : "translated", + "value" : "अपना उपकरण बदलें" + } + }, + "it" : { + "stringUnit" : { + "state" : "translated", + "value" : "Sostituisci il tuo dispositivo" + } + }, + "ja" : { + "stringUnit" : { + "state" : "translated", + "value" : "デバイスを交換してください" + } + }, + "ko" : { + "stringUnit" : { + "state" : "translated", + "value" : "장치 교체" + } + }, + "nb" : { + "stringUnit" : { + "state" : "translated", + "value" : "Bytt ut enheten din" + } + }, + "nl" : { + "stringUnit" : { + "state" : "translated", + "value" : "Vervang uw apparaat" + } + }, + "pl" : { + "stringUnit" : { + "state" : "translated", + "value" : "Wymień swoje urządzenie" + } + }, + "pt" : { + "stringUnit" : { + "state" : "translated", + "value" : "Substitua seu dispositivo" + } + }, + "ru" : { + "stringUnit" : { + "state" : "translated", + "value" : "Замените ваше устройство" + } + }, + "sv" : { + "stringUnit" : { + "state" : "translated", + "value" : "Byt ut din enhet" + } + }, + "uk" : { + "stringUnit" : { + "state" : "translated", + "value" : "Замініть свій пристрій" + } + }, + "zh" : { + "stringUnit" : { + "state" : "translated", + "value" : "更换您的设备" + } } } }, From e28f20160815d1a875f1e03d1d2c557b5812cbac Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 28 May 2024 10:24:58 -0500 Subject: [PATCH 071/141] explicitly log OS versioning errors --- Nudge/Utilities/OSVersion.swift | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/Nudge/Utilities/OSVersion.swift b/Nudge/Utilities/OSVersion.swift index fb8fdac9..bb3ea75b 100644 --- a/Nudge/Utilities/OSVersion.swift +++ b/Nudge/Utilities/OSVersion.swift @@ -36,11 +36,15 @@ public struct OSVersion { public init(_ string: String) throws { let parts = string.split(separator: ".", omittingEmptySubsequences: false) guard parts.count == 2 || parts.count == 3 else { - throw ParseError.badFormat(reason: "Input \(string) must have 2 or 3 parts, got \(parts.count).") + let error = "Input \(string) must have 2 or 3 parts, got \(parts.count)." + LogManager.error(error, logger: utilsLog) + throw ParseError.badFormat(reason: error) } guard let major = Int(parts[0]), let minor = Int(parts[1]) else { - throw ParseError.badFormat(reason: "Invalid format for major or minor version in \(string).") + let error = "Invalid format for major or minor version in \(string)." + LogManager.error(error, logger: utilsLog) + throw ParseError.badFormat(reason: error) } let patch = parts.count >= 3 ? Int(parts[2]) ?? 0 : 0 From 92c67f85344be4df002d9ef68b9029cffbe0a320 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Thu, 30 May 2024 12:10:19 -0500 Subject: [PATCH 072/141] add custom icon in the application termination notification --- CHANGELOG.md | 8 ++- .../com.github.macadmins.Nudge.tester.json | 11 ++-- .../Preferences/DefaultPreferencesNudge.swift | 20 +++++- Nudge/Preferences/PreferencesStructure.swift | 16 +++-- Nudge/UI/Defaults.swift | 1 + Nudge/UI/Main.swift | 62 ++++++++++++++++++- 6 files changed, 103 insertions(+), 15 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 05f134e9..1a12989a 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -31,6 +31,11 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - Fixed errors when moving to Swift 5.10 ### Added +- A local image path can now be specified for the notification event when Nudge terminates and application + - Please configure the `applicationTerminatedNotificationImagePath` key under `userInterface` + - Due to limitations within Apple's API, a local path is only supported at this time +- An admin can now alter the text when Nudge terminates and application + - Please configure the `applicationTerminatedTitleText` and `applicationTerminatedBodyText` keys under the `updateElements` key in `UserInterface` - Remote URLs can now be used on `iconDarkPath`, `iconLightPath`, `screenShotDarkPath` and `screenShotLightPath` - Please note that these files will be downloaded each time Nudge is ran and there is currently not a way to cache these objects. - If these files fail to download, a default company logo will be shown. @@ -47,8 +52,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - If you are utilizing a custom sofa feed, please configure the `customSOFAFeedURL` key under `optionalFeatures` - "Unsupported device" UI in standard mode that utilizes the SOFA feed - Set the `attemptToCheckForSupportedDevice` key `false` under `optionalFeatures` to disable this feature - - There are new keys to set all of text fields - - `informationButtonTextUnsupported`, `mainContentHeaderUnsupported`, `mainContentNoteUnsupported`, `mainContentSubHeaderUnsupported`, `mainContentTextUnsupported`, `subHeaderUnsupported` + - There are new keys to set all of text fields: `informationButtonTextUnsupported`, `mainContentHeaderUnsupported`, `mainContentNoteUnsupported`, `mainContentSubHeaderUnsupported`, `mainContentTextUnsupported`, `subHeaderUnsupported` under the `updateElements` key in `UserInterface` - `unsupportedURL` and `unsupportedURLs` can change the information button itself, but it will remain in the `osVersionRequirement` key with `unsupportedURLs` and `unsupportedURLs`. - An icon will appear as an overlay on top of the company image to further emphasize the device is no longer supported diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index 26200cda..27981914 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -11,17 +11,17 @@ "acceptableScreenSharingUsage": true, "attemptToBlockApplicationLaunches": true, "blockedApplicationBundleIDs": [ - "com.microsoft.VSCode", - "us.zoom.xos" + "com.apple.Safari" ], "disableNudgeForStandardInstalls": true, - "terminateApplicationsOnLaunch": false, - "utilizeSOFAFeed": true + "terminateApplicationsOnLaunch": true, + "utilizeSOFAFeed": false }, "osVersionRequirements": [ { "aboutUpdateURL": "https://apple.com", - "requiredMinimumOSVersion": "latest", + "requiredMinimumOSVersion": "14.6", + "requiredInstallationDate": "2024-01-01T00:00:00Z", "unsupportedURL": "https://google.com", } ], @@ -37,6 +37,7 @@ "iconLightPath": "https://github.com/macadmins/nudge/blob/main/assets/NudgeIcon.png?raw=true", "screenShotDarkPath": "https://github.com/macadmins/nudge/blob/main/assets/standard_mode/demo_dark_1_icon.png?raw=true", "screenShotLightPath": "https://github.com/macadmins/nudge/blob/main/assets/standard_mode/demo_light_1_icon.png?raw=true", + "applicationTerminatedNotificationImagePath": "/Library/Application Support/Nudge/logoLight.png", "simpleMode": false, "updateElements": [ { diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 8ebe3ea1..b1916572 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -377,7 +377,13 @@ struct UserInterfaceVariables { static var userInterfaceJSON: UserInterface? = getUserInterfaceJSON() static var userInterfaceUpdateElementsProfile: [String:AnyObject]? = getUserInterfaceUpdateElementsProfile() static var userInterfaceUpdateElementsJSON: UpdateElement? = getUserInterfaceUpdateElementsJSON() - + + static var applicationTerminatedNotificationImagePath: String { + userInterfaceProfile?["applicationTerminatedNotificationImagePath"] as? String ?? + userInterfaceJSON?.applicationTerminatedNotificationImagePath ?? + "" + } + static var fallbackLanguage: String { userInterfaceProfile?["fallbackLanguage"] as? String ?? userInterfaceJSON?.fallbackLanguage ?? @@ -420,6 +426,18 @@ struct UserInterfaceVariables { "Update Device" } + static var applicationTerminatedTitleText: String { + userInterfaceUpdateElementsProfile?["applicationTerminatedTitleText"] as? String ?? + userInterfaceUpdateElementsJSON?.applicationTerminatedTitleText ?? + "Application terminated" + } + + static var applicationTerminatedBodyText: String { + userInterfaceUpdateElementsProfile?["applicationTerminatedBodyText"] as? String ?? + userInterfaceUpdateElementsJSON?.applicationTerminatedBodyText ?? + "Please update your device to use this application" + } + static var informationButtonText: String { userInterfaceUpdateElementsProfile?["informationButtonText"] as? String ?? userInterfaceUpdateElementsJSON?.informationButtonText ?? diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index 5dffecb1..1c772b80 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -400,7 +400,7 @@ extension UserExperience { // MARK: - UserInterface struct UserInterface: Codable { - var actionButtonPath, fallbackLanguage: String? + var actionButtonPath, applicationTerminatedNotificationImagePath, fallbackLanguage: String? var forceFallbackLanguage, forceScreenShotIcon: Bool? var iconDarkPath, iconLightPath, screenShotDarkPath, screenShotLightPath: String? var showActivelyExploitedCVEs, showDeferralCount, simpleMode, singleQuitButton: Bool? @@ -427,6 +427,7 @@ extension UserInterface { func with( actionButtonPath: String? = nil, + applicationTerminatedNotificationImagePath: String? = nil, fallbackLanguage: String? = nil, forceFallbackLanguage: Bool? = nil, forceScreenShotIcon: Bool? = nil, @@ -442,6 +443,7 @@ extension UserInterface { ) -> UserInterface { return UserInterface( actionButtonPath: actionButtonPath ?? self.actionButtonPath, + applicationTerminatedNotificationImagePath: applicationTerminatedNotificationImagePath ?? self.applicationTerminatedNotificationImagePath, fallbackLanguage: fallbackLanguage ?? self.fallbackLanguage, forceFallbackLanguage: forceFallbackLanguage ?? self.forceFallbackLanguage, forceScreenShotIcon: forceScreenShotIcon ?? self.forceScreenShotIcon, @@ -460,14 +462,14 @@ extension UserInterface { // MARK: - UpdateElement struct UpdateElement: Codable { - var language, actionButtonText, customDeferralButtonText, customDeferralDropdownText, informationButtonText, informationButtonTextUnsupported: String? - var mainContentHeader, mainContentHeaderUnsupported, mainContentNote, mainContentNoteUnsupported, mainContentSubHeader, mainContentSubHeaderUnsupported: String? - var mainContentText, mainContentTextUnsupported, mainHeader, mainHeaderUnsupported: String? + var language, actionButtonText, applicationTerminatedTitleText, applicationTerminatedBodyText, customDeferralButtonText, customDeferralDropdownText: String? + var informationButtonText, informationButtonTextUnsupported, mainContentHeader, mainContentHeaderUnsupported, mainContentNote, mainContentNoteUnsupported: String? + var mainContentSubHeader, mainContentSubHeaderUnsupported, mainContentText, mainContentTextUnsupported, mainHeader, mainHeaderUnsupported: String? var oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, secondaryQuitButtonText, subHeader, subHeaderUnsupported, screenShotAltText: String? enum CodingKeys: String, CodingKey { case language = "_language" - case actionButtonText, customDeferralButtonText, customDeferralDropdownText, informationButtonText, informationButtonTextUnsupported, mainContentHeader, mainContentHeaderUnsupported, mainContentNote, mainContentNoteUnsupported, mainContentSubHeader, mainContentSubHeaderUnsupported, mainContentText, mainContentTextUnsupported, mainHeader, mainHeaderUnsupported, oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, secondaryQuitButtonText, subHeader, subHeaderUnsupported, screenShotAltText + case actionButtonText, applicationTerminatedTitleText, applicationTerminatedBodyText, customDeferralButtonText, customDeferralDropdownText, informationButtonText, informationButtonTextUnsupported, mainContentHeader, mainContentHeaderUnsupported, mainContentNote, mainContentNoteUnsupported, mainContentSubHeader, mainContentSubHeaderUnsupported, mainContentText, mainContentTextUnsupported, mainHeader, mainHeaderUnsupported, oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, secondaryQuitButtonText, subHeader, subHeaderUnsupported, screenShotAltText } } @@ -492,6 +494,8 @@ extension UpdateElement { func with( language: String? = nil, actionButtonText: String? = nil, + applicationTerminatedTitleText: String? = nil, + applicationTerminatedBodyText: String? = nil, customDeferralButtonText: String? = nil, customDeferralDropdownText: String? = nil, informationButtonText: String? = nil, @@ -517,6 +521,8 @@ extension UpdateElement { return UpdateElement( language: language ?? self.language, actionButtonText: actionButtonText ?? self.actionButtonText, + applicationTerminatedTitleText: applicationTerminatedTitleText ?? self.applicationTerminatedTitleText, + applicationTerminatedBodyText: applicationTerminatedBodyText ?? self.applicationTerminatedBodyText, customDeferralButtonText: customDeferralButtonText ?? self.customDeferralButtonText, customDeferralDropdownText: customDeferralDropdownText ?? self.customDeferralDropdownText, informationButtonText: informationButtonText ?? self.informationButtonText, diff --git a/Nudge/UI/Defaults.swift b/Nudge/UI/Defaults.swift index e4f87795..d43919e2 100644 --- a/Nudge/UI/Defaults.swift +++ b/Nudge/UI/Defaults.swift @@ -76,6 +76,7 @@ class AppState: ObservableObject { @Published var hasLoggedDeferralCountPastThreshold = false @Published var hasLoggedDeferralCountPastThresholdDualQuitButtons = false @Published var hasLoggedRequireDualQuitButtons = false + @Published var hasRenderedApplicationTerminatedNotificationImagePath = false @Published var hoursRemaining = DateManager().getNumberOfHoursRemaining() @Published var lastRefreshTime = DateManager().getFormattedDate() @Published var requireDualQuitButtons = false diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 5442052f..a63caba2 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -369,11 +369,69 @@ class AppDelegate: NSObject, NSApplicationDelegate { private func createNotificationContent(for applicationIdentifier: String) -> UNMutableNotificationContent { let content = UNMutableNotificationContent() - content.title = "Application terminated".localized(desiredLanguage: getDesiredLanguage()) + content.title = UserInterfaceVariables.applicationTerminatedTitleText.localized(desiredLanguage: getDesiredLanguage()) content.subtitle = "(\(applicationIdentifier))" - content.body = "Please update your device to use this application".localized(desiredLanguage: getDesiredLanguage()) + content.title = UserInterfaceVariables.applicationTerminatedBodyText.localized(desiredLanguage: getDesiredLanguage()) content.categoryIdentifier = "alert" content.sound = UNNotificationSound.default + content.attachments = [] + let applicationTerminatedNotificationImagePath = UserInterfaceVariables.applicationTerminatedNotificationImagePath + let tempImagePath = "/var/tmp/nudge-applicationTerminatedNotification.png" + if FileManager.default.fileExists(atPath: applicationTerminatedNotificationImagePath) { + if nudgePrimaryState.hasRenderedApplicationTerminatedNotificationImagePath { + do { + let fileURL = URL(fileURLWithPath: tempImagePath) + let attachment = try UNNotificationAttachment(identifier: "AttachedContent", url: fileURL, options: .none) + content.attachments = [attachment] + } catch let error { + LogManager.error("\(error)", logger: uiLog) + } + } else { + do { + // In order for the attachment to look properly, it has to be resized to a square + guard let sourceImage = NSImage(contentsOfFile: applicationTerminatedNotificationImagePath) else { + throw NSError(domain: "Failed to load image from path: \(applicationTerminatedNotificationImagePath)", code: 0, userInfo: nil) + } + // Find the maximum dimension and create a square based on it + let maxDimension = max(sourceImage.size.width, sourceImage.size.height) + let newSize = CGSize(width: maxDimension, height: maxDimension) + + // Create a new image with a square size, filling with transparent background + let targetImage = NSImage(size: newSize) + targetImage.lockFocus() + let context = NSGraphicsContext.current! + context.imageInterpolation = .high + NSColor.clear.set() + NSBezierPath(rect: NSRect(origin: .zero, size: newSize)).fill() + + // Calculate the origin point to center the source image + let x = (maxDimension - sourceImage.size.width) / 2 + let y = (maxDimension - sourceImage.size.height) / 2 + let targetRect = NSRect(x: x, y: y, width: sourceImage.size.width, height: sourceImage.size.height) + + sourceImage.draw(in: targetRect, from: NSRect(origin: .zero, size: sourceImage.size), operation: .sourceOver, fraction: 1.0) + targetImage.unlockFocus() + + guard let tiffData = targetImage.tiffRepresentation, + let bitmapImage = NSBitmapImageRep(data: tiffData), + let pngData = bitmapImage.representation(using: .png, properties: [:]) else { + throw NSError(domain: "Failed to create or convert image", code: 0, userInfo: nil) + } + + try pngData.write(to: URL(fileURLWithPath: tempImagePath)) + + // Load temporary file + let fileURL = URL(fileURLWithPath: tempImagePath) + let attachment = try UNNotificationAttachment(identifier: "AttachedContent", url: fileURL, options: .none) + content.attachments = [attachment] + nudgePrimaryState.hasRenderedApplicationTerminatedNotificationImagePath = true + } catch let error { + LogManager.error("\(error)", logger: uiLog) + } + } + } else { + print("applicationTerminatedNotificationImagePath does not exist on disk, skipping notification image.") + } return content } From febf07351cca935ffb81c200e27825259d6703c3 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Thu, 30 May 2024 16:25:49 -0500 Subject: [PATCH 073/141] fix issue with fallbackLanguage logic --- Nudge/Utilities/Preferences.swift | 27 +++++++++++++++++++++++++++ 1 file changed, 27 insertions(+) diff --git a/Nudge/Utilities/Preferences.swift b/Nudge/Utilities/Preferences.swift index 7bdd5d39..9bdef871 100644 --- a/Nudge/Utilities/Preferences.swift +++ b/Nudge/Utilities/Preferences.swift @@ -10,6 +10,33 @@ import Foundation // Generics func getDesiredLanguage(locale: Locale? = nil) -> String { if UserInterfaceVariables.forceFallbackLanguage { + if Globals.configProfile.isEmpty { + let userInterfaceUpdateElementsJSON: UserInterface? = getUserInterfaceJSON() + if let elements = userInterfaceUpdateElementsJSON?.updateElements { + for element in elements { + if element.language == UIConstants.languageCode { + return UIConstants.languageCode + } + } + } + } else { + do { + // Attempt to decode the plist Data into a dictionary + if let dictionary = try PropertyListSerialization.propertyList(from: Globals.configProfile, options: [], format: nil) as? [String: Any], + let userInterface = dictionary["userInterface"] as? [String: Any] { + guard let elements = userInterface["updateElements"] as? [[String: AnyObject]] else { + return UIConstants.languageCode + } + for element in elements { + if element["_language"] as? String == UIConstants.languageCode { + return UIConstants.languageCode + } + } + } + } catch { + print("Failed to decode plist: \(error)") + } + } return UserInterfaceVariables.fallbackLanguage } From 83356a28f9173775206506fba5ad1d89414eda1d Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Thu, 30 May 2024 16:30:25 -0500 Subject: [PATCH 074/141] update changelog --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 1a12989a..e4ae2b9f 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -26,6 +26,8 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t ### Fixed - `screenshotDisplay` view had a bug that may result in the screenshot being partially cut off or zoomable +- `fallbackLanguage` would return the wrong language even when specified in the configuration + - Fixes [582](https://github.com/macadmins/nudge/issues/582) - More descriptive logs when loading json/mdm profile keys - Refactor portions of the `softwareupdate` logic to reduce potential errors - Fixed errors when moving to Swift 5.10 From 5091266d83781d3fa2195f33ee29b2209d0dca74 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 31 May 2024 12:18:34 -0500 Subject: [PATCH 075/141] move timerCycle logic to seconds remaining vs hours --- Nudge/UI/Defaults.swift | 1 + Nudge/UI/Main.swift | 1 + Nudge/Utilities/Utils.swift | 18 ++++++++++++------ 3 files changed, 14 insertions(+), 6 deletions(-) diff --git a/Nudge/UI/Defaults.swift b/Nudge/UI/Defaults.swift index d43919e2..0dc8e5cf 100644 --- a/Nudge/UI/Defaults.swift +++ b/Nudge/UI/Defaults.swift @@ -78,6 +78,7 @@ class AppState: ObservableObject { @Published var hasLoggedRequireDualQuitButtons = false @Published var hasRenderedApplicationTerminatedNotificationImagePath = false @Published var hoursRemaining = DateManager().getNumberOfHoursRemaining() + @Published var secondsRemaining = DateManager().getNumberOfSecondsRemaining() @Published var lastRefreshTime = DateManager().getFormattedDate() @Published var requireDualQuitButtons = false @Published var requiredMinimumOSVersion = OSVersionRequirementVariables.requiredMinimumOSVersion diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index a63caba2..f6d0088d 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -110,6 +110,7 @@ struct ContentView: View { } appState.daysRemaining = DateManager().getNumberOfDaysBetween() appState.hoursRemaining = DateManager().getNumberOfHoursRemaining() + appState.secondsRemaining = DateManager().getNumberOfSecondsRemaining() } } diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 04820447..74380132 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -394,13 +394,13 @@ struct CommandLineUtilities { } struct ConfigurationManager { - private func determineTimerCycle(basedOn hoursRemaining: Int) -> Int { - switch hoursRemaining { + private func determineTimerCycle(basedOn secondsRemaining: Int) -> Int { + switch secondsRemaining { case ...0: return UserExperienceVariables.elapsedRefreshCycle - case ...UserExperienceVariables.imminentWindowTime: + case ...(UserExperienceVariables.imminentWindowTime * 3600): return UserExperienceVariables.imminentRefreshCycle - case ...UserExperienceVariables.approachingWindowTime: + case ...(UserExperienceVariables.approachingWindowTime * 3600): return UserExperienceVariables.approachingRefreshCycle default: return UserExperienceVariables.initialRefreshCycle @@ -461,8 +461,8 @@ struct ConfigurationManager { } func getTimerController() -> Int { - let hoursRemaining = DateManager().getNumberOfHoursRemaining() - let timerCycle = determineTimerCycle(basedOn: hoursRemaining) + let secondsRemaining = DateManager().getNumberOfSecondsRemaining() + let timerCycle = determineTimerCycle(basedOn: secondsRemaining) if timerCycle != nudgePrimaryState.timerCycle { LogManager.info("timerCycle: \(timerCycle)", logger: uiLog) @@ -525,6 +525,12 @@ struct DateManager { return Int(interval.timeIntervalSince(currentDate) / 3600) } + func getNumberOfSecondsRemaining(currentDate: Date = DateManager().getCurrentDate()) -> Int { + guard !CommandLineUtilities().demoModeEnabled() else { return 24 * 3600 } + let interval = CommandLineUtilities().unitTestingEnabled() ? PrefsWrapper.requiredInstallationDate : requiredInstallationDate + return Int(interval.timeIntervalSince(currentDate)) + } + func pastRequiredInstallationDate() -> Bool { let isPast = getCurrentDate() > requiredInstallationDate if !CommandLineUtilities().demoModeEnabled() && !nudgeLogState.hasLoggedPastRequiredInstallationDate { From d77b540ddfb562fcd1c198fd88f4d836275e6047 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 31 May 2024 12:20:32 -0500 Subject: [PATCH 076/141] update changelog for issue 568 --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index e4ae2b9f..2ec9fe53 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -28,6 +28,8 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - `screenshotDisplay` view had a bug that may result in the screenshot being partially cut off or zoomable - `fallbackLanguage` would return the wrong language even when specified in the configuration - Fixes [582](https://github.com/macadmins/nudge/issues/582) +- The timer controller logic was utilizing hours remaining vs seconds, which resulted in the `elapsedRefreshCycle` being used at the final hour of the nudge event vs the `imminentRefreshCycle`. This has been corrected to calculate the seconds remaining. + - Fixes [568](https://github.com/macadmins/nudge/issues/568) - More descriptive logs when loading json/mdm profile keys - Refactor portions of the `softwareupdate` logic to reduce potential errors - Fixed errors when moving to Swift 5.10 From 8258ebe997a66b50c03040df6feaaf71317f0d12 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 31 May 2024 16:12:27 -0500 Subject: [PATCH 077/141] optionally show required install date on left side UI --- CHANGELOG.md | 4 + .../com.github.macadmins.Nudge.tester.json | 5 +- Localizable.xcstrings | 108 ++++++++++++++++++ .../Preferences/DefaultPreferencesNudge.swift | 12 ++ Nudge/Preferences/PreferencesStructure.swift | 8 +- Nudge/UI/StandardMode/LeftSide.swift | 6 + Nudge/Utilities/Utils.swift | 6 + 7 files changed, 145 insertions(+), 4 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 2ec9fe53..d3c93bc2 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -59,6 +59,10 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - There are new keys to set all of text fields: `informationButtonTextUnsupported`, `mainContentHeaderUnsupported`, `mainContentNoteUnsupported`, `mainContentSubHeaderUnsupported`, `mainContentTextUnsupported`, `subHeaderUnsupported` under the `updateElements` key in `UserInterface` - `unsupportedURL` and `unsupportedURLs` can change the information button itself, but it will remain in the `osVersionRequirement` key with `unsupportedURLs` and `unsupportedURLs`. - An icon will appear as an overlay on top of the company image to further emphasize the device is no longer supported +- An admin can now show the `requiredInstallationDate` as a item on the left side of nudge. + - To enable this, please configure the `showRequiredInstallationDate` key under `userInterface` to true + - You can also expirement with the format of this date through the key `requiredInstallationDisplayFormat` under `userInterface` + - Be aware that the format you desire may not look good on the UI. ## [1.1.16] - 2024-03-13 This will be the **final Nudge release** for macOS 11 and potentially other versions of macOS. diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index 27981914..a6dfcd32 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -21,7 +21,7 @@ { "aboutUpdateURL": "https://apple.com", "requiredMinimumOSVersion": "14.6", - "requiredInstallationDate": "2024-01-01T00:00:00Z", + "requiredInstallationDate": "2025-01-01T00:00:00Z", "unsupportedURL": "https://google.com", } ], @@ -38,6 +38,7 @@ "screenShotDarkPath": "https://github.com/macadmins/nudge/blob/main/assets/standard_mode/demo_dark_1_icon.png?raw=true", "screenShotLightPath": "https://github.com/macadmins/nudge/blob/main/assets/standard_mode/demo_light_1_icon.png?raw=true", "applicationTerminatedNotificationImagePath": "/Library/Application Support/Nudge/logoLight.png", + "showRequiredInstallationDate": true, "simpleMode": false, "updateElements": [ { @@ -56,7 +57,7 @@ "mainContentText": "mainContentText", "mainContentTextUnsupported1": "mainContentTextUnsupported", "mainHeader": "mainHeader", - "mainHeaderUnsupported1": "mainHeaderUnsupported", + "mainHeaderUnsupported": "mainHeaderUnsupported", "oneDayDeferralButtonText": "oneDayDeferralButtonText", "oneHourDeferralButtonText": "oneHourDeferralButtonText", "primaryQuitButtonText": "primaryQuitButtonText", diff --git a/Localizable.xcstrings b/Localizable.xcstrings index a31290c6..4cc958c3 100644 --- a/Localizable.xcstrings +++ b/Localizable.xcstrings @@ -3125,6 +3125,114 @@ } } }, + "Required Date:" : { + "comment" : "Required Date:", + "extractionState" : "manual", + "localizations" : { + "da" : { + "stringUnit" : { + "state" : "translated", + "value" : "Påkrævet Dato:" + } + }, + "de" : { + "stringUnit" : { + "state" : "translated", + "value" : "Erforderliches Datum:" + } + }, + "en" : { + "stringUnit" : { + "state" : "translated", + "value" : "Required Date:" + } + }, + "es" : { + "stringUnit" : { + "state" : "translated", + "value" : "Fecha Requerida:" + } + }, + "fr" : { + "stringUnit" : { + "state" : "translated", + "value" : "Date Requise:" + } + }, + "hi" : { + "stringUnit" : { + "state" : "translated", + "value" : "तारीख चाहिए:" + } + }, + "it" : { + "stringUnit" : { + "state" : "translated", + "value" : "Data Richiesta:" + } + }, + "ja" : { + "stringUnit" : { + "state" : "translated", + "value" : "必要な日付:" + } + }, + "ko" : { + "stringUnit" : { + "state" : "translated", + "value" : "필수 날짜:" + } + }, + "nb" : { + "stringUnit" : { + "state" : "translated", + "value" : "Krevd Dato:" + } + }, + "nl" : { + "stringUnit" : { + "state" : "translated", + "value" : "Verplichte Datum:" + } + }, + "pl" : { + "stringUnit" : { + "state" : "translated", + "value" : "Wymagana Data:" + } + }, + "pt" : { + "stringUnit" : { + "state" : "translated", + "value" : "Data Requerida:" + } + }, + "ru" : { + "stringUnit" : { + "state" : "translated", + "value" : "Требуемая дата:" + } + }, + "sv" : { + "stringUnit" : { + "state" : "translated", + "value" : "Nödvändiga Datumet:" + } + }, + "uk" : { + "stringUnit" : { + "state" : "translated", + "value" : "Необхідна дата:" + } + }, + "zh" : { + "stringUnit" : { + "state" : "translated", + "value" : "要求日期:" + } + } + } + }, "Required OS Version:" : { "comment" : "Required OS Version:", "extractionState" : "manual", diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index b1916572..911dd300 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -504,6 +504,12 @@ struct UserInterfaceVariables { "Later" } + static var requiredInstallationDisplayFormat: String { + userInterfaceProfile?["requiredInstallationDisplayFormat"] as? String ?? + userInterfaceJSON?.requiredInstallationDisplayFormat ?? + "MM/dd/yyyy" + } + static var secondaryQuitButtonText: String { userInterfaceUpdateElementsProfile?["secondaryQuitButtonText"] as? String ?? userInterfaceUpdateElementsJSON?.secondaryQuitButtonText ?? @@ -522,6 +528,12 @@ struct UserInterfaceVariables { true } + static var showRequiredInstallationDate: Bool { + userInterfaceProfile?["showRequiredInstallationDate"] as? Bool ?? + userInterfaceJSON?.showRequiredInstallationDate ?? + false + } + static var singleQuitButton: Bool { userInterfaceProfile?["singleQuitButton"] as? Bool ?? userInterfaceJSON?.singleQuitButton ?? diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index 1c772b80..35aced37 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -402,8 +402,8 @@ extension UserExperience { struct UserInterface: Codable { var actionButtonPath, applicationTerminatedNotificationImagePath, fallbackLanguage: String? var forceFallbackLanguage, forceScreenShotIcon: Bool? - var iconDarkPath, iconLightPath, screenShotDarkPath, screenShotLightPath: String? - var showActivelyExploitedCVEs, showDeferralCount, simpleMode, singleQuitButton: Bool? + var iconDarkPath, iconLightPath, requiredInstallationDisplayFormat, screenShotDarkPath, screenShotLightPath: String? + var showActivelyExploitedCVEs, showDeferralCount, showRequiredInstallationDate, simpleMode, singleQuitButton: Bool? var updateElements: [UpdateElement]? } @@ -433,10 +433,12 @@ extension UserInterface { forceScreenShotIcon: Bool? = nil, iconDarkPath: String? = nil, iconLightPath: String? = nil, + requiredInstallationDisplayFormat: String? = nil, screenShotDarkPath: String? = nil, screenShotLightPath: String? = nil, showActivelyExploitedCVEs: Bool? = nil, showDeferralCount: Bool? = nil, + showRequiredInstallationDate: Bool? = nil, simpleMode: Bool? = nil, singleQuitButton: Bool? = nil, updateElements: [UpdateElement]? = nil @@ -449,10 +451,12 @@ extension UserInterface { forceScreenShotIcon: forceScreenShotIcon ?? self.forceScreenShotIcon, iconDarkPath: iconDarkPath ?? self.iconDarkPath, iconLightPath: iconLightPath ?? self.iconLightPath, + requiredInstallationDisplayFormat: requiredInstallationDisplayFormat ?? self.requiredInstallationDisplayFormat, screenShotDarkPath: screenShotDarkPath ?? self.screenShotDarkPath, screenShotLightPath: screenShotLightPath ?? self.screenShotLightPath, showActivelyExploitedCVEs: showActivelyExploitedCVEs ?? self.showActivelyExploitedCVEs, showDeferralCount: showDeferralCount ?? self.showDeferralCount, + showRequiredInstallationDate: showRequiredInstallationDate ?? self.showRequiredInstallationDate, simpleMode: simpleMode ?? self.simpleMode, singleQuitButton: singleQuitButton ?? self.singleQuitButton, updateElements: updateElements ?? self.updateElements diff --git a/Nudge/UI/StandardMode/LeftSide.swift b/Nudge/UI/StandardMode/LeftSide.swift index 0d5d858f..feeb799c 100644 --- a/Nudge/UI/StandardMode/LeftSide.swift +++ b/Nudge/UI/StandardMode/LeftSide.swift @@ -37,6 +37,9 @@ struct StandardModeLeftSide: View { private var informationStack: some View { VStack(alignment: .center, spacing: interLineSpacing) { InfoRow(label: "Required OS Version:", value: String(appState.requiredMinimumOSVersion), boldText: true) + if UserInterfaceVariables.showRequiredInstallationDate { + InfoRow(label: "Required Date:", value: DateManager().coerceDateToString(date: requiredInstallationDate, formatterString: UserInterfaceVariables.requiredInstallationDisplayFormat)) + } if OptionalFeatureVariables.utilizeSOFAFeed && UserInterfaceVariables.showActivelyExploitedCVEs { InfoRow(label: "Actively Exploited CVEs:", value: String(appState.activelyExploitedCVEs).capitalized, isHighlighted: appState.activelyExploitedCVEs ? true : false, boldText: appState.activelyExploitedCVEs) } @@ -82,12 +85,15 @@ struct InfoRow: View { Text(value) .foregroundColor(appState.differentiateWithoutColor ? .accessibleRed : .red) .fontWeight(.bold) + .minimumScaleFactor(0.01) } else { Text(value) .foregroundColor(colorScheme == .light ? .accessibleSecondaryLight : .accessibleSecondaryDark) .fontWeight(boldText ? .bold : .regular) + .minimumScaleFactor(0.01) } } + .lineLimit(1) } } diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 74380132..05e7d362 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -488,6 +488,12 @@ struct DateManager { return formatter }() + func coerceDateToString(date: Date, formatterString: String) -> String { + let formatter = DateFormatter() + formatter.dateFormat = formatterString + return formatter.string(from: date) + } + func coerceStringToDate(dateString: String) -> Date { dateFormatterISO8601.date(from: dateString) ?? getCurrentDate() } From 4b1f4dd39733cb5aedc562f1447970f0f9d8c310 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Fri, 31 May 2024 16:26:26 -0500 Subject: [PATCH 078/141] change key name to allow artificial nudge event delays --- CHANGELOG.md | 3 ++- .../Preferences/DefaultPreferencesNudge.swift | 6 +++--- Nudge/Preferences/PreferencesStructure.swift | 6 +++--- Nudge/UI/Main.swift | 6 +++--- Nudge/Utilities/Utils.swift | 19 ++++++++----------- 5 files changed, 19 insertions(+), 21 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index d3c93bc2..41ba76cd 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -21,7 +21,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - `nonActivelyExploitedCVEsSLA` under the `osVersionRequirement` key will default to 21 days - `standardInstallationSLA` under the `osVersionRequirement` key will default to 28 days - These dates are calculated against the `ReleaseDate` key in the SOFA feed - - To artificially change the `ReleaseDate` thereby giving your users a default grace period for all SOFA OS updates, please configure the `sofaPeriodLaunchDelay` key under `userExperience` + - To artificially delay the SOFA nudge events, see the details below for `nudgeEventLaunchDelay` - If you'd like to not have nudge events for releases without any known CVEs, please configure the `disableNudgeForStandardInstalls` key under `optionalFeatures` to true ### Fixed @@ -35,6 +35,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - Fixed errors when moving to Swift 5.10 ### Added +- To artificially change the `requredInstallationDate` thereby giving your users a default grace period for all Nudge events updates, please configure the `nudgeEventLaunchDelay` key under `userExperience` - A local image path can now be specified for the notification event when Nudge terminates and application - Please configure the `applicationTerminatedNotificationImagePath` key under `userInterface` - Due to limitations within Apple's API, a local path is only supported at this time diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 911dd300..43a7fd9e 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -364,9 +364,9 @@ struct UserExperienceVariables { false } - static var sofaPeriodLaunchDelay: Int { - userExperienceProfile?["sofaPeriodLaunchDelay"] as? Int ?? - userExperienceJSON?.sofaPeriodLaunchDelay ?? + static var nudgeEventLaunchDelay: Int { + userExperienceProfile?["nudgeEventLaunchDelay"] as? Int ?? + userExperienceJSON?.nudgeEventLaunchDelay ?? 0 } } diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index 35aced37..047d6404 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -324,7 +324,7 @@ struct UserExperience: Codable { var noTimers: Bool? var nudgeRefreshCycle: Int? var randomDelay: Bool? - var sofaPeriodLaunchDelay: Int? + var nudgeEventLaunchDelay: Int? } // MARK: UserExperience convenience initializers and mutators @@ -368,7 +368,7 @@ extension UserExperience { noTimers: Bool? = nil, nudgeRefreshCycle: Int? = nil, randomDelay: Bool? = nil, - sofaPeriodLaunchDelay: Int? = nil + nudgeEventLaunchDelay: Int? = nil ) -> UserExperience { return UserExperience( allowGracePeriods: allowGracePeriods ?? self.allowGracePeriods, @@ -393,7 +393,7 @@ extension UserExperience { noTimers: noTimers ?? self.noTimers, nudgeRefreshCycle: nudgeRefreshCycle ?? self.nudgeRefreshCycle, randomDelay: randomDelay ?? self.randomDelay, - sofaPeriodLaunchDelay: sofaPeriodLaunchDelay ?? self.sofaPeriodLaunchDelay + nudgeEventLaunchDelay: nudgeEventLaunchDelay ?? self.nudgeEventLaunchDelay ) } } diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index f6d0088d..c31aef6d 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -264,7 +264,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { handleCommandLineArguments() applyGracePeriodLogic() sofaPreLaunchLogic() - applySOFAPeriodLogic() + applydelayNudgeEventLogic() applyRandomDelayIfNecessary() updateNudgeState() handleSoftwareUpdateRequirements() @@ -340,8 +340,8 @@ class AppDelegate: NSObject, NSApplicationDelegate { } } - private func applySOFAPeriodLogic() { - _ = AppStateManager().sofaPeriodLogic() + private func applydelayNudgeEventLogic() { + _ = AppStateManager().delayNudgeEventLogic() if nudgePrimaryState.shouldExit { exit(0) } diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 05e7d362..59905716 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -176,18 +176,15 @@ struct AppStateManager { return calculateNewRequiredInstallationDateIfNeeded(currentDate: currentDate, gracePeriodPathCreationDate: gracePeriodPathCreationDate) } - func sofaPeriodLogic(currentDate: Date = DateManager().getCurrentDate(), testFileDate: Date? = nil) -> Date { - if OptionalFeatureVariables.utilizeSOFAFeed { - if releaseDate.addingTimeInterval(TimeInterval(UserExperienceVariables.sofaPeriodLaunchDelay * 86400)) > currentDate { - LogManager.info("Device within sofaPeriodLaunchDelay, exiting Nudge", logger: uiLog) - nudgePrimaryState.shouldExit = true - return currentDate - } else { - LogManager.info("Device outside sofaPeriodLaunchDelay", logger: uiLog) - return PrefsWrapper.requiredInstallationDate - } + func delayNudgeEventLogic(currentDate: Date = DateManager().getCurrentDate(), testFileDate: Date? = nil) -> Date { + if releaseDate.addingTimeInterval(TimeInterval(UserExperienceVariables.nudgeEventLaunchDelay * 86400)) > currentDate { + LogManager.info("Device within nudgeEventLaunchDelay, exiting Nudge", logger: uiLog) + nudgePrimaryState.shouldExit = true + return currentDate + } else { + LogManager.info("Device outside nudgeEventLaunchDelay", logger: uiLog) + return PrefsWrapper.requiredInstallationDate } - return PrefsWrapper.requiredInstallationDate } private func isDeferralAllowed(threshold: Int, logMessage: String) -> Bool { From 0add0657aa5dc03125db6834354dfbb82eaefe03 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 4 Jun 2024 14:51:56 -0500 Subject: [PATCH 079/141] fix bug in launch delay logic --- Nudge/Utilities/Utils.swift | 3 +++ 1 file changed, 3 insertions(+) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 59905716..3d3d36ff 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -177,6 +177,9 @@ struct AppStateManager { } func delayNudgeEventLogic(currentDate: Date = DateManager().getCurrentDate(), testFileDate: Date? = nil) -> Date { + if UserExperienceVariables.nudgeEventLaunchDelay == 0 { + return PrefsWrapper.requiredInstallationDate + } if releaseDate.addingTimeInterval(TimeInterval(UserExperienceVariables.nudgeEventLaunchDelay * 86400)) > currentDate { LogManager.info("Device within nudgeEventLaunchDelay, exiting Nudge", logger: uiLog) nudgePrimaryState.shouldExit = true From 9138fb8ab8bba46bffe046abe58e73ed574d8fe6 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Thu, 6 Jun 2024 10:46:47 -0500 Subject: [PATCH 080/141] add showDaysRemainingToUpdate rename showRequiredInstallationDate to howRequiredDate --- CHANGELOG.md | 4 +++- .../com.github.macadmins.Nudge.tester.json | 2 +- Nudge/Preferences/DefaultPreferencesNudge.swift | 12 +++++++++--- Nudge/Preferences/PreferencesStructure.swift | 8 +++++--- Nudge/UI/StandardMode/LeftSide.swift | 10 ++++++---- 5 files changed, 24 insertions(+), 12 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 41ba76cd..8021f798 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -23,6 +23,8 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - These dates are calculated against the `ReleaseDate` key in the SOFA feed - To artificially delay the SOFA nudge events, see the details below for `nudgeEventLaunchDelay` - If you'd like to not have nudge events for releases without any known CVEs, please configure the `disableNudgeForStandardInstalls` key under `optionalFeatures` to true +- You can now disable the `Days Remaining To Update:` item on the left side of the UI. + - Configure the `showDaysRemainingToUpdate` key under `userInterface` to false ### Fixed - `screenshotDisplay` view had a bug that may result in the screenshot being partially cut off or zoomable @@ -61,7 +63,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - `unsupportedURL` and `unsupportedURLs` can change the information button itself, but it will remain in the `osVersionRequirement` key with `unsupportedURLs` and `unsupportedURLs`. - An icon will appear as an overlay on top of the company image to further emphasize the device is no longer supported - An admin can now show the `requiredInstallationDate` as a item on the left side of nudge. - - To enable this, please configure the `showRequiredInstallationDate` key under `userInterface` to true + - To enable this, please configure the `showRequiredDate` key under `userInterface` to true - You can also expirement with the format of this date through the key `requiredInstallationDisplayFormat` under `userInterface` - Be aware that the format you desire may not look good on the UI. diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index a6dfcd32..bbb76397 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -38,7 +38,7 @@ "screenShotDarkPath": "https://github.com/macadmins/nudge/blob/main/assets/standard_mode/demo_dark_1_icon.png?raw=true", "screenShotLightPath": "https://github.com/macadmins/nudge/blob/main/assets/standard_mode/demo_light_1_icon.png?raw=true", "applicationTerminatedNotificationImagePath": "/Library/Application Support/Nudge/logoLight.png", - "showRequiredInstallationDate": true, + "showRequiredDate": true, "simpleMode": false, "updateElements": [ { diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 43a7fd9e..1f143d64 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -528,9 +528,15 @@ struct UserInterfaceVariables { true } - static var showRequiredInstallationDate: Bool { - userInterfaceProfile?["showRequiredInstallationDate"] as? Bool ?? - userInterfaceJSON?.showRequiredInstallationDate ?? + static var showDaysRemainingToUpdate: Bool { + userInterfaceProfile?["showDaysRemainingToUpdate"] as? Bool ?? + userInterfaceJSON?.showDaysRemainingToUpdate ?? + true + } + + static var showRequiredDate: Bool { + userInterfaceProfile?["showRequiredDate"] as? Bool ?? + userInterfaceJSON?.showRequiredDate ?? false } diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index 047d6404..1bce436c 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -403,7 +403,7 @@ struct UserInterface: Codable { var actionButtonPath, applicationTerminatedNotificationImagePath, fallbackLanguage: String? var forceFallbackLanguage, forceScreenShotIcon: Bool? var iconDarkPath, iconLightPath, requiredInstallationDisplayFormat, screenShotDarkPath, screenShotLightPath: String? - var showActivelyExploitedCVEs, showDeferralCount, showRequiredInstallationDate, simpleMode, singleQuitButton: Bool? + var showActivelyExploitedCVEs, showDeferralCount, showDaysRemainingToUpdate, showRequiredDate, simpleMode, singleQuitButton: Bool? var updateElements: [UpdateElement]? } @@ -438,7 +438,8 @@ extension UserInterface { screenShotLightPath: String? = nil, showActivelyExploitedCVEs: Bool? = nil, showDeferralCount: Bool? = nil, - showRequiredInstallationDate: Bool? = nil, + showDaysRemainingToUpdate: Bool? = nil, + showRequiredDate: Bool? = nil, simpleMode: Bool? = nil, singleQuitButton: Bool? = nil, updateElements: [UpdateElement]? = nil @@ -456,7 +457,8 @@ extension UserInterface { screenShotLightPath: screenShotLightPath ?? self.screenShotLightPath, showActivelyExploitedCVEs: showActivelyExploitedCVEs ?? self.showActivelyExploitedCVEs, showDeferralCount: showDeferralCount ?? self.showDeferralCount, - showRequiredInstallationDate: showRequiredInstallationDate ?? self.showRequiredInstallationDate, + showDaysRemainingToUpdate: showDaysRemainingToUpdate ?? self.showDaysRemainingToUpdate, + showRequiredDate: showRequiredDate ?? self.showRequiredDate, simpleMode: simpleMode ?? self.simpleMode, singleQuitButton: singleQuitButton ?? self.singleQuitButton, updateElements: updateElements ?? self.updateElements diff --git a/Nudge/UI/StandardMode/LeftSide.swift b/Nudge/UI/StandardMode/LeftSide.swift index feeb799c..3c429933 100644 --- a/Nudge/UI/StandardMode/LeftSide.swift +++ b/Nudge/UI/StandardMode/LeftSide.swift @@ -37,14 +37,16 @@ struct StandardModeLeftSide: View { private var informationStack: some View { VStack(alignment: .center, spacing: interLineSpacing) { InfoRow(label: "Required OS Version:", value: String(appState.requiredMinimumOSVersion), boldText: true) - if UserInterfaceVariables.showRequiredInstallationDate { + if UserInterfaceVariables.showRequiredDate { InfoRow(label: "Required Date:", value: DateManager().coerceDateToString(date: requiredInstallationDate, formatterString: UserInterfaceVariables.requiredInstallationDisplayFormat)) } if OptionalFeatureVariables.utilizeSOFAFeed && UserInterfaceVariables.showActivelyExploitedCVEs { InfoRow(label: "Actively Exploited CVEs:", value: String(appState.activelyExploitedCVEs).capitalized, isHighlighted: appState.activelyExploitedCVEs ? true : false, boldText: appState.activelyExploitedCVEs) } InfoRow(label: "Current OS Version:", value: GlobalVariables.currentOSVersion) - remainingTimeRow + if UserInterfaceVariables.showDaysRemainingToUpdate { + remainingTimeRow + } if UserInterfaceVariables.showDeferralCount { InfoRow(label: "Deferred Count:", value: String(appState.userDeferrals)) } @@ -54,7 +56,7 @@ struct StandardModeLeftSide: View { private var remainingTimeRow: some View { Group { - if shouldShowDaysRemaining { + if shouldshowDaysRemainingToUpdate { InfoRow(label: "Days Remaining To Update:", value: String(appState.daysRemaining), isHighlighted: 0 > appState.daysRemaining ? true : false) } else { InfoRow(label: "Hours Remaining To Update:", value: String(appState.hoursRemaining), isHighlighted: true) @@ -62,7 +64,7 @@ struct StandardModeLeftSide: View { } } - private var shouldShowDaysRemaining: Bool { + private var shouldshowDaysRemainingToUpdate: Bool { ((appState.daysRemaining > 0 || 0 > appState.hoursRemaining) && !CommandLineUtilities().demoModeEnabled()) || CommandLineUtilities().demoModeEnabled() } } From bfbbe7fd5c388ef224f8d2107c925cd53e527f69 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Thu, 6 Jun 2024 11:19:30 -0500 Subject: [PATCH 081/141] update changelog --- CHANGELOG.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 8021f798..130f0c88 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -20,7 +20,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - `activelyExploitedCVEsInstallationSLA` under the `osVersionRequirement` key will default to 14 days - `nonActivelyExploitedCVEsSLA` under the `osVersionRequirement` key will default to 21 days - `standardInstallationSLA` under the `osVersionRequirement` key will default to 28 days - - These dates are calculated against the `ReleaseDate` key in the SOFA feed + - These dates are calculated against the `ReleaseDate` key in the SOFA feed, which is UTC formatted. Local timezones will **not be supported** with the automatic sofa feed unless you use a custom feed and change this value yourself, following ISO-8601 date formats - To artificially delay the SOFA nudge events, see the details below for `nudgeEventLaunchDelay` - If you'd like to not have nudge events for releases without any known CVEs, please configure the `disableNudgeForStandardInstalls` key under `optionalFeatures` to true - You can now disable the `Days Remaining To Update:` item on the left side of the UI. @@ -54,7 +54,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - Utilizing Apple's markdown features, you can now utilize, bold, italic, underline, subscript and url links directly into any of the text fields - [SOFA](https://github.com/macadmins/sofa) feed support - Set the `utilizeSOFAFeed` key `true` under `optionalFeatures` to enable this feature - - Nudge will by default check the feed every 24 hours. + - Nudge will by default check the feed every 24 hours and save a cache file under `~/Library/Application Support/com.github.macadmins.Nudge/sofa-macos_data_feed.json` - In order to change this, please configure the `refreshSOFAFeedTime` key under `optionalFeatures` in seconds - If you are utilizing a custom sofa feed, please configure the `customSOFAFeedURL` key under `optionalFeatures` - "Unsupported device" UI in standard mode that utilizes the SOFA feed From 3bafde231f527ad32d0ed0486ff960e41ab9c861 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Thu, 13 Jun 2024 12:29:10 -0500 Subject: [PATCH 082/141] add some sequoia info --- Example Assets/com.github.macadmins.Nudge.tester.json | 2 +- Nudge/Preferences/DefaultPreferencesNudge.swift | 2 ++ Nudge/Utilities/Utils.swift | 2 ++ 3 files changed, 5 insertions(+), 1 deletion(-) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index bbb76397..6e005741 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -20,7 +20,7 @@ "osVersionRequirements": [ { "aboutUpdateURL": "https://apple.com", - "requiredMinimumOSVersion": "14.6", + "requiredMinimumOSVersion": "15.99", "requiredInstallationDate": "2025-01-01T00:00:00Z", "unsupportedURL": "https://google.com", } diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 1f143d64..fdfc7950 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -597,6 +597,7 @@ let builtInAcceptableApplicationBundleIDs = [ "com.apple.InstallAssistant.macOSMonterey", "com.apple.InstallAssistant.macOSVentura", "com.apple.InstallAssistant.macOSSonoma", + "com.apple.InstallAssistant.macOSSequoia", "com.apple.loginwindow", "com.apple.MobileAsset.MacSoftwareUpdate", "com.apple.ScreenSaver.Engine", @@ -609,6 +610,7 @@ let builtInAcceptableApplicationBundleIDs = [ "com.apple.InstallAssistant.macOSMonterey", "com.apple.InstallAssistant.macOSVentura", "com.apple.InstallAssistant.macOSSonoma", + "com.apple.InstallAssistant.macOSSequoia", "com.apple.loginwindow", "com.apple.MobileAsset.MacSoftwareUpdate", "com.apple.ScreenSaver.Engine", diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 3d3d36ff..8b335b85 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -976,6 +976,8 @@ struct NetworkFileManager { return "/Applications/Install macOS Ventura.app" case 14: return "/Applications/Install macOS Sonoma.app" + case 15: + return "/Applications/Install macOS Sequoia.app" default: return "/System/Library/CoreServices/Software Update.app" } From 6fa91df41b72afd73b709b50e222a0462a08db26 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 17 Jun 2024 12:06:37 -0500 Subject: [PATCH 083/141] add honorFocusModes key and logic --- .../com.github.macadmins.Nudge.tester.json | 1 + Nudge/Info.plist | 2 ++ Nudge/Nudge.entitlements | 4 ++-- Nudge/Preferences/DefaultPreferencesNudge.swift | 6 ++++++ Nudge/Preferences/PreferencesStructure.swift | 4 +++- Nudge/Utilities/Utils.swift | 17 +++++++++++++++++ 6 files changed, 31 insertions(+), 3 deletions(-) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index 6e005741..823d5ccf 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -14,6 +14,7 @@ "com.apple.Safari" ], "disableNudgeForStandardInstalls": true, + "honorFocusModes": true, "terminateApplicationsOnLaunch": true, "utilizeSOFAFeed": false }, diff --git a/Nudge/Info.plist b/Nudge/Info.plist index 9eee977e..8ae12fc5 100644 --- a/Nudge/Info.plist +++ b/Nudge/Info.plist @@ -24,5 +24,7 @@ $(MACOSX_DEPLOYMENT_TARGET) LSUIElement + NSFocusStatusUsageDescription + Obtains Focus status to determine if the user is in a Focus mode and honor its value. diff --git a/Nudge/Nudge.entitlements b/Nudge/Nudge.entitlements index e71e2ff7..92fa7c9b 100644 --- a/Nudge/Nudge.entitlements +++ b/Nudge/Nudge.entitlements @@ -2,11 +2,11 @@ - com.apple.security.get-task-allow - com.apple.security.app-sandbox com.apple.security.files.user-selected.read-write + com.apple.security.get-task-allow + diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index fdfc7950..89030b8a 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -155,6 +155,12 @@ struct OptionalFeatureVariables { true } + static var honorFocusModes: Bool { + optionalFeaturesProfile?["honorFocusModes"] as? Bool ?? + optionalFeaturesJSON?.honorFocusModes ?? + false + } + static var refreshSOFAFeedTime: Int { optionalFeaturesProfile?["refreshSOFAFeedTime"] as? Int ?? optionalFeaturesJSON?.refreshSOFAFeedTime ?? diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index 1bce436c..f1b14978 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -54,7 +54,7 @@ struct OptionalFeatures: Codable { var acceptableAssertionUsage, acceptableCameraUsage, acceptableScreenSharingUsage, aggressiveUserExperience, aggressiveUserFullScreenExperience, asynchronousSoftwareUpdate, attemptToBlockApplicationLaunches, attemptToCheckForSupportedDevice, attemptToFetchMajorUpgrade: Bool? var blockedApplicationBundleIDs: [String]? var customSOFAFeedURL: String? - var disableNudgeForStandardInstalls, disableSoftwareUpdateWorkflow, enforceMinorUpdates: Bool? + var disableNudgeForStandardInstalls, disableSoftwareUpdateWorkflow, enforceMinorUpdates, honorFocusModes: Bool? var refreshSOFAFeedTime: Int? var terminateApplicationsOnLaunch, utilizeSOFAFeed: Bool? } @@ -94,6 +94,7 @@ extension OptionalFeatures { disableNudgeForStandardInstalls: Bool? = nil, disableSoftwareUpdateWorkflow: Bool? = nil, enforceMinorUpdates: Bool? = nil, + honorFocusModes: Bool? = nil, refreshSOFAFeedTime: Int? = nil, terminateApplicationsOnLaunch: Bool? = nil, utilizeSOFAFeed: Bool? = nil @@ -115,6 +116,7 @@ extension OptionalFeatures { disableNudgeForStandardInstalls: disableNudgeForStandardInstalls ?? self.disableNudgeForStandardInstalls, disableSoftwareUpdateWorkflow: disableSoftwareUpdateWorkflow ?? self.disableSoftwareUpdateWorkflow, enforceMinorUpdates: enforceMinorUpdates ?? self.enforceMinorUpdates, + honorFocusModes: honorFocusModes ?? self.honorFocusModes, refreshSOFAFeedTime: refreshSOFAFeedTime ?? self.refreshSOFAFeedTime, terminateApplicationsOnLaunch: terminateApplicationsOnLaunch ?? self.terminateApplicationsOnLaunch, utilizeSOFAFeed: utilizeSOFAFeed ?? self.utilizeSOFAFeed diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 8b335b85..2bec5077 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -8,6 +8,7 @@ import AppKit import CoreMediaIO import Foundation +import Intents import IOKit #if canImport(ServiceManagement) import ServiceManagement @@ -15,8 +16,24 @@ import ServiceManagement import SwiftUI import SystemConfiguration + struct AppStateManager { func activateNudge() { + if OptionalFeatureVariables.honorFocusModes { + LogManager.info("honorFocusModes is configured - checking focus status", logger: utilsLog) + + // Request the current focus status + INFocusStatusCenter.default.requestAuthorization { status in + if status == .authorized { + if INFocusStatusCenter.default.focusStatus.isFocused == true { + LogManager.info("Device has focus modes set - bypassing activation event", logger: utilsLog) + return + } + } else { + LogManager.info("Focus status authorization not granted", logger: utilsLog) + } + } + } LogManager.info("Activating Nudge", logger: utilsLog) nudgePrimaryState.lastRefreshTime = DateManager().getCurrentDate() guard let mainWindow = NSApp.windows.first else { return } From fe28eda2608d2c212220422652a4cae0b69590c2 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 17 Jun 2024 12:20:21 -0500 Subject: [PATCH 084/141] disable honorFocusModes for now --- Nudge/Info.plist | 2 -- Nudge/Utilities/Utils.swift | 3 +++ 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/Nudge/Info.plist b/Nudge/Info.plist index 8ae12fc5..9eee977e 100644 --- a/Nudge/Info.plist +++ b/Nudge/Info.plist @@ -24,7 +24,5 @@ $(MACOSX_DEPLOYMENT_TARGET) LSUIElement - NSFocusStatusUsageDescription - Obtains Focus status to determine if the user is in a Focus mode and honor its value. diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 2bec5077..f942428b 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -20,9 +20,12 @@ import SystemConfiguration struct AppStateManager { func activateNudge() { if OptionalFeatureVariables.honorFocusModes { + LogManager.info("honorFocusModes is configured - but this feature is not being actively developed at this time", logger: utilsLog) + return LogManager.info("honorFocusModes is configured - checking focus status", logger: utilsLog) // Request the current focus status + // TODO: This will break Nudge unless you have NSFocusStatusUsageDescription in the Info.plist INFocusStatusCenter.default.requestAuthorization { status in if status == .authorized { if INFocusStatusCenter.default.focusStatus.isFocused == true { From 856db9284baab8a27fcee5d8d81756728cc53a1d Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 17 Jun 2024 13:16:35 -0500 Subject: [PATCH 085/141] try a different but less reliable method --- Nudge/Utilities/Utils.swift | 48 ++++++++++++++++++++++++------------- 1 file changed, 32 insertions(+), 16 deletions(-) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index f942428b..8a04ccfb 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -16,25 +16,13 @@ import ServiceManagement import SwiftUI import SystemConfiguration - struct AppStateManager { func activateNudge() { if OptionalFeatureVariables.honorFocusModes { - LogManager.info("honorFocusModes is configured - but this feature is not being actively developed at this time", logger: utilsLog) - return - LogManager.info("honorFocusModes is configured - checking focus status", logger: utilsLog) - - // Request the current focus status - // TODO: This will break Nudge unless you have NSFocusStatusUsageDescription in the Info.plist - INFocusStatusCenter.default.requestAuthorization { status in - if status == .authorized { - if INFocusStatusCenter.default.focusStatus.isFocused == true { - LogManager.info("Device has focus modes set - bypassing activation event", logger: utilsLog) - return - } - } else { - LogManager.info("Focus status authorization not granted", logger: utilsLog) - } + LogManager.info("honorFocusModes is configured - checking focus status. Warning: This feature may be unstable.", logger: utilsLog) + if isFocusModeEnabled() { + LogManager.info("Device has focus modes set - bypassing activation event", logger: utilsLog) + return } } LogManager.info("Activating Nudge", logger: utilsLog) @@ -222,6 +210,34 @@ struct AppStateManager { return isAllowed } + func isFocusModeEnabled() -> Bool { + let appID = "com.apple.controlcenter" as CFString + let key = "NSStatusItem Visible FocusModes" as CFString + let userName = kCFPreferencesCurrentUser + let hostName = kCFPreferencesAnyHost + + if let value = CFPreferencesCopyAppValue(key, appID) as? Bool { + return value + } else { + print("Key '\(key)' not found in preferences") + return false + } + + // + // // Request the current focus status + // // TODO: This will break Nudge unless you have NSFocusStatusUsageDescription in the Info.plist + // INFocusStatusCenter.default.requestAuthorization { status in + // if status == .authorized { + // if INFocusStatusCenter.default.focusStatus.isFocused == true { + // LogManager.info("Device has focus modes set - bypassing activation event", logger: utilsLog) + // return + // } + // } else { + // LogManager.info("Focus status authorization not granted", logger: utilsLog) + // } + // } + } + private func logOnce(_ message: String, state: inout Bool) { if !state { LogManager.info("\(message)", logger: uiLog) From 8707bd564efa2aedfad629ec832c39fa75ac0eed Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 17 Jun 2024 15:11:48 -0500 Subject: [PATCH 086/141] fix local and utc time --- Nudge/Preferences/PreferencesStructure.swift | 38 +++++++++++++++++--- Nudge/Utilities/Utils.swift | 12 ++++++- 2 files changed, 45 insertions(+), 5 deletions(-) diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index f1b14978..0aad29d4 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -18,8 +18,23 @@ struct NudgePreferences: Codable { extension NudgePreferences { init(data: Data) throws { let decoder = JSONDecoder() - decoder.dateDecodingStrategy = .iso8601 // Use ISO 8601 date format - self = try decoder.decode(NudgePreferences.self, from: data) + let dateFormatter = DateFormatter() + dateFormatter.dateFormat = "yyyy-MM-dd'T'HH:mm:ss" + // First attempt with ISO 8601 date format + do { + decoder.dateDecodingStrategy = .iso8601 + self = try decoder.decode(NudgePreferences.self, from: data) + return + } catch { + } + + // Second attempt with custom date format + do { + decoder.dateDecodingStrategy = .formatted(dateFormatter) + self = try decoder.decode(NudgePreferences.self, from: data) + return + } catch { + } } init(_ json: String, using encoding: String.Encoding = .utf8) throws { @@ -185,8 +200,23 @@ extension OSVersionRequirement { init(data: Data) throws { let decoder = JSONDecoder() - decoder.dateDecodingStrategy = .iso8601 // Use ISO 8601 date format - self = try decoder.decode(OSVersionRequirement.self, from: data) + let dateFormatter = DateFormatter() + dateFormatter.dateFormat = "yyyy-MM-dd'T'HH:mm:ss" + // First attempt with ISO 8601 date format + do { + decoder.dateDecodingStrategy = .iso8601 + self = try decoder.decode(OSVersionRequirement.self, from: data) + return + } catch { + } + + // Second attempt with custom date format + do { + decoder.dateDecodingStrategy = .formatted(dateFormatter) + self = try decoder.decode(OSVersionRequirement.self, from: data) + return + } catch { + } } init(_ json: String, using encoding: String.Encoding = .utf8) throws { diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 8a04ccfb..2c27f748 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -518,6 +518,12 @@ struct DateManager { return formatter }() + let dateFormatterLocalTime: DateFormatter = { + let formatter = DateFormatter() + formatter.dateFormat = "yyyy-MM-dd'T'HH:mm:ss" + return formatter + }() + private let dateFormatterCurrent: DateFormatter = { let formatter = DateFormatter() formatter.dateFormat = "yyyy-MM-dd HH:mm:ss Z" @@ -531,7 +537,11 @@ struct DateManager { } func coerceStringToDate(dateString: String) -> Date { - dateFormatterISO8601.date(from: dateString) ?? getCurrentDate() + if dateString.contains("Z") { + dateFormatterISO8601.date(from: dateString) ?? getCurrentDate() + } else { + dateFormatterLocalTime.date(from: dateString) ?? getCurrentDate() + } } func getCurrentDate() -> Date { From b8d81e222e08f19bd7043989d671b2f8360baa9b Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 17 Jun 2024 15:16:24 -0500 Subject: [PATCH 087/141] update changelog --- CHANGELOG.md | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 130f0c88..4bb9fa1d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -11,6 +11,10 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - Now built on Swift 5.10, Xcode 15.4 and macOS 14 - New Xcode Scheme `-bundle-mode-profile` to test profile logic - `-bundle-mode` has been renamed to `-bundle-mode-json` +- You can now pass two formats of **strings** to `requiredInstallationDate` + - `2025-01-01T00:00:00Z` for UTC + - `2025-01-01T00:00:00` for local time + - If you are using a MDM profile and passing the original `Date` key, you must change to utilizing `String` as Apple requires ISO8601 formatted dates - You can now pass the strings `latest`, `latest-supported` and `latest-minor` in the `requiredMinimumOSVersion` key - `latest`: always force latest release and if the machine can't this version, show the new "unsupported device" user interface - `latest-supported`: always get the latest version sofa shows that is supported by this device From 840e781df6806e1516a75f80096b32b1fe5f002c Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 17 Jun 2024 16:06:17 -0500 Subject: [PATCH 088/141] fix non gregorian requiredInstallationDate --- Nudge/Utilities/Utils.swift | 24 ++++++++++++++++-------- 1 file changed, 16 insertions(+), 8 deletions(-) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 2c27f748..1f8d81b0 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -524,12 +524,6 @@ struct DateManager { return formatter }() - private let dateFormatterCurrent: DateFormatter = { - let formatter = DateFormatter() - formatter.dateFormat = "yyyy-MM-dd HH:mm:ss Z" - return formatter - }() - func coerceDateToString(date: Date, formatterString: String) -> String { let formatter = DateFormatter() formatter.dateFormat = formatterString @@ -544,6 +538,20 @@ struct DateManager { } } + func convertToUserCalendar(date: Date) -> Date { + let userCalendar = Calendar.current + + // Get date components in the user's calendar + let components = userCalendar.dateComponents([.year, .month, .day, .hour, .minute, .second], from: date) + + // Create a new Date object in the user's calendar + if let userCalendarDate = userCalendar.date(from: components) { + return userCalendarDate + } else { + return date + } + } + func getCurrentDate() -> Date { switch Calendar.current.identifier { case .buddhist, .japanese, .gregorian, .coptic, .ethiopicAmeteMihret, .hebrew, .iso8601, .indian, .islamic, .islamicCivil, .islamicTabular, .islamicUmmAlQura, .persian: @@ -556,10 +564,10 @@ struct DateManager { func getFormattedDate(date: Date? = nil) -> Date { let initialDate = dateFormatterISO8601.date(from: dateFormatterISO8601.string(from: date ?? Date())) ?? Date() switch Calendar.current.identifier { - case .gregorian, .buddhist, .iso8601, .japanese: + case .gregorian: return initialDate default: - return dateFormatterCurrent.date(from: dateFormatterISO8601.string(from: initialDate)) ?? Date() + return convertToUserCalendar(date: initialDate) } } From 175549c475606cd72bc0a052688f2bd1e4c3c85f Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 17 Jun 2024 16:08:27 -0500 Subject: [PATCH 089/141] update changelog for non gregorian calendars --- CHANGELOG.md | 1 + 1 file changed, 1 insertion(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4bb9fa1d..f7ec8482 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -39,6 +39,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - More descriptive logs when loading json/mdm profile keys - Refactor portions of the `softwareupdate` logic to reduce potential errors - Fixed errors when moving to Swift 5.10 +- Fixed wrong `requiredInstallationDate` calculations when using [Non-Gregorian calendars](https://github.com/macadmins/nudge/issues/509) ### Added - To artificially change the `requredInstallationDate` thereby giving your users a default grace period for all Nudge events updates, please configure the `nudgeEventLaunchDelay` key under `userExperience` From b473b1fc298f8d2efe598ab6979406cdbf18972b Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 26 Jun 2024 11:34:11 -0500 Subject: [PATCH 090/141] move to multiple hardware checks --- .../com.github.macadmins.Nudge.tester.json | 3 +- Nudge/UI/Defaults.swift | 2 +- Nudge/UI/Main.swift | 8 ++++-- Nudge/Utilities/Utils.swift | 28 +++++++------------ 4 files changed, 17 insertions(+), 24 deletions(-) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index 823d5ccf..8a258cc3 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -15,8 +15,7 @@ ], "disableNudgeForStandardInstalls": true, "honorFocusModes": true, - "terminateApplicationsOnLaunch": true, - "utilizeSOFAFeed": false + "terminateApplicationsOnLaunch": true }, "osVersionRequirements": [ { diff --git a/Nudge/UI/Defaults.swift b/Nudge/UI/Defaults.swift index 0dc8e5cf..bcf3cdc6 100644 --- a/Nudge/UI/Defaults.swift +++ b/Nudge/UI/Defaults.swift @@ -29,7 +29,7 @@ struct Globals { // Device Properties static let gdmfAssets = NetworkFileManager().getGDMFAssets() static let sofaAssets = NetworkFileManager().getSOFAAssets() - static let hardwareModelID = DeviceManager().getHardwareModelID() + static let hardwareModelIDs = DeviceManager().getHardwareModelIDs() } struct Intervals { diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index c31aef6d..716386e3 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -192,7 +192,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { } else if PrefsWrapper.requiredMinimumOSVersion == "latest-supported" { if OptionalFeatureVariables.attemptToCheckForSupportedDevice { selectedOS = osVersion.securityReleases.first - if !selectedOS!.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }) { + if !selectedOS!.supportedDevices.contains(where: { supportedDevice in Globals.hardwareModelIDs.contains { $0.uppercased() == supportedDevice.uppercased() } }) { continue } } else { @@ -238,8 +238,10 @@ class AppDelegate: NSObject, NSApplicationDelegate { LogManager.notice("SOFA CVEs: \(selectedOS!.cves)", logger: sofaLog) if OptionalFeatureVariables.attemptToCheckForSupportedDevice { - LogManager.notice("Assessed Model ID: \(Globals.hardwareModelID)", logger: sofaLog) - let deviceMatchFound = selectedOS!.supportedDevices.contains(where: { $0.uppercased() == Globals.hardwareModelID.uppercased() }) + LogManager.notice("Assessed Model IDs: \(Globals.hardwareModelIDs)", logger: sofaLog) + let deviceMatchFound = selectedOS!.supportedDevices.contains { supportedDevice in + Globals.hardwareModelIDs.contains { $0.uppercased() == supportedDevice.uppercased() } + } LogManager.notice("Assessed Model ID found in SOFA Entry: \(deviceMatchFound)", logger: sofaLog) nudgePrimaryState.deviceSupportedByOSVersion = deviceMatchFound } diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 1f8d81b0..13a1c726 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -648,25 +648,17 @@ struct DeviceManager { getSysctlValue(for: "hw.model") ?? "" } - func getHardwareModelID() -> String { - var hardwareModelID = "" - if DeviceManager().getCPUTypeString() == "Apple Silicon" { - // There is no local bridge - hardwareModelID = getIORegInfo(serviceTarget: "target-sub-type") ?? "Unknown" - } else { - // Attempt localbridge for T2, if it fails, it's likely a T1 or lower - let bridgeID = getBridgeModelID() - let boardID = getIORegInfo(serviceTarget: "board-id") - if bridgeID.isEmpty { - // Fallback to boardID for T1 - hardwareModelID = boardID ?? "Unknown" - } else { - // T2 uses bridge ID for it's update brain via gdmf - hardwareModelID = bridgeID - } - } + func getHardwareModelIDs() -> [String] { + let boardID = getIORegInfo(serviceTarget: "board-id") ?? "Unknown" + let bridgeID = getBridgeModelID() + let hardwareModelID = getIORegInfo(serviceTarget: "target-sub-type") ?? "Unknown" + + LogManager.debug("Hardware Board ID: \(boardID)", logger: utilsLog) + LogManager.debug("Hardware Bridge ID: \(bridgeID)", logger: utilsLog) LogManager.debug("Hardware Model ID: \(hardwareModelID)", logger: utilsLog) - return hardwareModelID.trimmingCharacters(in: .whitespacesAndNewlines) + + + return [boardID.trimmingCharacters(in: .whitespacesAndNewlines), bridgeID.trimmingCharacters(in: .whitespacesAndNewlines), hardwareModelID.trimmingCharacters(in: .whitespacesAndNewlines)] } func getHardwareUUID() -> String { From 3cf7b62e07ad85ade2fe0cf83a4273223830d118 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 26 Jun 2024 11:34:55 -0500 Subject: [PATCH 091/141] utilizeSOFAFeed is not opt-out --- Nudge/Preferences/DefaultPreferencesNudge.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 89030b8a..41f6abea 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -176,7 +176,7 @@ struct OptionalFeatureVariables { static var utilizeSOFAFeed: Bool { optionalFeaturesProfile?["utilizeSOFAFeed"] as? Bool ?? optionalFeaturesJSON?.utilizeSOFAFeed ?? - false + true } } From b27875e937067c53a6c446460f91b1928231bc91 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 26 Jun 2024 11:36:23 -0500 Subject: [PATCH 092/141] changelog utilizeSOFAFeed per opt-out --- CHANGELOG.md | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index f7ec8482..91439a55 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -19,7 +19,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - `latest`: always force latest release and if the machine can't this version, show the new "unsupported device" user interface - `latest-supported`: always get the latest version sofa shows that is supported by this device - `latest-minor`: stay in the current major release and get the latest minor updates available - - This requires utilizing the SOFA feed features to properly work + - This requires utilizing the SOFA feed features to properly work, which is opt-out by default - Nudge will then utilize two date integers to automatically calculate the `requiredInstallationDate` - `activelyExploitedCVEsInstallationSLA` under the `osVersionRequirement` key will default to 14 days - `nonActivelyExploitedCVEsSLA` under the `osVersionRequirement` key will default to 21 days @@ -58,7 +58,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - Basic SwiftUI support for Markdown text options - Utilizing Apple's markdown features, you can now utilize, bold, italic, underline, subscript and url links directly into any of the text fields - [SOFA](https://github.com/macadmins/sofa) feed support - - Set the `utilizeSOFAFeed` key `true` under `optionalFeatures` to enable this feature + - Set the `utilizeSOFAFeed` key `false` under `optionalFeatures` to disable this feature - Nudge will by default check the feed every 24 hours and save a cache file under `~/Library/Application Support/com.github.macadmins.Nudge/sofa-macos_data_feed.json` - In order to change this, please configure the `refreshSOFAFeedTime` key under `optionalFeatures` in seconds - If you are utilizing a custom sofa feed, please configure the `customSOFAFeedURL` key under `optionalFeatures` From 98e177ae7eb6afeb8cea75aa7c78817dcf9ae5d1 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 26 Jun 2024 15:24:40 -0500 Subject: [PATCH 093/141] fix typos --- CHANGELOG.md | 2 +- Schema/jamf/com.github.macadmins.Nudge.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 91439a55..6bbc3947 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -265,7 +265,7 @@ Almost all of these changes were sent by others. Thank you for continuing to sup - `attemptToBlockApplicationLaunches` - When enabled, Nudge will attempt to block application launches after the required installation date. This key must be used in conjunction with `blockedApplicationBundleIDs`. - `blockedApplicationBundleIDs` - - The application Bundle ID which Nudge disallows from lauching after the required installation date. You can specify one or more Bundle ID. + - The application Bundle ID which Nudge disallows from launching after the required installation date. You can specify one or more Bundle ID. - `terminateApplicationsOnLaunch` - When enabled, Nudge will terminate the applications listed in blockedApplicationBundleIDs upon initial launch. diff --git a/Schema/jamf/com.github.macadmins.Nudge.json b/Schema/jamf/com.github.macadmins.Nudge.json index c5096efe..58437e20 100644 --- a/Schema/jamf/com.github.macadmins.Nudge.json +++ b/Schema/jamf/com.github.macadmins.Nudge.json @@ -171,7 +171,7 @@ ] }, "blockedApplicationBundleIDs": { - "description": "The application Bundle ID which Nudge disallows from lauching after the required installation date. (You can specify one or more Bundle ID.)", + "description": "The application Bundle ID which Nudge disallows from launching after the required installation date. (You can specify one or more Bundle ID.)", "anyOf": [ { "title": "Not Configured", From bb6dee953e59f7d30730b00b7e457d7614ea21f0 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 26 Jun 2024 15:56:55 -0500 Subject: [PATCH 094/141] add honorCycleTimersOnExit support --- CHANGELOG.md | 3 +++ Nudge/Preferences/DefaultPreferencesNudge.swift | 6 ++++++ Nudge/Preferences/PreferencesStructure.swift | 4 +++- Nudge/UI/Common/QuitButtons.swift | 6 +++++- 4 files changed, 17 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 6bbc3947..91ec325f 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -71,6 +71,9 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - To enable this, please configure the `showRequiredDate` key under `userInterface` to true - You can also expirement with the format of this date through the key `requiredInstallationDisplayFormat` under `userInterface` - Be aware that the format you desire may not look good on the UI. +- Nudge can now honor the current cycle timers when user's press the `Quit` button. + - Set the `honorCycleTimersOnExit` key to `true` under `optionalFeatures` to enable this feature + - [Issue 548](https://github.com/macadmins/nudge/issues/548) ## [1.1.16] - 2024-03-13 This will be the **final Nudge release** for macOS 11 and potentially other versions of macOS. diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 41f6abea..5c90a97c 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -161,6 +161,12 @@ struct OptionalFeatureVariables { false } + static var honorCycleTimersOnExit: Bool { + optionalFeaturesProfile?["honorCycleTimersOnExit"] as? Bool ?? + optionalFeaturesJSON?.honorCycleTimersOnExit ?? + false + } + static var refreshSOFAFeedTime: Int { optionalFeaturesProfile?["refreshSOFAFeedTime"] as? Int ?? optionalFeaturesJSON?.refreshSOFAFeedTime ?? diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index 0aad29d4..e94218a6 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -69,7 +69,7 @@ struct OptionalFeatures: Codable { var acceptableAssertionUsage, acceptableCameraUsage, acceptableScreenSharingUsage, aggressiveUserExperience, aggressiveUserFullScreenExperience, asynchronousSoftwareUpdate, attemptToBlockApplicationLaunches, attemptToCheckForSupportedDevice, attemptToFetchMajorUpgrade: Bool? var blockedApplicationBundleIDs: [String]? var customSOFAFeedURL: String? - var disableNudgeForStandardInstalls, disableSoftwareUpdateWorkflow, enforceMinorUpdates, honorFocusModes: Bool? + var disableNudgeForStandardInstalls, disableSoftwareUpdateWorkflow, enforceMinorUpdates, honorFocusModes, honorCycleTimersOnExit: Bool? var refreshSOFAFeedTime: Int? var terminateApplicationsOnLaunch, utilizeSOFAFeed: Bool? } @@ -110,6 +110,7 @@ extension OptionalFeatures { disableSoftwareUpdateWorkflow: Bool? = nil, enforceMinorUpdates: Bool? = nil, honorFocusModes: Bool? = nil, + honorCycleTimersOnExit: Bool? = nil, refreshSOFAFeedTime: Int? = nil, terminateApplicationsOnLaunch: Bool? = nil, utilizeSOFAFeed: Bool? = nil @@ -132,6 +133,7 @@ extension OptionalFeatures { disableSoftwareUpdateWorkflow: disableSoftwareUpdateWorkflow ?? self.disableSoftwareUpdateWorkflow, enforceMinorUpdates: enforceMinorUpdates ?? self.enforceMinorUpdates, honorFocusModes: honorFocusModes ?? self.honorFocusModes, + honorCycleTimersOnExit: honorCycleTimersOnExit ?? self.honorCycleTimersOnExit, refreshSOFAFeedTime: refreshSOFAFeedTime ?? self.refreshSOFAFeedTime, terminateApplicationsOnLaunch: terminateApplicationsOnLaunch ?? self.terminateApplicationsOnLaunch, utilizeSOFAFeed: utilizeSOFAFeed ?? self.utilizeSOFAFeed diff --git a/Nudge/UI/Common/QuitButtons.swift b/Nudge/UI/Common/QuitButtons.swift index 4c200d7a..357d5f13 100644 --- a/Nudge/UI/Common/QuitButtons.swift +++ b/Nudge/UI/Common/QuitButtons.swift @@ -106,7 +106,11 @@ struct QuitButtons: View { private func standardDeferralAction() { appState.nudgeEventDate = DateManager().getCurrentDate() - UIUtilities().setDeferralTime(deferralTime: appState.nudgeEventDate) + if OptionalFeatureVariables.honorCycleTimersOnExit { + UIUtilities().setDeferralTime(deferralTime: appState.nudgeEventDate.addingTimeInterval(TimeInterval(nudgePrimaryState.timerCycle))) + } else { + UIUtilities().setDeferralTime(deferralTime: appState.nudgeEventDate) + } updateDeferralUI() } From 75870f19bea124743734a386306c44235277606c Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 26 Jun 2024 15:57:04 -0500 Subject: [PATCH 095/141] fix some localization issues --- Nudge/UI/Common/QuitButtons.swift | 10 ++++------ 1 file changed, 4 insertions(+), 6 deletions(-) diff --git a/Nudge/UI/Common/QuitButtons.swift b/Nudge/UI/Common/QuitButtons.swift index 357d5f13..006628c9 100644 --- a/Nudge/UI/Common/QuitButtons.swift +++ b/Nudge/UI/Common/QuitButtons.swift @@ -59,13 +59,13 @@ struct QuitButtons: View { private var deferralOptions: some View { Group { if UserExperienceVariables.allowLaterDeferralButton { - deferralButton(title: UserInterfaceVariables.primaryQuitButtonText, action: standardDeferralAction) + deferralButton(title: UserInterfaceVariables.primaryQuitButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)), action: standardDeferralAction) } if AppStateManager().allow1HourDeferral() { - deferralButton(title: UserInterfaceVariables.oneHourDeferralButtonText, action: { deferAction(by: Intervals.hourTimeInterval) }) + deferralButton(title: UserInterfaceVariables.oneHourDeferralButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)), action: { deferAction(by: Intervals.hourTimeInterval) }) } if AppStateManager().allow24HourDeferral() { - deferralButton(title: UserInterfaceVariables.oneDayDeferralButtonText, action: { deferAction(by: Intervals.dayTimeInterval) }) + deferralButton(title: UserInterfaceVariables.oneDayDeferralButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)), action: { deferAction(by: Intervals.dayTimeInterval) }) } if AppStateManager().allowCustomDeferral() { customDeferralButton @@ -115,9 +115,7 @@ struct QuitButtons: View { } private var standardQuitButton: some View { - Button(action: UIUtilities().userInitiatedExit) { - Text(UserInterfaceVariables.primaryQuitButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) - } + deferralButton(title: UserInterfaceVariables.primaryQuitButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)), action: standardDeferralAction) } private func updateDeferralUI() { From d0f002185c78962b75230234a140f08443c19a5c Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 1 Jul 2024 15:29:25 -0500 Subject: [PATCH 096/141] add additional support for not exiting when you can't fetch the upgrade --- Nudge/UI/Main.swift | 10 +++++++--- 1 file changed, 7 insertions(+), 3 deletions(-) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 716386e3..d1d01796 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -500,9 +500,13 @@ class AppDelegate: NSObject, NSApplicationDelegate { private func handleAttemptToFetchMajorUpgrade() { if GlobalVariables.fetchMajorUpgradeSuccessful == false && !majorUpgradeAppPathExists && !majorUpgradeBackupAppPathExists { - LogManager.error("Unable to fetch major upgrade and application missing, exiting Nudge", logger: uiLog) - nudgePrimaryState.shouldExit = true - exit(1) + if VersionManager.versionGreaterThan(currentVersion: GlobalVariables.currentOSVersion, newVersion: "12.3") { + LogManager.info("Unable to fetch major upgrade and application missing, but macOS 12.3 and higher support delta major upgrades. Using new logic.", logger: uiLog) + } else { + LogManager.error("Unable to fetch major upgrade and application missing, exiting Nudge", logger: uiLog) + nudgePrimaryState.shouldExit = true + exit(1) + } } } From 54ee10e3887c7fb19c0bff1c16a6b2e108014911 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 1 Jul 2024 15:33:47 -0500 Subject: [PATCH 097/141] update changelog --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 91ec325f..13497ea8 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -74,6 +74,8 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - Nudge can now honor the current cycle timers when user's press the `Quit` button. - Set the `honorCycleTimersOnExit` key to `true` under `optionalFeatures` to enable this feature - [Issue 548](https://github.com/macadmins/nudge/issues/548) +- When the device is running macOS 12.3 or higher, Nudge uses the delta logic for macOS Upgrades + - [Issue 417](https://github.com/macadmins/nudge/issues/417) ## [1.1.16] - 2024-03-13 This will be the **final Nudge release** for macOS 11 and potentially other versions of macOS. From 921a087d5b55431449c0a1d515e7bc7dac1811af Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 1 Jul 2024 16:41:54 -0500 Subject: [PATCH 098/141] return gestalt information --- Nudge/Utilities/Utils.swift | 28 ++++++++++++++++++++++++++-- 1 file changed, 26 insertions(+), 2 deletions(-) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 13a1c726..7617871d 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -652,13 +652,14 @@ struct DeviceManager { let boardID = getIORegInfo(serviceTarget: "board-id") ?? "Unknown" let bridgeID = getBridgeModelID() let hardwareModelID = getIORegInfo(serviceTarget: "target-sub-type") ?? "Unknown" + let gestaltModelStringID = getKeyResultFromGestalt("HWModelStr") LogManager.debug("Hardware Board ID: \(boardID)", logger: utilsLog) LogManager.debug("Hardware Bridge ID: \(bridgeID)", logger: utilsLog) LogManager.debug("Hardware Model ID: \(hardwareModelID)", logger: utilsLog) + LogManager.debug("Gestalt Hardware Model ID: \(gestaltModelStringID)", logger: utilsLog) - - return [boardID.trimmingCharacters(in: .whitespacesAndNewlines), bridgeID.trimmingCharacters(in: .whitespacesAndNewlines), hardwareModelID.trimmingCharacters(in: .whitespacesAndNewlines)] + return [boardID.trimmingCharacters(in: .whitespacesAndNewlines), bridgeID.trimmingCharacters(in: .whitespacesAndNewlines), hardwareModelID.trimmingCharacters(in: .whitespacesAndNewlines), gestaltModelStringID.trimmingCharacters(in: .whitespacesAndNewlines)] } func getHardwareUUID() -> String { @@ -706,6 +707,29 @@ struct DeviceManager { } } + func getKeyResultFromGestalt(_ keyname: String) -> String { + let handle = dlopen("/usr/lib/libMobileGestalt.dylib", RTLD_NOW) + guard handle != nil else { + return "Unknown" + } + defer { + dlclose(handle) + } + + let symbol = dlsym(handle, "MGGetStringAnswer") + guard symbol != nil else { + return "Unknown" + } + + let function = unsafeBitCast(symbol, to: (@convention(c) (String) -> String?).self) + + guard let result = function(keyname) else { + return "Unknown" + } + + return result + } + func getPatchOSVersion() -> Int { let PatchOSVersion = ProcessInfo().operatingSystemVersion.patchVersion LogManager.info("Patch OS Version: \(PatchOSVersion)", logger: utilsLog) From 45362aef7a58dafb24374ceb3e1d29b87237d48f Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 2 Jul 2024 10:23:31 -0500 Subject: [PATCH 099/141] add logic for acceptableUpdatePreparingUsage --- CHANGELOG.md | 4 + .../Preferences/DefaultPreferencesNudge.swift | 6 ++ Nudge/Preferences/PreferencesStructure.swift | 4 +- Nudge/Utilities/UILogic.swift | 96 +++++++++++++++++++ 4 files changed, 109 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 13497ea8..66826ad2 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -76,6 +76,10 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - [Issue 548](https://github.com/macadmins/nudge/issues/548) - When the device is running macOS 12.3 or higher, Nudge uses the delta logic for macOS Upgrades - [Issue 417](https://github.com/macadmins/nudge/issues/417) +- Nudge can now bypass activations and re-activations when a macOS update is `Downloading`, `Preparing` or `Staged` for installation. + - To enable this, please configure the `acceptableUpdatePreparingUsage` key under `optionalFeatures` to true + - Be aware that the current logic used for this **cannot differentiate** when an update has completed preparing and is in the `Staged` phase, waiting for a user to reboot. This is due to an Apple process staying in memory. This will result in a reduction in Nudge re-activations + - Issue [555](https://github.com/macadmins/nudge/issues/555) ## [1.1.16] - 2024-03-13 This will be the **final Nudge release** for macOS 11 and potentially other versions of macOS. diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 5c90a97c..870cfb6b 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -82,6 +82,12 @@ struct OptionalFeatureVariables { optionalFeaturesJSON?.acceptableCameraUsage ?? false } + + static var acceptableUpdatePreparingUsage: Bool { + optionalFeaturesProfile?["acceptableUpdatePreparingUsage"] as? Bool ?? + optionalFeaturesJSON?.acceptableUpdatePreparingUsage ?? + false + } static var acceptableScreenSharingUsage: Bool { optionalFeaturesProfile?["acceptableScreenSharingUsage"] as? Bool ?? diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index e94218a6..c564e58e 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -66,7 +66,7 @@ extension NudgePreferences { // MARK: - OptionalFeatures struct OptionalFeatures: Codable { var acceptableApplicationBundleIDs, acceptableAssertionApplicationNames: [String]? - var acceptableAssertionUsage, acceptableCameraUsage, acceptableScreenSharingUsage, aggressiveUserExperience, aggressiveUserFullScreenExperience, asynchronousSoftwareUpdate, attemptToBlockApplicationLaunches, attemptToCheckForSupportedDevice, attemptToFetchMajorUpgrade: Bool? + var acceptableAssertionUsage, acceptableCameraUsage, acceptableUpdatePreparingUsage, acceptableScreenSharingUsage, aggressiveUserExperience, aggressiveUserFullScreenExperience, asynchronousSoftwareUpdate, attemptToBlockApplicationLaunches, attemptToCheckForSupportedDevice, attemptToFetchMajorUpgrade: Bool? var blockedApplicationBundleIDs: [String]? var customSOFAFeedURL: String? var disableNudgeForStandardInstalls, disableSoftwareUpdateWorkflow, enforceMinorUpdates, honorFocusModes, honorCycleTimersOnExit: Bool? @@ -97,6 +97,7 @@ extension OptionalFeatures { acceptableAssertionApplicationNames: [String]? = nil, acceptableAssertionUsage: Bool? = nil, acceptableCameraUsage: Bool? = nil, + acceptableUpdatePreparingUsage: Bool? = nil, acceptableScreenSharingUsage: Bool? = nil, aggressiveUserExperience: Bool? = nil, aggressiveUserFullScreenExperience: Bool? = nil, @@ -120,6 +121,7 @@ extension OptionalFeatures { acceptableAssertionApplicationNames: acceptableAssertionApplicationNames ?? self.acceptableAssertionApplicationNames, acceptableAssertionUsage: acceptableAssertionUsage ?? self.acceptableAssertionUsage, acceptableCameraUsage: acceptableCameraUsage ?? self.acceptableCameraUsage, + acceptableUpdatePreparingUsage: acceptableUpdatePreparingUsage ?? self.acceptableUpdatePreparingUsage, acceptableScreenSharingUsage: acceptableScreenSharingUsage ?? self.acceptableScreenSharingUsage, aggressiveUserExperience: aggressiveUserExperience ?? self.aggressiveUserExperience, aggressiveUserFullScreenExperience: aggressiveUserFullScreenExperience ?? self.aggressiveUserFullScreenExperience, diff --git a/Nudge/Utilities/UILogic.swift b/Nudge/Utilities/UILogic.swift index 8414b272..cd2e2d9b 100644 --- a/Nudge/Utilities/UILogic.swift +++ b/Nudge/Utilities/UILogic.swift @@ -6,10 +6,17 @@ // import AppKit +import Darwin import Foundation import IOKit.pwr_mgt // Asertions import SwiftUI +struct ProcessInfoStruct { + var pid: Int32 + var command: String + var arguments: [String] +} + func initialLaunchLogic() { guard !CommandLineUtilities().unitTestingEnabled() else { LogManager.debug("App being ran in test mode", logger: uiLog) @@ -177,6 +184,88 @@ private func logDeferralStates() { LoggerUtilities().logUserDeferrals() } +func getAllProcesses() -> [ProcessInfoStruct] { + var processes = [ProcessInfoStruct]() + + // Get the number of processes + var mib: [Int32] = [CTL_KERN, KERN_PROC, KERN_PROC_ALL] + var size = 0 + sysctl(&mib, u_int(mib.count), nil, &size, nil, 0) + + let processCount = size / MemoryLayout.size + var processList = [kinfo_proc](repeating: kinfo_proc(), count: processCount) + + // Get the list of processes + sysctl(&mib, u_int(mib.count), &processList, &size, nil, 0) + + // Extract process info + for process in processList { + let command = withUnsafePointer(to: process.kp_proc.p_comm) { + $0.withMemoryRebound(to: CChar.self, capacity: Int(MAXCOMLEN)) { + String(cString: $0) + } + } + let pid = process.kp_proc.p_pid + let arguments = getArgumentsForPID(pid: pid) + processes.append(ProcessInfoStruct(pid: pid, command: command, arguments: arguments)) + } + + return processes +} + +func getArgumentsForPID(pid: Int32) -> [String] { + var args = [String]() + + var mib: [Int32] = [CTL_KERN, KERN_PROCARGS2, pid] + var size = 0 + sysctl(&mib, u_int(mib.count), nil, &size, nil, 0) + + var buffer = [CChar](repeating: 0, count: size) + sysctl(&mib, u_int(mib.count), &buffer, &size, nil, 0) + + // Convert buffer to a string with proper bounds checking + let bufferString = String(bytesNoCopy: &buffer, length: size, encoding: .ascii, freeWhenDone: false) + + // Split the string into arguments + if let bufferString = bufferString { + args = bufferString.split(separator: "\0").map { String($0) } + } + + // Drop the first element which is the full path to the executable + if !args.isEmpty { + args.removeFirst() + } + + return args +} + +func isAnyProcessRunning(commandsWithArgs: [(commandPattern: String, arguments: [String]?)]) -> Bool { + let processes = getAllProcesses() + for (commandPattern, arguments) in commandsWithArgs { + let matchingProcesses = processes.filter { process in + fnmatch(commandPattern, process.command, 0) == 0 && + (arguments == nil || arguments!.allSatisfy { arg in + process.arguments.contains(where: { $0.contains(arg) }) + }) + } + if !matchingProcesses.isEmpty { + return true + } + } + return false +} + +func isDownloadingOrPreparingSoftwareUpdate() -> Bool { + let commandsWithArgs: [(commandPattern: String, arguments: [String]?)] = [ + ("softwareupdated", ["/System/Library/PrivateFrameworks/MobileSoftwareUpdate.framework/Support/softwareupdated"]), // When downloading a minor update, this process is running. + ("installcoordinationd", ["/System/Library/PrivateFrameworks/InstallCoordination.framework/Support/installcoordinationd"]), // When preparing a minor update, this process is running. Unfortunately, after preparing the update, this process appears to stay running. + ("softwareupdate", ["/usr/bin/softwareupdate", "--fetch-full-installer"]), // When downloading a major upgrade via SoftwareUpdate prefpane, it triggers a --fetch-full-installer run. Nudge also performs this method. + ("osinstallersetupd" ,["/Applications/*Install macOS *.app/Contents/Frameworks/OSInstallerSetup.framework/Resources/osinstallersetupd"]), // When installing a major upgrade, this process is running. + // /System/Library/PrivateFrameworks/PackageKit.framework/Resources/installd||system_installd - system_installd may be interesting, but I think installd is being used for any package + ] + return isAnyProcessRunning(commandsWithArgs: commandsWithArgs) +} + func needToActivateNudge() -> Bool { if NSApplication.shared.isActive && nudgeLogState.afterFirstLaunch { LogManager.notice("Nudge is currently the frontmostApplication", logger: uiLog) @@ -265,6 +354,7 @@ private func shouldBailOutEarly() -> Bool { /// 6. Acceptable Assertions are on /// 7. Acceptable Apps are in front /// 8. Refresh Timer hasn't been met + /// 9. macOS Updates are downloading or preparing for installation let frontmostApplication = NSWorkspace.shared.frontmostApplication let pastRequiredInstallationDate = DateManager().pastRequiredInstallationDate() @@ -313,6 +403,12 @@ private func shouldBailOutEarly() -> Bool { if isRefreshTimerPassedThreshold() { return true } + + // Check if downloading or preparing updates + if OptionalFeatureVariables.acceptableUpdatePreparingUsage && isDownloadingOrPreparingSoftwareUpdate() { + LogManager.info("Ignoring Nudge activation - macOS is currently downloading or preparing an update", logger: uiLog) + return true + } return false } From fcf7b180f41ea90a20d29fb75c875abb496f3ba2 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 2 Jul 2024 10:25:51 -0500 Subject: [PATCH 100/141] add another issue --- CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 66826ad2..3eca0cd7 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -79,7 +79,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - Nudge can now bypass activations and re-activations when a macOS update is `Downloading`, `Preparing` or `Staged` for installation. - To enable this, please configure the `acceptableUpdatePreparingUsage` key under `optionalFeatures` to true - Be aware that the current logic used for this **cannot differentiate** when an update has completed preparing and is in the `Staged` phase, waiting for a user to reboot. This is due to an Apple process staying in memory. This will result in a reduction in Nudge re-activations - - Issue [555](https://github.com/macadmins/nudge/issues/555) + - Issue [555](https://github.com/macadmins/nudge/issues/555) and [571](https://github.com/macadmins/nudge/issues/571) ## [1.1.16] - 2024-03-13 This will be the **final Nudge release** for macOS 11 and potentially other versions of macOS. From d904dff5540ec21307340d8e22bb77e44a91b0d0 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 2 Jul 2024 10:42:20 -0500 Subject: [PATCH 101/141] fix ui issue with requiredinstalldate under hour and no quit button set --- CHANGELOG.md | 2 ++ Nudge/UI/Common/QuitButtons.swift | 19 ++++++++++++++++--- 2 files changed, 18 insertions(+), 3 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 3eca0cd7..e79716b5 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -40,6 +40,8 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - Refactor portions of the `softwareupdate` logic to reduce potential errors - Fixed errors when moving to Swift 5.10 - Fixed wrong `requiredInstallationDate` calculations when using [Non-Gregorian calendars](https://github.com/macadmins/nudge/issues/509) +- Fixed UI logic when requiredInstallationDate is under an hour and `allowLaterDeferralButton` is set to false + - Issue [475](https://github.com/macadmins/nudge/issues/475) ### Added - To artificially change the `requredInstallationDate` thereby giving your users a default grace period for all Nudge events updates, please configure the `nudgeEventLaunchDelay` key under `userExperience` diff --git a/Nudge/UI/Common/QuitButtons.swift b/Nudge/UI/Common/QuitButtons.swift index 006628c9..336b950c 100644 --- a/Nudge/UI/Common/QuitButtons.swift +++ b/Nudge/UI/Common/QuitButtons.swift @@ -14,8 +14,15 @@ struct QuitButtons: View { var body: some View { HStack { if shouldShowSecondaryQuitButton { - secondaryQuitButton - .frame(maxWidth:215, maxHeight: 30) + if UserExperienceVariables.allowLaterDeferralButton { + secondaryQuitButton + .frame(maxWidth:215, maxHeight: 30) + } else { + if appState.secondsRemaining > 3600 { + secondaryQuitButton + .frame(maxWidth:215, maxHeight: 30) + } + } Spacer() } if shouldShowPrimaryQuitButton { @@ -76,7 +83,13 @@ struct QuitButtons: View { private var primaryQuitButton: some View { Group { if UserExperienceVariables.allowUserQuitDeferrals { - deferralMenu + if UserExperienceVariables.allowLaterDeferralButton { + deferralMenu + } else { + if appState.secondsRemaining > 3600 { + deferralMenu + } + } } else { standardQuitButton } From 6e8c7fe73cda9e34c36e8a622103cd384c9b8b56 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 2 Jul 2024 12:14:26 -0500 Subject: [PATCH 102/141] add optional features to jamf json --- Schema/jamf/com.github.macadmins.Nudge.json | 107 ++++++++++++++++++++ 1 file changed, 107 insertions(+) diff --git a/Schema/jamf/com.github.macadmins.Nudge.json b/Schema/jamf/com.github.macadmins.Nudge.json index 58437e20..9fbef56a 100644 --- a/Schema/jamf/com.github.macadmins.Nudge.json +++ b/Schema/jamf/com.github.macadmins.Nudge.json @@ -86,6 +86,20 @@ } ] }, + "acceptableUpdatePreparingUsage": { + "description": "When enabled, Nudge will not activate or re-activate when an update is being downloaded, prepared or staged. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "title": "Not Configured", + "type": "null" + }, + { + "title": "Configured", + "default": false, + "type": "boolean" + } + ] + }, "acceptableScreenSharingUsage": { "description": "When enabled, Nudge will not activate or re-activate when screen sharing is active.", "anyOf": [ @@ -156,6 +170,20 @@ } ] }, + "attemptToCheckForSupportedDevice": { + "description": "When disabled, Nudge will no longer compare the current device against the SOFA feed for the required update. If the device cannot install this update, Nudge will not present the Unsupported UI (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "boolean", + "default": true + } + ] + }, "attemptToFetchMajorUpgrade": { "description": "When a major upgrade is required, Nudge will attempt to download it through the softwareupdate binary. (Note: This key is only used with Nudge v1.1 and will not be honored in v1.0.)", "anyOf": [ @@ -192,6 +220,24 @@ } ] }, + "customSOFAFeedURL": { + "description": "A url path to use a custom SOFA feed. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "https://sofa.macadmins.io/v1/macos_data_feed.json" + } + } + } + ] + }, "disableSoftwareUpdateWorkflow": { "description": "When disableSoftwareUpdateWorkflow is true, Nudge will not attempt to run the softwareupdate process. Defaults to false.", "anyOf": [ @@ -220,6 +266,53 @@ } ] }, + "honorFocusModes": { + "description": "When enabled, Nudge will not activate or re-activate when a user is in DoNotDisturb/Focus status. This feature is expiremental and may not work in all user settings. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "boolean", + "default": false + } + ] + }, + "honorCycleTimersOnExit": { + "description": "When enabled, Nudge will honor the current cycle timers when user's press the `Quit` button. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "boolean", + "default": false + } + ] + }, + "refreshSOFAFeedTime": { + "description": "The maximum age the cached SOFA feed file can be on disk. When this file age expires, Nudge will re-assess the SOFA feed for updates. Please be mindful of changing this value as there is an associated cost for maintaining the SOFA service. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "title": "Not Configured", + "type": "null" + }, + { + "title": "Configured", + "default": 86400, + "type": "integer", + "options": { + "inputAttributes": { + "placeholder": "86400" + } + } + } + ] + }, "terminateApplicationsOnLaunch": { "description": "When enabled, Nudge will terminate the applications listed in blockedApplicationBundleIDs upon initial launch.", "anyOf": [ @@ -233,6 +326,20 @@ "default": false } ] + }, + "utilizeSOFAFeed": { + "description": "When enabled, Nudge will utilize the SOFA feed url for update data. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "boolean", + "default": false + } + ] } } } From 4ff5ebf7fdafaa3ad726c72970a722ea9ee86dcb Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 2 Jul 2024 12:14:44 -0500 Subject: [PATCH 103/141] add expiremental feature to changelog --- CHANGELOG.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index e79716b5..d83cb431 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -82,6 +82,9 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - To enable this, please configure the `acceptableUpdatePreparingUsage` key under `optionalFeatures` to true - Be aware that the current logic used for this **cannot differentiate** when an update has completed preparing and is in the `Staged` phase, waiting for a user to reboot. This is due to an Apple process staying in memory. This will result in a reduction in Nudge re-activations - Issue [555](https://github.com/macadmins/nudge/issues/555) and [571](https://github.com/macadmins/nudge/issues/571) +- Nudge can now attempt to honor DoNotDisturb/Focus times + - To enable this, please configure the `honorFocusModes` key in `optionalFeatures` to true + - This is an **expiremental feature** and may not work due to significant changes that Apple has designed for detecting these events. ## [1.1.16] - 2024-03-13 This will be the **final Nudge release** for macOS 11 and potentially other versions of macOS. From 044e2a03762d1f9fd0c41859a6881f29af6f04d8 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 2 Jul 2024 12:23:44 -0500 Subject: [PATCH 104/141] osVersionRequirements jamf json --- Schema/jamf/com.github.macadmins.Nudge.json | 133 ++++++++++++++++++-- 1 file changed, 121 insertions(+), 12 deletions(-) diff --git a/Schema/jamf/com.github.macadmins.Nudge.json b/Schema/jamf/com.github.macadmins.Nudge.json index 9fbef56a..182dbf3c 100644 --- a/Schema/jamf/com.github.macadmins.Nudge.json +++ b/Schema/jamf/com.github.macadmins.Nudge.json @@ -452,6 +452,25 @@ } ] }, + "activelyExploitedCVEsInstallationSLA": { + "description": "When an update is under active exploit, this is the amount of days a user has to install the update. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "title": "Not Configured", + "type": "null" + }, + { + "title": "Configured", + "default": 14, + "type": "integer", + "options": { + "inputAttributes": { + "placeholder": "14" + } + } + } + ] + }, "majorUpgradeAppPath": { "description": "The app path for a major upgrade. (Note: Requires Nudge v1.0.1 or higher.)", "anyOf": [ @@ -470,6 +489,25 @@ } ] }, + "nonActivelyExploitedCVEsSLA": { + "description": "When an update is not under active exploit but contains CVEs, this is the amount of days a user has to install the update. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "title": "Not Configured", + "type": "null" + }, + { + "title": "Configured", + "default": 21, + "type": "integer", + "options": { + "inputAttributes": { + "placeholder": "21" + } + } + } + ] + }, "requiredInstallationDate": { "description": "The required installation date for Nudge to enforce the required operating system version. You must follow a standard date string as YYYY-MM-DDTHH:MM:SSZ - Example: 2021-09-15T00:00:00Z", "anyOf": [ @@ -506,8 +544,8 @@ } ] }, - "targetedOSVersions": { - "description": "The versions of macOS that require a security update. You can specify single version or multiple versions. This key is only used with Nudge v1.0 and will not be honored in v1.1.", + "standardInstallationSLA": { + "description": "When an update has no known CVEs, this is the amount of days a user has to install the update. (Note: This key is only used with Nudge v2.0 and higher)", "anyOf": [ { "title": "Not Configured", @@ -515,21 +553,18 @@ }, { "title": "Configured", - "type": "array", - "items": { - "options": { - "inputAttributes": { - "placeholder": "11.5.1" - } - }, - "type": "string", - "title": "targetedOSVersion" + "default": 28, + "type": "integer", + "options": { + "inputAttributes": { + "placeholder": "28" + } } } ] }, "targetedOSVersionsRule": { - "description": "The OS string rule for targeting Nudge events. You can target with \"default\", the full OS version (example: \"11.5.1\"). or the major OS version (example: \"11\"). This key is only used with Nudge v1.1 and will not be honored in v1.0.", + "description": "The OS string rule for targeting Nudge events. You can target with \"default\", the full OS version (example: \"11.5.1\"). or the major OS version (example: \"11\"). (Note: This key is only used with Nudge v1.1 and higher)", "anyOf": [ { "type": "null", @@ -545,6 +580,80 @@ } } ] + }, + "unsupportedURL": { + "description": "A single URL, enabling the More Info button URL path when using the unsupported UI. While this accepts a string, it must be a valid URL (http://, https://, file://). Note: If this value is passed with aboutUpdateURLs, the aboutUpdateURLs key will be ignored. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "https://github.com/macadmins/nudge" + } + } + } + ] + }, + "unsupportedURLs": { + "description": "The unsupportedURL - per country localization.", + "title": "aboutUpdateURLs", + "anyOf": [ + { + "title": "Not Configured", + "type": "null" + }, + { + "title": "Configured", + "type": "array", + "items": { + "title": "unsupportedURL - Dictionary", + "type": "object", + "properties": { + "_language": { + "description": "The targeted language locale for the user interface. Note: For a list of locales, please run the following command in Terminal: /usr/bin/locale -a", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "EX: en" + } + } + } + ] + }, + "unsupportedURL": { + "description": "A single URL, enabling the More Info button URL path when using the unsupported UI. While this accepts a string, it must be a valid URL (http://, https://, file://). Note: If this value is passed with aboutUpdateURLs, the aboutUpdateURLs key will be ignored. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "https://github.com/macadmins/nudge" + } + } + } + ] + } + } + } + } + ] } } } From fbd39fe4bc35fe1b7f6e0687ed43899f5ae04bb1 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 2 Jul 2024 12:28:17 -0500 Subject: [PATCH 105/141] sort keys --- Nudge/Preferences/DefaultPreferencesNudge.swift | 12 ++++++------ Nudge/Preferences/PreferencesStructure.swift | 10 +++++----- 2 files changed, 11 insertions(+), 11 deletions(-) diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 870cfb6b..86155642 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -370,6 +370,12 @@ struct UserExperienceVariables { false } + static var nudgeEventLaunchDelay: Int { + userExperienceProfile?["nudgeEventLaunchDelay"] as? Int ?? + userExperienceJSON?.nudgeEventLaunchDelay ?? + 0 + } + static var nudgeRefreshCycle: Int { userExperienceProfile?["nudgeRefreshCycle"] as? Int ?? userExperienceJSON?.nudgeRefreshCycle ?? @@ -381,12 +387,6 @@ struct UserExperienceVariables { userExperienceJSON?.randomDelay ?? false } - - static var nudgeEventLaunchDelay: Int { - userExperienceProfile?["nudgeEventLaunchDelay"] as? Int ?? - userExperienceJSON?.nudgeEventLaunchDelay ?? - 0 - } } // User Interface diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index c564e58e..3df47e4d 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -358,9 +358,9 @@ struct UserExperience: Codable { var loadLaunchAgent: Bool? var maxRandomDelayInSeconds: Int? var noTimers: Bool? + var nudgeEventLaunchDelay: Int? var nudgeRefreshCycle: Int? var randomDelay: Bool? - var nudgeEventLaunchDelay: Int? } // MARK: UserExperience convenience initializers and mutators @@ -402,9 +402,9 @@ extension UserExperience { loadLaunchAgent: Bool? = nil, maxRandomDelayInSeconds: Int? = nil, noTimers: Bool? = nil, + nudgeEventLaunchDelay: Int? = nil, nudgeRefreshCycle: Int? = nil, - randomDelay: Bool? = nil, - nudgeEventLaunchDelay: Int? = nil + randomDelay: Bool? = nil ) -> UserExperience { return UserExperience( allowGracePeriods: allowGracePeriods ?? self.allowGracePeriods, @@ -427,9 +427,9 @@ extension UserExperience { loadLaunchAgent: loadLaunchAgent ?? self.loadLaunchAgent, maxRandomDelayInSeconds: maxRandomDelayInSeconds ?? self.maxRandomDelayInSeconds, noTimers: noTimers ?? self.noTimers, + nudgeEventLaunchDelay: nudgeEventLaunchDelay ?? self.nudgeEventLaunchDelay, nudgeRefreshCycle: nudgeRefreshCycle ?? self.nudgeRefreshCycle, - randomDelay: randomDelay ?? self.randomDelay, - nudgeEventLaunchDelay: nudgeEventLaunchDelay ?? self.nudgeEventLaunchDelay + randomDelay: randomDelay ?? self.randomDelay ) } } From 20c5a49e71ec4ab4262817541148d6c4fe92cf8e Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 2 Jul 2024 12:28:33 -0500 Subject: [PATCH 106/141] userExperience jamf json --- Schema/jamf/com.github.macadmins.Nudge.json | 33 +++++++++++++++++++++ 1 file changed, 33 insertions(+) diff --git a/Schema/jamf/com.github.macadmins.Nudge.json b/Schema/jamf/com.github.macadmins.Nudge.json index 182dbf3c..3ff2ef93 100644 --- a/Schema/jamf/com.github.macadmins.Nudge.json +++ b/Schema/jamf/com.github.macadmins.Nudge.json @@ -700,6 +700,20 @@ } ] }, + "allowMovableWindow": { + "description": "Allows the user to move the Nudge window. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "title": "Not Configured", + "type": "null" + }, + { + "title": "Configured", + "default": false, + "type": "boolean" + } + ] + }, "allowUserQuitDeferrals": { "description": "Allows the user to specify when they will next be prompted by Nudge. (Set to `False` to maintain v1.0.0 behavior.) When using this feature, Nudge will no longer adhere to your LaunchAgent logic as the user is specifying their own execution time for the next Nudge event.(See: `~/Library/Preferences/com.github.macadmins.Nudge.plist`.)", "anyOf": [ @@ -1012,6 +1026,25 @@ } ] }, + "nudgeEventLaunchDelay": { + "description": "When a new update is posted to the SOFA feed, this can artificially delay the SOFA nudge events by x amount of days. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "title": "Not Configured", + "type": "null" + }, + { + "title": "Configured", + "default": 0, + "type": "integer", + "options": { + "inputAttributes": { + "placeholder": "0" + } + } + } + ] + }, "nudgeRefreshCycle": { "description": "The amount of time in seconds Nudge will use as its core timer to refresh all the core code paths. Note: While you can lower this setting, it could make Nudge too aggressive. Be mindful of decreasing this value.", "anyOf": [ From 9a3da3c200fa1be8214f752f5cc7ca356b0a2a4d Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 2 Jul 2024 12:59:23 -0500 Subject: [PATCH 107/141] sort more keys --- .../Preferences/DefaultPreferencesNudge.swift | 141 +++++++++--------- Nudge/Preferences/PreferencesStructure.swift | 12 +- 2 files changed, 76 insertions(+), 77 deletions(-) diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 86155642..3e284af8 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -396,11 +396,41 @@ struct UserInterfaceVariables { static var userInterfaceUpdateElementsProfile: [String:AnyObject]? = getUserInterfaceUpdateElementsProfile() static var userInterfaceUpdateElementsJSON: UpdateElement? = getUserInterfaceUpdateElementsJSON() + static var actionButtonText: String { + userInterfaceUpdateElementsProfile?["actionButtonText"] as? String ?? + userInterfaceUpdateElementsJSON?.actionButtonText ?? + "Update Device" + } + static var applicationTerminatedNotificationImagePath: String { userInterfaceProfile?["applicationTerminatedNotificationImagePath"] as? String ?? userInterfaceJSON?.applicationTerminatedNotificationImagePath ?? "" } + + static var applicationTerminatedTitleText: String { + userInterfaceUpdateElementsProfile?["applicationTerminatedTitleText"] as? String ?? + userInterfaceUpdateElementsJSON?.applicationTerminatedTitleText ?? + "Application terminated" + } + + static var applicationTerminatedBodyText: String { + userInterfaceUpdateElementsProfile?["applicationTerminatedBodyText"] as? String ?? + userInterfaceUpdateElementsJSON?.applicationTerminatedBodyText ?? + "Please update your device to use this application" + } + + static var customDeferralButtonText: String { + userInterfaceUpdateElementsProfile?["customDeferralButtonText"] as? String ?? + userInterfaceUpdateElementsJSON?.customDeferralButtonText ?? + "Custom" + } + + static var customDeferralDropdownText: String { + userInterfaceUpdateElementsProfile?["customDeferralDropdownText"] as? String ?? + userInterfaceUpdateElementsJSON?.customDeferralDropdownText ?? + "Defer" + } static var fallbackLanguage: String { userInterfaceProfile?["fallbackLanguage"] as? String ?? @@ -426,108 +456,108 @@ struct UserInterfaceVariables { "" } - static var screenShotDarkPath: String { - userInterfaceProfile?["screenShotDarkPath"] as? String ?? - userInterfaceJSON?.screenShotDarkPath ?? - "" - } - - static var screenShotLightPath: String { - userInterfaceProfile?["screenShotLightPath"] as? String ?? - userInterfaceJSON?.screenShotLightPath ?? - "" - } - - static var actionButtonText: String { - userInterfaceUpdateElementsProfile?["actionButtonText"] as? String ?? - userInterfaceUpdateElementsJSON?.actionButtonText ?? - "Update Device" - } - - static var applicationTerminatedTitleText: String { - userInterfaceUpdateElementsProfile?["applicationTerminatedTitleText"] as? String ?? - userInterfaceUpdateElementsJSON?.applicationTerminatedTitleText ?? - "Application terminated" - } - - static var applicationTerminatedBodyText: String { - userInterfaceUpdateElementsProfile?["applicationTerminatedBodyText"] as? String ?? - userInterfaceUpdateElementsJSON?.applicationTerminatedBodyText ?? - "Please update your device to use this application" - } - static var informationButtonText: String { userInterfaceUpdateElementsProfile?["informationButtonText"] as? String ?? userInterfaceUpdateElementsJSON?.informationButtonText ?? "More Info" } - + static var informationButtonTextUnsupported: String { userInterfaceUpdateElementsProfile?["informationButtonTextUnsupported"] as? String ?? userInterfaceUpdateElementsJSON?.informationButtonTextUnsupported ?? "Replace Your Device" } - + static var mainContentHeader: String { userInterfaceUpdateElementsProfile?["mainContentHeader"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentHeader ?? "**Your device will restart during this update**" } - + static var mainContentHeaderUnsupported: String { userInterfaceUpdateElementsProfile?["mainContentHeaderUnsupported"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentHeaderUnsupported ?? "**Your device is no longer capable of receving critical security updates**" } - + static var mainContentNote: String { userInterfaceUpdateElementsProfile?["mainContentNote"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentNote ?? "**Important Notes**" } - + static var mainContentNoteUnsupported: String { userInterfaceUpdateElementsProfile?["mainContentNoteUnsupported"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentNoteUnsupported ?? "**Important Notes**" } - + static var mainContentSubHeader: String { userInterfaceUpdateElementsProfile?["mainContentSubHeader"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentSubHeader ?? "Updates can take around 30 minutes to complete" } - + static var mainContentSubHeaderUnsupported: String { userInterfaceUpdateElementsProfile?["mainContentSubHeaderUnsupported"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentSubHeaderUnsupported ?? "Please work with your local IT team to obtain a replacement device" } - + static var mainContentText: String { userInterfaceUpdateElementsProfile?["mainContentText"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentText ?? "A fully up-to-date device is required to ensure that IT can accurately protect your device.\n\nIf you do not update your device, you may lose access to some items necessary for your day-to-day tasks.\n\nTo begin the update, simply click on the **Update Device** button and follow the provided steps." } - + static var mainContentTextUnsupported: String { userInterfaceUpdateElementsProfile?["mainContentTextUnsupported"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentTextUnsupported ?? "A fully up-to-date device is required to ensure that IT can accurately protect your device.\n\nIf you do not obtain a replacement device, you will lose access to some items necessary for your day-to-day tasks.\n\nFor more information about this, please click on the **Replace Your Device** button." } + static var oneDayDeferralButtonText: String { + userInterfaceUpdateElementsProfile?["oneDayDeferralButtonText"] as? String ?? + userInterfaceUpdateElementsJSON?.oneDayDeferralButtonText ?? + "One Day" + } + + static var oneHourDeferralButtonText: String { + userInterfaceUpdateElementsProfile?["oneHourDeferralButtonText"] as? String ?? + userInterfaceUpdateElementsJSON?.oneHourDeferralButtonText ?? + "One Hour" + } + static var primaryQuitButtonText: String { userInterfaceUpdateElementsProfile?["primaryQuitButtonText"] as? String ?? userInterfaceUpdateElementsJSON?.primaryQuitButtonText ?? "Later" } - + static var requiredInstallationDisplayFormat: String { userInterfaceProfile?["requiredInstallationDisplayFormat"] as? String ?? userInterfaceJSON?.requiredInstallationDisplayFormat ?? "MM/dd/yyyy" } + static var screenShotAltText: String { + userInterfaceUpdateElementsProfile?["screenShotAltText"] as? String ?? + userInterfaceUpdateElementsJSON?.screenShotAltText ?? + "Click to zoom into screenshot" + } + + static var screenShotDarkPath: String { + userInterfaceProfile?["screenShotDarkPath"] as? String ?? + userInterfaceJSON?.screenShotDarkPath ?? + "" + } + + static var screenShotLightPath: String { + userInterfaceProfile?["screenShotLightPath"] as? String ?? + userInterfaceJSON?.screenShotLightPath ?? + "" + } + static var secondaryQuitButtonText: String { userInterfaceUpdateElementsProfile?["secondaryQuitButtonText"] as? String ?? userInterfaceUpdateElementsJSON?.secondaryQuitButtonText ?? @@ -575,37 +605,6 @@ struct UserInterfaceVariables { userInterfaceUpdateElementsJSON?.subHeaderUnsupported ?? "**A friendly reminder from your local IT team**" } - - - static var customDeferralDropdownText: String { - userInterfaceUpdateElementsProfile?["customDeferralDropdownText"] as? String ?? - userInterfaceUpdateElementsJSON?.customDeferralDropdownText ?? - "Defer" - } - - static var customDeferralButtonText: String { - userInterfaceUpdateElementsProfile?["customDeferralButtonText"] as? String ?? - userInterfaceUpdateElementsJSON?.customDeferralButtonText ?? - "Custom" - } - - static var oneDayDeferralButtonText: String { - userInterfaceUpdateElementsProfile?["oneDayDeferralButtonText"] as? String ?? - userInterfaceUpdateElementsJSON?.oneDayDeferralButtonText ?? - "One Day" - } - - static var oneHourDeferralButtonText: String { - userInterfaceUpdateElementsProfile?["oneHourDeferralButtonText"] as? String ?? - userInterfaceUpdateElementsJSON?.oneHourDeferralButtonText ?? - "One Hour" - } - - static var screenShotAltText: String { - userInterfaceUpdateElementsProfile?["screenShotAltText"] as? String ?? - userInterfaceUpdateElementsJSON?.screenShotAltText ?? - "Click to zoom into screenshot" - } } // Other important defaults diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index 3df47e4d..9abf263a 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -507,11 +507,11 @@ struct UpdateElement: Codable { var language, actionButtonText, applicationTerminatedTitleText, applicationTerminatedBodyText, customDeferralButtonText, customDeferralDropdownText: String? var informationButtonText, informationButtonTextUnsupported, mainContentHeader, mainContentHeaderUnsupported, mainContentNote, mainContentNoteUnsupported: String? var mainContentSubHeader, mainContentSubHeaderUnsupported, mainContentText, mainContentTextUnsupported, mainHeader, mainHeaderUnsupported: String? - var oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, secondaryQuitButtonText, subHeader, subHeaderUnsupported, screenShotAltText: String? + var oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, screenShotAltText, secondaryQuitButtonText, subHeader, subHeaderUnsupported: String? enum CodingKeys: String, CodingKey { case language = "_language" - case actionButtonText, applicationTerminatedTitleText, applicationTerminatedBodyText, customDeferralButtonText, customDeferralDropdownText, informationButtonText, informationButtonTextUnsupported, mainContentHeader, mainContentHeaderUnsupported, mainContentNote, mainContentNoteUnsupported, mainContentSubHeader, mainContentSubHeaderUnsupported, mainContentText, mainContentTextUnsupported, mainHeader, mainHeaderUnsupported, oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, secondaryQuitButtonText, subHeader, subHeaderUnsupported, screenShotAltText + case actionButtonText, applicationTerminatedTitleText, applicationTerminatedBodyText, customDeferralButtonText, customDeferralDropdownText, informationButtonText, informationButtonTextUnsupported, mainContentHeader, mainContentHeaderUnsupported, mainContentNote, mainContentNoteUnsupported, mainContentSubHeader, mainContentSubHeaderUnsupported, mainContentText, mainContentTextUnsupported, mainHeader, mainHeaderUnsupported, oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, screenShotAltText, secondaryQuitButtonText, subHeader, subHeaderUnsupported } } @@ -555,10 +555,10 @@ extension UpdateElement { oneDayDeferralButtonText: String? = nil, oneHourDeferralButtonText: String? = nil, primaryQuitButtonText: String? = nil, + screenShotAltText: String? = nil, secondaryQuitButtonText: String? = nil, subHeader: String? = nil, - subHeaderUnsupported: String? = nil, - screenShotAltText: String? = nil + subHeaderUnsupported: String? = nil ) -> UpdateElement { return UpdateElement( language: language ?? self.language, @@ -582,10 +582,10 @@ extension UpdateElement { oneDayDeferralButtonText: oneDayDeferralButtonText ?? self.oneDayDeferralButtonText, oneHourDeferralButtonText: oneHourDeferralButtonText ?? self.oneHourDeferralButtonText, primaryQuitButtonText: primaryQuitButtonText ?? self.primaryQuitButtonText, + screenShotAltText: screenShotAltText ?? self.screenShotAltText, secondaryQuitButtonText: secondaryQuitButtonText ?? self.secondaryQuitButtonText, subHeader: subHeader ?? self.subHeader, - subHeaderUnsupported: subHeaderUnsupported ?? self.subHeaderUnsupported, - screenShotAltText: screenShotAltText ?? self.screenShotAltText + subHeaderUnsupported: subHeaderUnsupported ?? self.subHeaderUnsupported ) } } From d7fe24fe4b84df5aac305f0b965a7ee37862db3a Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 2 Jul 2024 12:59:40 -0500 Subject: [PATCH 108/141] userInterface keys jamf json --- Schema/jamf/com.github.macadmins.Nudge.json | 280 +++++++++++++++++++- 1 file changed, 269 insertions(+), 11 deletions(-) diff --git a/Schema/jamf/com.github.macadmins.Nudge.json b/Schema/jamf/com.github.macadmins.Nudge.json index 3ff2ef93..aa443390 100644 --- a/Schema/jamf/com.github.macadmins.Nudge.json +++ b/Schema/jamf/com.github.macadmins.Nudge.json @@ -1117,6 +1117,24 @@ } ] }, + "applicationTerminatedNotificationImagePath": { + "description": "A local image path for the notification event when Nudge terminates and application. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "/Library/Application Support/Nudge/LogoLight.png" + } + } + } + ] + }, "fallbackLanguage": { "description": "The language to revert to if no localizations are available for the device's current language.", "anyOf": [ @@ -1200,6 +1218,24 @@ } ] }, + "requiredInstallationDisplayFormat": { + "description": "When utilizing showRequiredDate, set a custom display format. Be aware that the format you desire may not look good on the UI. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "MM/dd/yyyy" + } + } + } + ] + }, "screenShotDarkPath": { "description": "A path to a local jpg, png, icns that contains the screen shot for dark mode. This will replace the Big Sur logo on the lower right side of Nudge.", "anyOf": [ @@ -1236,6 +1272,20 @@ } ] }, + "showActivelyExploitedCVEs": { + "description": "When disabled, Nudge will not show the Actively Exploited CVEs in the left sidebar. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "title": "Not Configured", + "type": "null" + }, + { + "title": "Configured", + "default": true, + "type": "boolean" + } + ] + }, "showDeferralCount": { "description": "Enables or disables the deferral count of the current Nudge event.", "anyOf": [ @@ -1250,6 +1300,34 @@ } ] }, + "showDaysRemainingToUpdate": { + "description": "When disabled, Nudge will not show the `Days Remaining To Update:` item on the left side of the UI. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "title": "Not Configured", + "type": "null" + }, + { + "title": "Configured", + "default": true, + "type": "boolean" + } + ] + }, + "showRequiredDate": { + "description": "When enabled, Nudge will also show the requiredInstallationDate as string formatted date. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "title": "Not Configured", + "type": "null" + }, + { + "title": "Configured", + "default": false, + "type": "boolean" + } + ] + }, "simpleMode": { "description": "Enables Nudge to launch in the simplified user experience.", "anyOf": [ @@ -1329,6 +1407,42 @@ } ] }, + "applicationTerminatedTitleText": { + "description": "Modifies the terminated application notification title. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "Application terminated" + } + } + } + ] + }, + "applicationTerminatedBodyText": { + "description": "Modifies the terminated application notification body. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "Please update your device to use this application" + } + } + } + ] + }, "customDeferralButtonText": { "description": "Modifies the customDeferralButtonText, also known as the \"Custom\" button.", "anyOf": [ @@ -1383,8 +1497,8 @@ } ] }, - "oneDayDeferralButtonText": { - "description": "Modifies the oneDayDeferralButtonText, also known as the \"One Day\" button.", + "informationButtonTextUnsupported": { + "description": "Modifies the informationButton, also known as the \"More Info\" button when using the Unsupported UI. (Note: This key is only used with Nudge v2.0 and higher)", "anyOf": [ { "type": "null", @@ -1395,14 +1509,14 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "One Day" + "placeholder": "Replace Your Device" } } } ] }, - "oneHourDeferralButtonText": { - "description": "Modifies the oneHourDeferralButtonText, also known as the \"One Hour\" button.", + "mainContentHeader": { + "description": "Modifies the mainContentHeader. This is the \"Your device will restart during this update\" text.", "anyOf": [ { "type": "null", @@ -1413,14 +1527,14 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "One Hour" + "placeholder": "**Your device will restart during this update**" } } } ] }, - "mainContentHeader": { - "description": "Modifies the mainContentHeader. This is the \"Your device will restart during this update\" text.", + "mainContentHeaderUnsupported": { + "description": "Modifies the mainContentHeader. This is the \"Your device is no longer capable of receving critical security updates\" text when using the Unsupported UI. (Note: This key is only used with Nudge v2.0 and higher)", "anyOf": [ { "type": "null", @@ -1431,7 +1545,7 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "Your device will restart during this update" + "placeholder": "**Your device is no longer capable of receving critical security updates**" } } } @@ -1449,7 +1563,25 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "Important Notes" + "placeholder": "**Important Notes**" + } + } + } + ] + }, + "mainContentNoteUnsupported": { + "description": "Modifies the mainContentNote. This is the \"Important Notes\" text when using the Unsupported UI. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "**Important Notes**" } } } @@ -1473,6 +1605,24 @@ } ] }, + "mainContentSubHeaderUnsupported": { + "description": "Modifies the mainContentSubHeader. This is the \"Please work with your local IT team to obtain a replacement device\" text when using the Unsupported UI. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "Please work with your local IT team to obtain a replacement device" + } + } + } + ] + }, "mainContentText": { "description": "Modifies the `mainContentText`. This is the \"A fully up-to-date device is required to ensure that IT can your accurately protect your device.\" text. (See the Wiki for information on adding line breaks.)", "anyOf": [ @@ -1491,6 +1641,24 @@ } ] }, + "mainContentTextUnsupported": { + "description": "Modifies the `mainContentText`. This is the \"A fully up-to-date device is required to ensure that IT can your accurately protect your device.\" text when using the Unsupported UI. See the Wiki for information on adding line breaks. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "A fully up-to-date device …" + } + } + } + ] + }, "mainHeader": { "description": "Modifies the `mainHeader`. This is the \"Your device requires a security update\" text.", "anyOf": [ @@ -1509,6 +1677,60 @@ } ] }, + "mainHeaderUnsupported": { + "description": "Modifies the `mainHeader`. This is the \"Your device requires a security update\" text when using the Unsupported UI. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "Your device requires a security update" + } + } + } + ] + }, + "oneDayDeferralButtonText": { + "description": "Modifies the oneDayDeferralButtonText, also known as the \"One Day\" button.", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "One Day" + } + } + } + ] + }, + "oneHourDeferralButtonText": { + "description": "Modifies the oneHourDeferralButtonText, also known as the \"One Hour\" button.", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "One Hour" + } + } + } + ] + }, "primaryQuitButtonText": { "description": "Modifies the `primaryQuitButton`, also known as the \"Later\" button.", "anyOf": [ @@ -1527,6 +1749,24 @@ } ] }, + "screenShotAltText": { + "description": "Modifies the accessible hover over on screen shots.", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "Click to zoom into screenshot" + } + } + } + ] + }, "secondaryQuitButtonText": { "description": "Modifies the `secondaryQuitButton`, also known as the \"I understand\" button.", "anyOf": [ @@ -1557,7 +1797,25 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "A friendly reminder from your local IT team" + "placeholder": "**A friendly reminder from your local IT team**" + } + } + } + ] + }, + "subHeaderUnsupported": { + "description": "Modifies the `subHeader`. This is the \"A friendly reminder from your local IT team\" text when using the Unsupported UI. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "type": "null", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "**A friendly reminder from your local IT team**" } } } From 0fb6fbe52b3613daff79c86e6f5c3a3c8abc11e6 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 2 Jul 2024 13:34:50 -0500 Subject: [PATCH 109/141] attempt to check for MDM profile changes --- CHANGELOG.md | 2 ++ Nudge/UI/Defaults.swift | 2 +- Nudge/UI/Main.swift | 10 ++++++++++ 3 files changed, 13 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index d83cb431..0be267d1 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -85,6 +85,8 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - Nudge can now attempt to honor DoNotDisturb/Focus times - To enable this, please configure the `honorFocusModes` key in `optionalFeatures` to true - This is an **expiremental feature** and may not work due to significant changes that Apple has designed for detecting these events. +- Nudge now attempts to reload the preferences if MDM profile is updated + - Issue [370](https://github.com/macadmins/nudge/issues/370) ## [1.1.16] - 2024-03-13 This will be the **final Nudge release** for macOS 11 and potentially other versions of macOS. diff --git a/Nudge/UI/Defaults.swift b/Nudge/UI/Defaults.swift index bcf3cdc6..7f750d07 100644 --- a/Nudge/UI/Defaults.swift +++ b/Nudge/UI/Defaults.swift @@ -23,7 +23,7 @@ struct Globals { static let snc = NSWorkspace.shared.notificationCenter // Preferences static let configJSON = ConfigurationManager().getConfigurationAsJSON() - static let configProfile = ConfigurationManager().getConfigurationAsProfile() + static var configProfile = ConfigurationManager().getConfigurationAsProfile() static let nudgeDefaults = UserDefaults.standard static let nudgeJSONPreferences = NetworkFileManager().getNudgeJSONPreferences() // Device Properties diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index d1d01796..6596f8db 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -606,6 +606,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { setupScreenChangeObservers() setupScreenLockObservers() setupWorkspaceNotificationCenterObservers() + setupUserDefaultsObservers() } private func setupNotificationCenterObservers() { @@ -619,6 +620,15 @@ class AppDelegate: NSObject, NSApplicationDelegate { } } + private func setupUserDefaultsObservers() { + Globals.nc.addObserver( + forName: UserDefaults.didChangeNotification, + object: nil, + queue: .main) { [weak self] _ in + Globals.configProfile = ConfigurationManager().getConfigurationAsProfile() + } + } + private func setupScreenChangeObservers() { Globals.nc.addObserver( self, From 539ff5f41038d8dde3f32f33a2034670472a69ba Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 2 Jul 2024 14:57:12 -0500 Subject: [PATCH 110/141] fix typo --- CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 0be267d1..dc34b2d8 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -85,7 +85,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - Nudge can now attempt to honor DoNotDisturb/Focus times - To enable this, please configure the `honorFocusModes` key in `optionalFeatures` to true - This is an **expiremental feature** and may not work due to significant changes that Apple has designed for detecting these events. -- Nudge now attempts to reload the preferences if MDM profile is updated +- Nudge now attempts to reload the preferences if the MDM profile is updated - Issue [370](https://github.com/macadmins/nudge/issues/370) ## [1.1.16] - 2024-03-13 From 2ec3b7152b7fd92281a74f16baef0eb2188b0eae Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 10 Jul 2024 11:31:31 -0500 Subject: [PATCH 111/141] handle sofa feed failing when there is no internet --- Nudge/UI/Main.swift | 4 ++-- Nudge/Utilities/Utils.swift | 8 ++++++-- 2 files changed, 8 insertions(+), 4 deletions(-) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 6596f8db..37311cad 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -174,8 +174,6 @@ class AppDelegate: NSObject, NSApplicationDelegate { func sofaPreLaunchLogic() { // TODO: Add more logging to "unsupported devices" UI. - // TODO: Add localization for "unsupported devices" text fields - // TODO: Get someone to update JAMF JSON schema for all the new keys and wiki if OptionalFeatureVariables.utilizeSOFAFeed { var selectedOS: OSInformation? var foundMatch = false @@ -254,6 +252,8 @@ class AppDelegate: NSObject, NSApplicationDelegate { } } else { LogManager.error("Could not fetch SOFA feed", logger: sofaLog) + nudgePrimaryState.shouldExit = true + exit(1) } } } diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 7617871d..7713088b 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -1004,7 +1004,7 @@ struct NetworkFileManager { if let url = URL(string: OptionalFeatureVariables.customSOFAFeedURL) { let sofaData = SOFA().URLSync(url: url) - if (sofaData.error == nil) { + if (sofaData.responseCode != nil) { if sofaData.responseCode == 304 && sofaJSONExists { LogManager.info("Utilizing previously cached SOFA json due to Etag not changing", logger: sofaLog) do { @@ -1030,7 +1030,11 @@ struct NetworkFileManager { } } } else { - LogManager.error("Failed to fetch sofa JSON: \(sofaData.error!.localizedDescription)", logger: sofaLog) + if sofaData.responseCode == nil { + LogManager.error("Failed to fetch sofa JSON: Device likely has no network connectivity.", logger: sofaLog) + } else { + LogManager.error("Failed to fetch sofa JSON: \(sofaData.error!.localizedDescription)", logger: sofaLog) + } } } else { LogManager.error("Failed to decode sofa JSON URL string", logger: sofaLog) From e32887fb02678488b380eafabf6386ddec16944c Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 10 Jul 2024 11:47:54 -0500 Subject: [PATCH 112/141] add more logic for when json is 0bytes and fails to decode --- Nudge/Utilities/Utils.swift | 39 ++++++++++++++++++++++++++++++++++--- 1 file changed, 36 insertions(+), 3 deletions(-) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 7713088b..c5d09738 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -973,7 +973,21 @@ struct NetworkFileManager { let appDirectory = appSupportDirectory.appendingPathComponent(Globals.bundleID) let sofaFile = "sofa-macos_data_feed.json" let sofaPath = appDirectory.appendingPathComponent(sofaFile) - let sofaJSONExists = fileManager.fileExists(atPath: sofaPath.path) + let sofaJSON0Bytes = isFileEmpty(atPath: sofaPath.path) + var sofaJSONExists = fileManager.fileExists(atPath: sofaPath.path) + + // Force delete if bad + if sofaJSONExists { + if sofaJSON0Bytes { + do { + try fileManager.removeItem(atPath: sofaPath.path) + sofaJSONExists = false + } catch { + print("Error deleting file: \(error.localizedDescription)") + } + } + } + if sofaJSONExists { let sofaPathCreationDate = AppStateManager().getModifiedDateForPath(sofaPath.path, testFileDate: nil) // Use Cache as it is within time inverval @@ -984,7 +998,7 @@ struct NetworkFileManager { let assetInfo = try MacOSDataFeed(data: sofaData) return assetInfo } catch { - LogManager.error("Failed to decode local sofa JSON: \(error.localizedDescription)", logger: sofaLog) + LogManager.error("Failed to decode previously cached local sofa JSON: \(error.localizedDescription)", logger: sofaLog) LogManager.error("Failed to decode sofa JSON: \(error)", logger: sofaLog) return nil } @@ -1012,7 +1026,7 @@ struct NetworkFileManager { let assetInfo = try MacOSDataFeed(data: sofaData) return assetInfo } catch { - LogManager.error("Failed to decode local sofa JSON: \(error.localizedDescription)", logger: sofaLog) + LogManager.error("Failed to decode previously cached (Etag) local sofa JSON: \(error.localizedDescription)", logger: sofaLog) LogManager.error("Failed to decode sofa JSON: \(error)", logger: sofaLog) return nil } @@ -1024,6 +1038,12 @@ struct NetworkFileManager { let assetInfo = try MacOSDataFeed(data: sofaData.data!) return assetInfo } catch { + do { + try fileManager.removeItem(atPath: sofaPath.path) + sofaJSONExists = false + } catch { + print("Error deleting file: \(error.localizedDescription)") + } LogManager.error("Failed to decode sofa JSON: \(error.localizedDescription)", logger: sofaLog) LogManager.error("Failed to decode sofa JSON: \(error)", logger: sofaLog) return nil @@ -1086,6 +1106,19 @@ struct NetworkFileManager { LogManager.error("Could not find or decode JSON configuration", logger: prefsJSONLog) return nil } + + func isFileEmpty(atPath path: String) -> Bool { + let fileManager = FileManager.default + do { + let attributes = try fileManager.attributesOfItem(atPath: path) + if let fileSize = attributes[.size] as? NSNumber { + return fileSize.intValue == 0 + } + } catch { + print("Error getting file attributes: \(error.localizedDescription)") + } + return false + } } struct SubProcessUtilities { From 219784870e5ccea1061ae7d37f6dce476bf7476c Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 10 Jul 2024 11:48:20 -0500 Subject: [PATCH 113/141] bump to test sofa feed on 14.5 --- Example Assets/com.github.macadmins.Nudge.tester.json | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index 8a258cc3..23bb133c 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -20,9 +20,9 @@ "osVersionRequirements": [ { "aboutUpdateURL": "https://apple.com", - "requiredMinimumOSVersion": "15.99", - "requiredInstallationDate": "2025-01-01T00:00:00Z", + "requiredMinimumOSVersion": "latest", "unsupportedURL": "https://google.com", + "nonActivelyExploitedCVEsSLA": 60 } ], "userExperience": { From 94749b5485e826e83fe4bcd720e96351256506fd Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 10 Jul 2024 11:49:10 -0500 Subject: [PATCH 114/141] add manual tester 14.4.1 --- Nudge/Preferences/DefaultPreferencesNudge.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 3e284af8..8a59e17a 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -9,7 +9,7 @@ import Foundation // Global Variables struct GlobalVariables { - static let currentOSVersion = OSVersion(ProcessInfo().operatingSystemVersion).description + static let currentOSVersion = OSVersion(ProcessInfo().operatingSystemVersion).description // "14.4.1" static var fetchMajorUpgradeSuccessful = false } From 03a32406380cfb87ddf5aa55ce786781a6273e37 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 10 Jul 2024 15:09:37 -0500 Subject: [PATCH 115/141] create escape hatch for replacing device logic --- CHANGELOG.md | 2 +- .../com.github.macadmins.Nudge.tester.json | 7 ++- .../com.github.macadmins.Nudge.tester.plist | 4 +- .../Preferences/DefaultPreferencesNudge.swift | 12 ++--- Nudge/Preferences/PreferencesStructure.swift | 10 ++-- Nudge/UI/Common/InformationButton.swift | 46 ++++++++++--------- Nudge/UI/SimpleMode/SimpleMode.swift | 20 ++++---- Nudge/UI/StandardMode/RightSide.swift | 2 + Nudge/Utilities/Utils.swift | 8 ++-- Schema/jamf/com.github.macadmins.Nudge.json | 36 +++++++-------- 10 files changed, 74 insertions(+), 73 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index dc34b2d8..0167dc45 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -66,7 +66,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - If you are utilizing a custom sofa feed, please configure the `customSOFAFeedURL` key under `optionalFeatures` - "Unsupported device" UI in standard mode that utilizes the SOFA feed - Set the `attemptToCheckForSupportedDevice` key `false` under `optionalFeatures` to disable this feature - - There are new keys to set all of text fields: `informationButtonTextUnsupported`, `mainContentHeaderUnsupported`, `mainContentNoteUnsupported`, `mainContentSubHeaderUnsupported`, `mainContentTextUnsupported`, `subHeaderUnsupported` under the `updateElements` key in `UserInterface` + - There are new keys to set all of text fields: `actionButtonTextUnsupported`, `mainContentHeaderUnsupported`, `mainContentNoteUnsupported`, `mainContentSubHeaderUnsupported`, `mainContentTextUnsupported`, `subHeaderUnsupported` under the `updateElements` key in `UserInterface` - `unsupportedURL` and `unsupportedURLs` can change the information button itself, but it will remain in the `osVersionRequirement` key with `unsupportedURLs` and `unsupportedURLs`. - An icon will appear as an overlay on top of the company image to further emphasize the device is no longer supported - An admin can now show the `requiredInstallationDate` as a item on the left side of nudge. diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index 23bb133c..ca700057 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -21,8 +21,7 @@ { "aboutUpdateURL": "https://apple.com", "requiredMinimumOSVersion": "latest", - "unsupportedURL": "https://google.com", - "nonActivelyExploitedCVEsSLA": 60 + "unsupportedURL": "https://google.com" } ], "userExperience": { @@ -44,12 +43,12 @@ { "_language": "en", "actionButtonText": "actionButtonText", + "actionButtonTextUnsupported": "actionButtonTextUnsupported", "customDeferralButtonText": "customDeferralButtonText", "customDeferralDropdownText": "customDeferralDropdownText", "informationButtonText": "informationButtonText", - "informationButtonTextUnsupported1": "informationButtonTextUnsupported", "mainContentHeader": "mainContentHeader", - "mainContentHeaderUnsupported1": "mainContentHeaderUnsupported", + "mainContentHeaderUnsupported": "mainContentHeaderUnsupported", "mainContentNote": "mainContentNote", "mainContentNoteUnsupported1": "mainContentNoteUnsupported", "mainContentSubHeader": "mainContentSubHeader", diff --git a/Example Assets/com.github.macadmins.Nudge.tester.plist b/Example Assets/com.github.macadmins.Nudge.tester.plist index 9654d0a5..01368c3b 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.plist +++ b/Example Assets/com.github.macadmins.Nudge.tester.plist @@ -78,8 +78,8 @@ customDeferralDropdownText informationButtonText informationButtonText - informationButtonTextUnsupported1 - informationButtonTextUnsupported + actionButtonTextUnsupported + actionButtonTextUnsupported mainContentHeader mainContentHeader mainContentHeaderUnsupported1 diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 8a59e17a..65798feb 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -401,6 +401,12 @@ struct UserInterfaceVariables { userInterfaceUpdateElementsJSON?.actionButtonText ?? "Update Device" } + + static var actionButtonTextUnsupported: String { + userInterfaceUpdateElementsProfile?["actionButtonTextUnsupported"] as? String ?? + userInterfaceUpdateElementsJSON?.actionButtonTextUnsupported ?? + "Replace Your Device" + } static var applicationTerminatedNotificationImagePath: String { userInterfaceProfile?["applicationTerminatedNotificationImagePath"] as? String ?? @@ -462,12 +468,6 @@ struct UserInterfaceVariables { "More Info" } - static var informationButtonTextUnsupported: String { - userInterfaceUpdateElementsProfile?["informationButtonTextUnsupported"] as? String ?? - userInterfaceUpdateElementsJSON?.informationButtonTextUnsupported ?? - "Replace Your Device" - } - static var mainContentHeader: String { userInterfaceUpdateElementsProfile?["mainContentHeader"] as? String ?? userInterfaceUpdateElementsJSON?.mainContentHeader ?? diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index 9abf263a..bac9c632 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -504,14 +504,14 @@ extension UserInterface { // MARK: - UpdateElement struct UpdateElement: Codable { - var language, actionButtonText, applicationTerminatedTitleText, applicationTerminatedBodyText, customDeferralButtonText, customDeferralDropdownText: String? - var informationButtonText, informationButtonTextUnsupported, mainContentHeader, mainContentHeaderUnsupported, mainContentNote, mainContentNoteUnsupported: String? + var language, actionButtonText, actionButtonTextUnsupported, applicationTerminatedTitleText, applicationTerminatedBodyText, customDeferralButtonText, customDeferralDropdownText: String? + var informationButtonText, mainContentHeader, mainContentHeaderUnsupported, mainContentNote, mainContentNoteUnsupported: String? var mainContentSubHeader, mainContentSubHeaderUnsupported, mainContentText, mainContentTextUnsupported, mainHeader, mainHeaderUnsupported: String? var oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, screenShotAltText, secondaryQuitButtonText, subHeader, subHeaderUnsupported: String? enum CodingKeys: String, CodingKey { case language = "_language" - case actionButtonText, applicationTerminatedTitleText, applicationTerminatedBodyText, customDeferralButtonText, customDeferralDropdownText, informationButtonText, informationButtonTextUnsupported, mainContentHeader, mainContentHeaderUnsupported, mainContentNote, mainContentNoteUnsupported, mainContentSubHeader, mainContentSubHeaderUnsupported, mainContentText, mainContentTextUnsupported, mainHeader, mainHeaderUnsupported, oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, screenShotAltText, secondaryQuitButtonText, subHeader, subHeaderUnsupported + case actionButtonText, actionButtonTextUnsupported, applicationTerminatedTitleText, applicationTerminatedBodyText, customDeferralButtonText, customDeferralDropdownText, informationButtonText, mainContentHeader, mainContentHeaderUnsupported, mainContentNote, mainContentNoteUnsupported, mainContentSubHeader, mainContentSubHeaderUnsupported, mainContentText, mainContentTextUnsupported, mainHeader, mainHeaderUnsupported, oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, screenShotAltText, secondaryQuitButtonText, subHeader, subHeaderUnsupported } } @@ -536,12 +536,12 @@ extension UpdateElement { func with( language: String? = nil, actionButtonText: String? = nil, + actionButtonTextUnsupported: String? = nil, applicationTerminatedTitleText: String? = nil, applicationTerminatedBodyText: String? = nil, customDeferralButtonText: String? = nil, customDeferralDropdownText: String? = nil, informationButtonText: String? = nil, - informationButtonTextUnsupported: String? = nil, mainContentHeader: String? = nil, mainContentHeaderUnsupported: String? = nil, mainContentNote: String? = nil, @@ -563,12 +563,12 @@ extension UpdateElement { return UpdateElement( language: language ?? self.language, actionButtonText: actionButtonText ?? self.actionButtonText, + actionButtonTextUnsupported: actionButtonTextUnsupported ?? self.actionButtonTextUnsupported, applicationTerminatedTitleText: applicationTerminatedTitleText ?? self.applicationTerminatedTitleText, applicationTerminatedBodyText: applicationTerminatedBodyText ?? self.applicationTerminatedBodyText, customDeferralButtonText: customDeferralButtonText ?? self.customDeferralButtonText, customDeferralDropdownText: customDeferralDropdownText ?? self.customDeferralDropdownText, informationButtonText: informationButtonText ?? self.informationButtonText, - informationButtonTextUnsupported: informationButtonTextUnsupported ?? self.informationButtonTextUnsupported, mainContentHeader: mainContentHeader ?? self.mainContentHeader, mainContentHeaderUnsupported: mainContentHeaderUnsupported ?? self.mainContentHeaderUnsupported, mainContentNote: mainContentNote ?? self.mainContentNote, diff --git a/Nudge/UI/Common/InformationButton.swift b/Nudge/UI/Common/InformationButton.swift index 04e7763d..80dd4ea6 100644 --- a/Nudge/UI/Common/InformationButton.swift +++ b/Nudge/UI/Common/InformationButton.swift @@ -19,31 +19,17 @@ struct InformationButton: View { } private var informationButton: some View { - if appState.deviceSupportedByOSVersion { - guard OSVersionRequirementVariables.aboutUpdateURL != "" else { return AnyView(EmptyView()) } + guard OSVersionRequirementVariables.aboutUpdateURL != "" else { return AnyView(EmptyView()) } - return AnyView( - Button(action: UIUtilities().openMoreInfo) { - Text(.init(UserInterfaceVariables.informationButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)))) - .foregroundColor(dynamicTextColor) - } - .buttonStyle(.plain) - .help("Click for more information about the security update".localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) - .onHoverEffect() - ) - } else { - guard OSVersionRequirementVariables.unsupportedURL != "" else { return AnyView(EmptyView()) } - - return AnyView( - Button(action: UIUtilities().openMoreInfoUnsupported) { - Text(.init(UserInterfaceVariables.informationButtonTextUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)))) - .foregroundColor(dynamicTextColor) - } + return AnyView( + Button(action: UIUtilities().openMoreInfo) { + Text(.init(UserInterfaceVariables.informationButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)))) + .foregroundColor(dynamicTextColor) + } .buttonStyle(.plain) - .help("Click for more information about replacing your device".localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + .help("Click for more information about the security update.".localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .onHoverEffect() - ) - } + ) } private var dynamicTextColor: Color { @@ -51,6 +37,22 @@ struct InformationButton: View { } } +// Technically not information button as this is using the actionButtonTextUnsupported +struct InformationButtonAsAction: View { + @EnvironmentObject var appState: AppState + @Environment(\.colorScheme) var colorScheme + + var body: some View { + Button(action: { + UIUtilities().openMoreInfo() + UIUtilities().postUpdateDeviceActions(userClicked: true, unSupportedUI: true) + }) { + Text(.init(UserInterfaceVariables.actionButtonTextUnsupported.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)))) + } + .help("Click for more information about replacing your device.".localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + } +} + #if DEBUG #Preview { ForEach(["en", "es"], id: \.self) { id in diff --git a/Nudge/UI/SimpleMode/SimpleMode.swift b/Nudge/UI/SimpleMode/SimpleMode.swift index 5ef26dce..26cadae0 100644 --- a/Nudge/UI/SimpleMode/SimpleMode.swift +++ b/Nudge/UI/SimpleMode/SimpleMode.swift @@ -41,7 +41,15 @@ struct SimpleMode: View { Spacer() if appState.deviceSupportedByOSVersion { - updateButton + Button(action: { + UIUtilities().updateDevice() + }) { + Text(UserInterfaceVariables.actionButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + .frame(minWidth: 120) + } + .keyboardShortcut(.defaultAction) + } else { + InformationButtonAsAction() } Spacer() } @@ -78,16 +86,6 @@ struct SimpleMode: View { } } - private var updateButton: some View { - Button(action: { - UIUtilities().updateDevice() - }) { - Text(UserInterfaceVariables.actionButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) - .frame(minWidth: 120) - } - .keyboardShortcut(.defaultAction) - } - private var bottomButtons: some View { HStack { InformationButton() diff --git a/Nudge/UI/StandardMode/RightSide.swift b/Nudge/UI/StandardMode/RightSide.swift index 5ff5966a..00c31283 100644 --- a/Nudge/UI/StandardMode/RightSide.swift +++ b/Nudge/UI/StandardMode/RightSide.swift @@ -82,6 +82,8 @@ struct StandardModeRightSide: View { Text(.init(UserInterfaceVariables.actionButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)))) } .keyboardShortcut(.defaultAction) + } else { + InformationButtonAsAction() } } diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index c5d09738..8b453ad3 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -1302,9 +1302,9 @@ struct UIUtilities { NSWorkspace.shared.open(url) } - private func postUpdateDeviceActions(userClicked: Bool) { + func postUpdateDeviceActions(userClicked: Bool, unSupportedUI: Bool) { if userClicked { - LogManager.notice("User clicked updateDevice", logger: uiLog) + LogManager.notice(unSupportedUI ? "User clicked updateDevice" : "User clicked updateDevice via Unsupported UI", logger: uiLog) // Remove forced blur and reset window level if !nudgePrimaryState.backgroundBlur.isEmpty { nudgePrimaryState.backgroundBlur.forEach { blurWindowController in @@ -1315,7 +1315,7 @@ struct UIUtilities { NSApp.windows.first?.level = .normal } } else { - LogManager.notice("Synthetically clicked updateDevice due to allowedDeferral count", logger: uiLog) + LogManager.notice(unSupportedUI ? "Synthetically clicked updateDevice due to allowedDeferral count" : "Synthetically clicked updateDevice via Unsupported UI due to allowedDeferral count", logger: uiLog) } } @@ -1356,7 +1356,7 @@ struct UIUtilities { } } - postUpdateDeviceActions(userClicked: userClicked) + postUpdateDeviceActions(userClicked: userClicked, unSupportedUI: false) } func userInitiatedExit() { diff --git a/Schema/jamf/com.github.macadmins.Nudge.json b/Schema/jamf/com.github.macadmins.Nudge.json index aa443390..ecb961f0 100644 --- a/Schema/jamf/com.github.macadmins.Nudge.json +++ b/Schema/jamf/com.github.macadmins.Nudge.json @@ -1407,8 +1407,8 @@ } ] }, - "applicationTerminatedTitleText": { - "description": "Modifies the terminated application notification title. (Note: This key is only used with Nudge v2.0 and higher)", + "actionButtonTextUnsupported": { + "description": "Modifies the primaryQuitButton, also known as the \"Update Device\" button when using the Unsupported UI. (Note: This key is only used with Nudge v2.0 and higher)", "anyOf": [ { "type": "null", @@ -1419,14 +1419,14 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "Application terminated" + "placeholder": "Replace Your Device" } } } ] }, - "applicationTerminatedBodyText": { - "description": "Modifies the terminated application notification body. (Note: This key is only used with Nudge v2.0 and higher)", + "applicationTerminatedTitleText": { + "description": "Modifies the terminated application notification title. (Note: This key is only used with Nudge v2.0 and higher)", "anyOf": [ { "type": "null", @@ -1437,14 +1437,14 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "Please update your device to use this application" + "placeholder": "Application terminated" } } } ] }, - "customDeferralButtonText": { - "description": "Modifies the customDeferralButtonText, also known as the \"Custom\" button.", + "applicationTerminatedBodyText": { + "description": "Modifies the terminated application notification body. (Note: This key is only used with Nudge v2.0 and higher)", "anyOf": [ { "type": "null", @@ -1455,14 +1455,14 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "Custom" + "placeholder": "Please update your device to use this application" } } } ] }, - "customDeferralDropdownText": { - "description": "customDeferralDropdownText, also known as the \"Defer\" button.", + "customDeferralButtonText": { + "description": "Modifies the customDeferralButtonText, also known as the \"Custom\" button.", "anyOf": [ { "type": "null", @@ -1473,14 +1473,14 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "Defer" + "placeholder": "Custom" } } } ] }, - "informationButtonText": { - "description": "Modifies the informationButton, also known as the \"More Info\" button.", + "customDeferralDropdownText": { + "description": "customDeferralDropdownText, also known as the \"Defer\" button.", "anyOf": [ { "type": "null", @@ -1491,14 +1491,14 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "More Info" + "placeholder": "Defer" } } } ] }, - "informationButtonTextUnsupported": { - "description": "Modifies the informationButton, also known as the \"More Info\" button when using the Unsupported UI. (Note: This key is only used with Nudge v2.0 and higher)", + "informationButtonText": { + "description": "Modifies the informationButton, also known as the \"More Info\" button.", "anyOf": [ { "type": "null", @@ -1509,7 +1509,7 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "Replace Your Device" + "placeholder": "More Info" } } } From 64b3a7e1b44bcac5d34498d8a7ed6e14e7056279 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 10 Jul 2024 15:10:28 -0500 Subject: [PATCH 116/141] add a false to remember where the test is --- Nudge/UI/Main.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 37311cad..848176b9 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -241,7 +241,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { Globals.hardwareModelIDs.contains { $0.uppercased() == supportedDevice.uppercased() } } LogManager.notice("Assessed Model ID found in SOFA Entry: \(deviceMatchFound)", logger: sofaLog) - nudgePrimaryState.deviceSupportedByOSVersion = deviceMatchFound + nudgePrimaryState.deviceSupportedByOSVersion = deviceMatchFound // false } foundMatch = true break From 81594c8b8e28b4fe4e52711f6dcd6b50e8ed300c Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 10 Jul 2024 15:45:24 -0500 Subject: [PATCH 117/141] add another potential update assertion for macOS 15 --- Example Assets/com.github.macadmins.Nudge.tester.json | 6 ++++-- Nudge/Utilities/UILogic.swift | 3 ++- 2 files changed, 6 insertions(+), 3 deletions(-) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index ca700057..cf6a71cb 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -6,7 +6,8 @@ "Google Chrome", "Safari" ], - "acceptableAssertionUsage": true, + "acceptableAssertionUsage": false, + "acceptableUpdatePreparingUsage": true, "acceptableCameraUsage": true, "acceptableScreenSharingUsage": true, "attemptToBlockApplicationLaunches": true, @@ -20,7 +21,8 @@ "osVersionRequirements": [ { "aboutUpdateURL": "https://apple.com", - "requiredMinimumOSVersion": "latest", + "requiredMinimumOSVersion": "15.99", + "requiredInstallationDate": "2025-01-01T00:00:00", "unsupportedURL": "https://google.com" } ], diff --git a/Nudge/Utilities/UILogic.swift b/Nudge/Utilities/UILogic.swift index cd2e2d9b..94c028e1 100644 --- a/Nudge/Utilities/UILogic.swift +++ b/Nudge/Utilities/UILogic.swift @@ -260,7 +260,8 @@ func isDownloadingOrPreparingSoftwareUpdate() -> Bool { ("softwareupdated", ["/System/Library/PrivateFrameworks/MobileSoftwareUpdate.framework/Support/softwareupdated"]), // When downloading a minor update, this process is running. ("installcoordinationd", ["/System/Library/PrivateFrameworks/InstallCoordination.framework/Support/installcoordinationd"]), // When preparing a minor update, this process is running. Unfortunately, after preparing the update, this process appears to stay running. ("softwareupdate", ["/usr/bin/softwareupdate", "--fetch-full-installer"]), // When downloading a major upgrade via SoftwareUpdate prefpane, it triggers a --fetch-full-installer run. Nudge also performs this method. - ("osinstallersetupd" ,["/Applications/*Install macOS *.app/Contents/Frameworks/OSInstallerSetup.framework/Resources/osinstallersetupd"]), // When installing a major upgrade, this process is running. + ("osinstallersetupd", ["/Applications/*Install macOS *.app/Contents/Frameworks/OSInstallerSetup.framework/Resources/osinstallersetupd"]), // When installing a major upgrade, this process is running. + ("installerauthage", ["/System/Library/PrivateFrameworks/IASUtilities.framework/Versions/A/Resources/installerauthagent", "/System/Library/PrivateFrameworks/IASUtilities.framework/Versions/A/Resources/installerauthagent"]), // Possibly on macOS 15, this is running when preparing an update. // /System/Library/PrivateFrameworks/PackageKit.framework/Resources/installd||system_installd - system_installd may be interesting, but I think installd is being used for any package ] return isAnyProcessRunning(commandsWithArgs: commandsWithArgs) From 16f4090c762cd02fe0f72bf664c9b61d32ad3c13 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 10 Jul 2024 16:15:48 -0500 Subject: [PATCH 118/141] grab full path of the process and compare to relative path --- Nudge/Utilities/UILogic.swift | 21 ++++++++++++++------- 1 file changed, 14 insertions(+), 7 deletions(-) diff --git a/Nudge/Utilities/UILogic.swift b/Nudge/Utilities/UILogic.swift index 94c028e1..b9531658 100644 --- a/Nudge/Utilities/UILogic.swift +++ b/Nudge/Utilities/UILogic.swift @@ -200,12 +200,17 @@ func getAllProcesses() -> [ProcessInfoStruct] { // Extract process info for process in processList { - let command = withUnsafePointer(to: process.kp_proc.p_comm) { + let pid = process.kp_proc.p_pid + + // Get full command path + var pathBuffer = [CChar](repeating: 0, count: Int(PATH_MAX)) + let result = proc_pidpath(pid, &pathBuffer, UInt32(PATH_MAX)) + let command = result > 0 ? String(cString: pathBuffer) : withUnsafePointer(to: process.kp_proc.p_comm) { $0.withMemoryRebound(to: CChar.self, capacity: Int(MAXCOMLEN)) { String(cString: $0) } } - let pid = process.kp_proc.p_pid + let arguments = getArgumentsForPID(pid: pid) processes.append(ProcessInfoStruct(pid: pid, command: command, arguments: arguments)) } @@ -239,11 +244,12 @@ func getArgumentsForPID(pid: Int32) -> [String] { return args } -func isAnyProcessRunning(commandsWithArgs: [(commandPattern: String, arguments: [String]?)]) -> Bool { +func isAnyProcessRunning(commandsWithArgs: [(processRelativeName: String, arguments: [String]?)]) -> Bool { let processes = getAllProcesses() - for (commandPattern, arguments) in commandsWithArgs { + for (processRelativeName, arguments) in commandsWithArgs { let matchingProcesses = processes.filter { process in - fnmatch(commandPattern, process.command, 0) == 0 && + // Check if the command pattern is a substring of the full command path + process.command.lowercased().contains(processRelativeName.lowercased()) && (arguments == nil || arguments!.allSatisfy { arg in process.arguments.contains(where: { $0.contains(arg) }) }) @@ -256,12 +262,13 @@ func isAnyProcessRunning(commandsWithArgs: [(commandPattern: String, arguments: } func isDownloadingOrPreparingSoftwareUpdate() -> Bool { - let commandsWithArgs: [(commandPattern: String, arguments: [String]?)] = [ + let commandsWithArgs: [(processRelativeName: String, arguments: [String]?)] = [ ("softwareupdated", ["/System/Library/PrivateFrameworks/MobileSoftwareUpdate.framework/Support/softwareupdated"]), // When downloading a minor update, this process is running. ("installcoordinationd", ["/System/Library/PrivateFrameworks/InstallCoordination.framework/Support/installcoordinationd"]), // When preparing a minor update, this process is running. Unfortunately, after preparing the update, this process appears to stay running. ("softwareupdate", ["/usr/bin/softwareupdate", "--fetch-full-installer"]), // When downloading a major upgrade via SoftwareUpdate prefpane, it triggers a --fetch-full-installer run. Nudge also performs this method. + ("softwareupdate", ["/usr/sbin/softwareupdate", "--fetch-full-installer"]), // When downloading a major upgrade via softwareupdate cli, it triggers a --fetch-full-installer run. Nudge also performs this method. ("osinstallersetupd", ["/Applications/*Install macOS *.app/Contents/Frameworks/OSInstallerSetup.framework/Resources/osinstallersetupd"]), // When installing a major upgrade, this process is running. - ("installerauthage", ["/System/Library/PrivateFrameworks/IASUtilities.framework/Versions/A/Resources/installerauthagent", "/System/Library/PrivateFrameworks/IASUtilities.framework/Versions/A/Resources/installerauthagent"]), // Possibly on macOS 15, this is running when preparing an update. + ("installerauthagent", ["/System/Library/PrivateFrameworks/IASUtilities.framework/Versions/A/Resources/installerauthagent", "/System/Library/PrivateFrameworks/IASUtilities.framework/Versions/A/Resources/installerauthagent"]), // Possibly on macOS 15, this is running when preparing an update. // /System/Library/PrivateFrameworks/PackageKit.framework/Resources/installd||system_installd - system_installd may be interesting, but I think installd is being used for any package ] return isAnyProcessRunning(commandsWithArgs: commandsWithArgs) From 93ce34ab1b307c5266f6ea8a4a80756fefe452f1 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 10 Jul 2024 17:04:47 -0500 Subject: [PATCH 119/141] add major and minor SLAs for sofa --- CHANGELOG.md | 9 ++- .../Preferences/DefaultPreferencesNudge.swift | 36 +++++++--- Nudge/Preferences/PreferencesStructure.swift | 36 ++++++---- Nudge/UI/Main.swift | 26 ++++--- Schema/jamf/com.github.macadmins.Nudge.json | 69 +++++++++++++++++-- 5 files changed, 136 insertions(+), 40 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 0167dc45..4b7265ab 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -21,9 +21,12 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - `latest-minor`: stay in the current major release and get the latest minor updates available - This requires utilizing the SOFA feed features to properly work, which is opt-out by default - Nudge will then utilize two date integers to automatically calculate the `requiredInstallationDate` - - `activelyExploitedCVEsInstallationSLA` under the `osVersionRequirement` key will default to 14 days - - `nonActivelyExploitedCVEsSLA` under the `osVersionRequirement` key will default to 21 days - - `standardInstallationSLA` under the `osVersionRequirement` key will default to 28 days + - `activelyExploitedCVEsMajorUpgradeSLA` under the `osVersionRequirement` key will default to 14 days + - `activelyExploitedCVEsMinorUpdateSLA` under the `osVersionRequirement` key will default to 14 days + - `nonActivelyExploitedCVEsMajorUpgradeSLA` under the `osVersionRequirement` key will default to 21 days + - `nonActivelyExploitedCVEsMinorUpdateSLA` under the `osVersionRequirement` key will default to 21 days + - `standardMajorUpgradeSLA` under the `osVersionRequirement` key will default to 28 days + - `standardMinorUpdateSLA` under the `osVersionRequirement` key will default to 28 days - These dates are calculated against the `ReleaseDate` key in the SOFA feed, which is UTC formatted. Local timezones will **not be supported** with the automatic sofa feed unless you use a custom feed and change this value yourself, following ISO-8601 date formats - To artificially delay the SOFA nudge events, see the details below for `nudgeEventLaunchDelay` - If you'd like to not have nudge events for releases without any known CVEs, please configure the `disableNudgeForStandardInstalls` key under `optionalFeatures` to true diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 65798feb..d04b10ac 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -207,9 +207,15 @@ struct OSVersionRequirementVariables { "" } - static var activelyExploitedCVEsInstallationSLA: Int { - osVersionRequirementsProfile?.activelyExploitedCVEsInstallationSLA ?? - osVersionRequirementsJSON?.activelyExploitedCVEsInstallationSLA ?? + static var activelyExploitedCVEsMajorUpgradeSLA: Int { + osVersionRequirementsProfile?.activelyExploitedCVEsMajorUpgradeSLA ?? + osVersionRequirementsJSON?.activelyExploitedCVEsMajorUpgradeSLA ?? + 14 + } + + static var activelyExploitedCVEsMinorUpdateSLA: Int { + osVersionRequirementsProfile?.activelyExploitedCVEsMinorUpdateSLA ?? + osVersionRequirementsJSON?.activelyExploitedCVEsMinorUpdateSLA ?? 14 } @@ -219,9 +225,15 @@ struct OSVersionRequirementVariables { "" } - static var nonActivelyExploitedCVEsSLA: Int { - osVersionRequirementsProfile?.nonActivelyExploitedCVEsSLA ?? - osVersionRequirementsJSON?.nonActivelyExploitedCVEsSLA ?? + static var nonActivelyExploitedCVEsMajorUpgradeSLA: Int { + osVersionRequirementsProfile?.nonActivelyExploitedCVEsMajorUpgradeSLA ?? + osVersionRequirementsJSON?.nonActivelyExploitedCVEsMajorUpgradeSLA ?? + 21 + } + + static var nonActivelyExploitedCVEsMinorUpdateSLA: Int { + osVersionRequirementsProfile?.nonActivelyExploitedCVEsMinorUpdateSLA ?? + osVersionRequirementsJSON?.nonActivelyExploitedCVEsMinorUpdateSLA ?? 21 } @@ -233,9 +245,15 @@ struct OSVersionRequirementVariables { } } - static var standardInstallationSLA: Int { - osVersionRequirementsProfile?.standardInstallationSLA ?? - osVersionRequirementsJSON?.standardInstallationSLA ?? + static var standardMajorUpgradeSLA: Int { + osVersionRequirementsProfile?.standardMajorUpgradeSLA ?? + osVersionRequirementsJSON?.standardMajorUpgradeSLA ?? + 28 + } + + static var standardMinorUpdateSLA: Int { + osVersionRequirementsProfile?.standardMinorUpdateSLA ?? + osVersionRequirementsJSON?.standardMinorUpdateSLA ?? 28 } diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index bac9c632..bacfe162 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -148,12 +148,15 @@ struct OSVersionRequirement: Codable { var aboutUpdateURL: String? var aboutUpdateURLs: [AboutUpdateURL]? var actionButtonPath: String? - var activelyExploitedCVEsInstallationSLA: Int? + var activelyExploitedCVEsMajorUpgradeSLA: Int? + var activelyExploitedCVEsMinorUpdateSLA: Int? var majorUpgradeAppPath: String? - var nonActivelyExploitedCVEsSLA: Int? + var nonActivelyExploitedCVEsMajorUpgradeSLA: Int? + var nonActivelyExploitedCVEsMinorUpdateSLA: Int? var requiredInstallationDate: Date? var requiredMinimumOSVersion: String? - var standardInstallationSLA: Int? + var standardMajorUpgradeSLA: Int? + var standardMinorUpdateSLA: Int? var targetedOSVersionsRule: String? var unsupportedURL: String? var unsupportedURLs: [UnsupportedURL]? @@ -164,11 +167,14 @@ extension OSVersionRequirement { init(fromDictionary: [String: AnyObject]) { self.aboutUpdateURL = fromDictionary["aboutUpdateURL"] as? String self.actionButtonPath = fromDictionary["actionButtonPath"] as? String - self.activelyExploitedCVEsInstallationSLA = fromDictionary["activelyExploitedCVEsInstallationSLA"] as? Int + self.activelyExploitedCVEsMajorUpgradeSLA = fromDictionary["activelyExploitedCVEsMajorUpgradeSLA"] as? Int + self.activelyExploitedCVEsMinorUpdateSLA = fromDictionary["activelyExploitedCVEsMinorUpdateSLA"] as? Int self.majorUpgradeAppPath = fromDictionary["majorUpgradeAppPath"] as? String - self.nonActivelyExploitedCVEsSLA = fromDictionary["nonActivelyExploitedCVEsSLA"] as? Int + self.nonActivelyExploitedCVEsMajorUpgradeSLA = fromDictionary["nonActivelyExploitedCVEsMajorUpgradeSLA"] as? Int + self.nonActivelyExploitedCVEsMinorUpdateSLA = fromDictionary["nonActivelyExploitedCVEsMinorUpdateSLA"] as? Int self.requiredMinimumOSVersion = fromDictionary["requiredMinimumOSVersion"] as? String - self.standardInstallationSLA = fromDictionary["standardInstallationSLA"] as? Int + self.standardMajorUpgradeSLA = fromDictionary["standardMajorUpgradeSLA"] as? Int + self.standardMinorUpdateSLA = fromDictionary["standardMinorUpdateSLA"] as? Int self.targetedOSVersionsRule = fromDictionary["targetedOSVersionsRule"] as? String self.unsupportedURL = fromDictionary["unsupportedURL"] as? String @@ -239,12 +245,15 @@ extension OSVersionRequirement { aboutUpdateURL: String? = nil, aboutUpdateURLs: [AboutUpdateURL]? = nil, actionButtonPath: String? = nil, - activelyExploitedCVEsInstallationSLA: Int? = nil, + activelyExploitedCVEsMajorUpgradeSLA: Int? = nil, + activelyExploitedCVEsMinorUpdateSLA: Int? = nil, majorUpgradeAppPath: String? = nil, - nonActivelyExploitedCVEsSLA: Int? = nil, + nonActivelyExploitedCVEsMajorUpgradeSLA: Int? = nil, + nonActivelyExploitedCVEsMinorUpdateSLA: Int? = nil, requiredInstallationDate: Date? = nil, requiredMinimumOSVersion: String? = nil, - standardInstallationSLA: Int? = nil, + standardMajorUpgradeSLA: Int? = nil, + standardMinorUpdateSLA: Int? = nil, targetedOSVersionsRule: String? = nil, unsupportedURL: String? = nil, unsupportedURLs: [UnsupportedURL]? = nil @@ -253,12 +262,15 @@ extension OSVersionRequirement { aboutUpdateURL: aboutUpdateURL ?? self.aboutUpdateURL, aboutUpdateURLs: aboutUpdateURLs ?? self.aboutUpdateURLs, actionButtonPath: actionButtonPath ?? self.actionButtonPath, - activelyExploitedCVEsInstallationSLA: activelyExploitedCVEsInstallationSLA ?? self.activelyExploitedCVEsInstallationSLA, + activelyExploitedCVEsMajorUpgradeSLA: activelyExploitedCVEsMajorUpgradeSLA ?? self.activelyExploitedCVEsMajorUpgradeSLA, + activelyExploitedCVEsMinorUpdateSLA: activelyExploitedCVEsMinorUpdateSLA ?? self.activelyExploitedCVEsMinorUpdateSLA, majorUpgradeAppPath: majorUpgradeAppPath ?? self.majorUpgradeAppPath, - nonActivelyExploitedCVEsSLA: nonActivelyExploitedCVEsSLA ?? self.nonActivelyExploitedCVEsSLA, + nonActivelyExploitedCVEsMajorUpgradeSLA: nonActivelyExploitedCVEsMajorUpgradeSLA ?? self.nonActivelyExploitedCVEsMajorUpgradeSLA, + nonActivelyExploitedCVEsMinorUpdateSLA: nonActivelyExploitedCVEsMinorUpdateSLA ?? self.nonActivelyExploitedCVEsMinorUpdateSLA, requiredInstallationDate: requiredInstallationDate ?? self.requiredInstallationDate, requiredMinimumOSVersion: requiredMinimumOSVersion ?? self.requiredMinimumOSVersion, - standardInstallationSLA: standardInstallationSLA ?? self.standardInstallationSLA, + standardMajorUpgradeSLA: standardMajorUpgradeSLA ?? self.standardMajorUpgradeSLA, + standardMinorUpdateSLA: standardMinorUpdateSLA ?? self.standardMinorUpdateSLA, targetedOSVersionsRule: targetedOSVersionsRule ?? self.targetedOSVersionsRule, unsupportedURL: unsupportedURL ?? self.unsupportedURL, unsupportedURLs: unsupportedURLs ?? self.unsupportedURLs diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 848176b9..4c704f41 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -207,15 +207,21 @@ class AppDelegate: NSObject, NSApplicationDelegate { let activelyExploitedCVEs = selectedOS!.activelyExploitedCVEs.count > 0 let presentCVEs = selectedOS!.cves.count > 0 let slaExtension: TimeInterval - switch (activelyExploitedCVEs, presentCVEs) { - case (false, true): - slaExtension = TimeInterval(OSVersionRequirementVariables.nonActivelyExploitedCVEsSLA * 86400) - case (true, true): - slaExtension = TimeInterval(OSVersionRequirementVariables.activelyExploitedCVEsInstallationSLA * 86400) - case (false, false): - slaExtension = TimeInterval(OSVersionRequirementVariables.standardInstallationSLA * 86400) - default: - slaExtension = TimeInterval(OSVersionRequirementVariables.standardInstallationSLA * 86400) + switch (activelyExploitedCVEs, presentCVEs, AppStateManager().requireMajorUpgrade()) { + case (false, true, true): + slaExtension = TimeInterval(OSVersionRequirementVariables.nonActivelyExploitedCVEsMajorUpgradeSLA * 86400) + case (false, true, false): + slaExtension = TimeInterval(OSVersionRequirementVariables.nonActivelyExploitedCVEsMinorUpdateSLA * 86400) + case (true, true, true): + slaExtension = TimeInterval(OSVersionRequirementVariables.activelyExploitedCVEsMajorUpgradeSLA * 86400) + case (true, true, false): + slaExtension = TimeInterval(OSVersionRequirementVariables.activelyExploitedCVEsMinorUpdateSLA * 86400) + case (false, false, true): + slaExtension = TimeInterval(OSVersionRequirementVariables.standardMajorUpgradeSLA * 86400) + case (false, false, false): + slaExtension = TimeInterval(OSVersionRequirementVariables.standardMinorUpdateSLA * 86400) + default: // If we get here, something is wrong, use 90 days as a safety + slaExtension = TimeInterval(90 * 86400) } if OptionalFeatureVariables.disableNudgeForStandardInstalls && !presentCVEs { @@ -228,7 +234,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { nudgePrimaryState.requiredMinimumOSVersion = osVersion.latest.productVersion nudgePrimaryState.activelyExploitedCVEs = activelyExploitedCVEs releaseDate = selectedOS!.releaseDate ?? Date() - requiredInstallationDate = selectedOS!.releaseDate?.addingTimeInterval(slaExtension) ?? DateManager().getCurrentDate().addingTimeInterval(TimeInterval(OSVersionRequirementVariables.standardInstallationSLA * 86400)) + requiredInstallationDate = selectedOS!.releaseDate?.addingTimeInterval(slaExtension) ?? DateManager().getCurrentDate().addingTimeInterval(TimeInterval(90 * 86400)) LogManager.notice("Extending requiredInstallationDate to \(requiredInstallationDate)", logger: sofaLog) LogManager.notice("SOFA Matched OS Version: \(selectedOS!.productVersion)", logger: sofaLog) diff --git a/Schema/jamf/com.github.macadmins.Nudge.json b/Schema/jamf/com.github.macadmins.Nudge.json index ecb961f0..2639be7e 100644 --- a/Schema/jamf/com.github.macadmins.Nudge.json +++ b/Schema/jamf/com.github.macadmins.Nudge.json @@ -452,8 +452,27 @@ } ] }, - "activelyExploitedCVEsInstallationSLA": { - "description": "When an update is under active exploit, this is the amount of days a user has to install the update. (Note: This key is only used with Nudge v2.0 and higher)", + "activelyExploitedCVEsMajorUpgradeSLA": { + "description": "When a major upgrade is under active exploit, this is the amount of days a user has to install the update. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "title": "Not Configured", + "type": "null" + }, + { + "title": "Configured", + "default": 14, + "type": "integer", + "options": { + "inputAttributes": { + "placeholder": "14" + } + } + } + ] + }, + "activelyExploitedCVEsMinorUpdateSLA": { + "description": "When a minor update is under active exploit, this is the amount of days a user has to install the update. (Note: This key is only used with Nudge v2.0 and higher)", "anyOf": [ { "title": "Not Configured", @@ -489,8 +508,27 @@ } ] }, - "nonActivelyExploitedCVEsSLA": { - "description": "When an update is not under active exploit but contains CVEs, this is the amount of days a user has to install the update. (Note: This key is only used with Nudge v2.0 and higher)", + "nonActivelyExploitedCVEsMajorUpgradeSLA": { + "description": "When a major upgrade is not under active exploit but contains CVEs, this is the amount of days a user has to install the update. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "title": "Not Configured", + "type": "null" + }, + { + "title": "Configured", + "default": 21, + "type": "integer", + "options": { + "inputAttributes": { + "placeholder": "21" + } + } + } + ] + }, + "nonActivelyExploitedCVEsMinorUpdateSLA": { + "description": "When a minor update is not under active exploit but contains CVEs, this is the amount of days a user has to install the update. (Note: This key is only used with Nudge v2.0 and higher)", "anyOf": [ { "title": "Not Configured", @@ -544,8 +582,27 @@ } ] }, - "standardInstallationSLA": { - "description": "When an update has no known CVEs, this is the amount of days a user has to install the update. (Note: This key is only used with Nudge v2.0 and higher)", + "standardMajorUpgradeSLA": { + "description": "When a major upgrade has no known CVEs, this is the amount of days a user has to install the update. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "title": "Not Configured", + "type": "null" + }, + { + "title": "Configured", + "default": 28, + "type": "integer", + "options": { + "inputAttributes": { + "placeholder": "28" + } + } + } + ] + }, + "standardMinorupdateSLA": { + "description": "When a minor update has no known CVEs, this is the amount of days a user has to install the update. (Note: This key is only used with Nudge v2.0 and higher)", "anyOf": [ { "title": "Not Configured", From 29ce1c6b299437c36c3625cde2a44060f1570825 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 10 Jul 2024 20:15:46 -0500 Subject: [PATCH 120/141] add changelog into the project so I can modify within xcode --- Nudge.xcodeproj/project.pbxproj | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/Nudge.xcodeproj/project.pbxproj b/Nudge.xcodeproj/project.pbxproj index c51b9710..ece1e934 100644 --- a/Nudge.xcodeproj/project.pbxproj +++ b/Nudge.xcodeproj/project.pbxproj @@ -19,6 +19,7 @@ 5836861C25DAD01C0004514C /* SoftwareUpdate.swift in Sources */ = {isa = PBXBuildFile; fileRef = 5836861B25DAD01C0004514C /* SoftwareUpdate.swift */; }; 6316F0E72832CA0700E1354D /* Schema in Resources */ = {isa = PBXBuildFile; fileRef = 6316F0E62832CA0700E1354D /* Schema */; }; 631A6D762BF2654000DC1EF3 /* sofa.swift in Sources */ = {isa = PBXBuildFile; fileRef = 631A6D752BF2654000DC1EF3 /* sofa.swift */; }; + 633C4C712C3F6465005720F0 /* CHANGELOG.md in Resources */ = {isa = PBXBuildFile; fileRef = 633C4C702C3F6465005720F0 /* CHANGELOG.md */; }; 6347351D2B45DC2400C3401D /* CloseButton.swift in Sources */ = {isa = PBXBuildFile; fileRef = 6347351C2B45DC2400C3401D /* CloseButton.swift */; }; 634CE1092BB47480002C26C4 /* gdmf.swift in Sources */ = {isa = PBXBuildFile; fileRef = 634CE1082BB47480002C26C4 /* gdmf.swift */; }; 634CE10A2BB47480002C26C4 /* gdmf.swift in Sources */ = {isa = PBXBuildFile; fileRef = 634CE1082BB47480002C26C4 /* gdmf.swift */; }; @@ -96,6 +97,7 @@ 5836861B25DAD01C0004514C /* SoftwareUpdate.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = SoftwareUpdate.swift; sourceTree = ""; }; 6316F0E62832CA0700E1354D /* Schema */ = {isa = PBXFileReference; lastKnownFileType = folder; path = Schema; sourceTree = ""; }; 631A6D752BF2654000DC1EF3 /* sofa.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = sofa.swift; sourceTree = ""; }; + 633C4C702C3F6465005720F0 /* CHANGELOG.md */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = net.daringfireball.markdown; path = CHANGELOG.md; sourceTree = SOURCE_ROOT; }; 6347351C2B45DC2400C3401D /* CloseButton.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = CloseButton.swift; sourceTree = ""; }; 634CE1082BB47480002C26C4 /* gdmf.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = gdmf.swift; sourceTree = ""; }; 636B9C0126CACCAB0007BE3B /* DeferView.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = DeferView.swift; sourceTree = ""; }; @@ -252,6 +254,7 @@ 63D7D0D625C9E9A400236281 = { isa = PBXGroup; children = ( + 633C4C702C3F6465005720F0 /* CHANGELOG.md */, 031B0F2125D8AE3200E68A28 /* Example Assets */, 6316F0E62832CA0700E1354D /* Schema */, 637CEBC02A30C9E700EFA3E9 /* Localizable.xcstrings */, @@ -474,6 +477,7 @@ isa = PBXResourcesBuildPhase; buildActionMask = 2147483647; files = ( + 633C4C712C3F6465005720F0 /* CHANGELOG.md in Resources */, ); runOnlyForDeploymentPostprocessing = 0; }; From ba316452b4ee1a7d9de3bf7dcb974ee234d31025 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 10 Jul 2024 20:16:11 -0500 Subject: [PATCH 121/141] nudgeEventLaunchDelay is now nudgeMajorUpgradeEventLaunchDelay and nudgeMinorUpdateEventLaunchDelay --- CHANGELOG.md | 4 ++-- .../Preferences/DefaultPreferencesNudge.swift | 12 +++++++--- Nudge/Preferences/PreferencesStructure.swift | 9 +++++--- Nudge/Utilities/Utils.swift | 14 +++++++---- Schema/jamf/com.github.macadmins.Nudge.json | 23 +++++++++++++++++-- 5 files changed, 48 insertions(+), 14 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4b7265ab..58321686 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -28,7 +28,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - `standardMajorUpgradeSLA` under the `osVersionRequirement` key will default to 28 days - `standardMinorUpdateSLA` under the `osVersionRequirement` key will default to 28 days - These dates are calculated against the `ReleaseDate` key in the SOFA feed, which is UTC formatted. Local timezones will **not be supported** with the automatic sofa feed unless you use a custom feed and change this value yourself, following ISO-8601 date formats - - To artificially delay the SOFA nudge events, see the details below for `nudgeEventLaunchDelay` + - To artificially delay the SOFA nudge events, see the details below for `nudgeMajorUpgradeEventLaunchDelay` and `nudgeMinorUpdateEventLaunchDelay` - If you'd like to not have nudge events for releases without any known CVEs, please configure the `disableNudgeForStandardInstalls` key under `optionalFeatures` to true - You can now disable the `Days Remaining To Update:` item on the left side of the UI. - Configure the `showDaysRemainingToUpdate` key under `userInterface` to false @@ -47,7 +47,7 @@ Requires macOS 12.0 and higher. Further releases and feature requests may make t - Issue [475](https://github.com/macadmins/nudge/issues/475) ### Added -- To artificially change the `requredInstallationDate` thereby giving your users a default grace period for all Nudge events updates, please configure the `nudgeEventLaunchDelay` key under `userExperience` +- To artificially change the `requredInstallationDate` thereby giving your users a default grace period for all Nudge events updates, please configure the `nudgeMajorUpgradeEventLaunchDelay` and `nudgeMinorUpdateEventLaunchDelay` keys under `userExperience` in amount of days. - A local image path can now be specified for the notification event when Nudge terminates and application - Please configure the `applicationTerminatedNotificationImagePath` key under `userInterface` - Due to limitations within Apple's API, a local path is only supported at this time diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index d04b10ac..049196f8 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -388,9 +388,15 @@ struct UserExperienceVariables { false } - static var nudgeEventLaunchDelay: Int { - userExperienceProfile?["nudgeEventLaunchDelay"] as? Int ?? - userExperienceJSON?.nudgeEventLaunchDelay ?? + static var nudgeMinorUpdateEventLaunchDelay: Int { + userExperienceProfile?["nudgeMinorUpdateEventLaunchDelay"] as? Int ?? + userExperienceJSON?.nudgeMinorUpdateEventLaunchDelay ?? + 0 + } + + static var nudgeMajorUpgradeEventLaunchDelay: Int { + userExperienceProfile?["nudgeMajorUpgradeEventLaunchDelay"] as? Int ?? + userExperienceJSON?.nudgeMajorUpgradeEventLaunchDelay ?? 0 } diff --git a/Nudge/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index bacfe162..085b6e8b 100644 --- a/Nudge/Preferences/PreferencesStructure.swift +++ b/Nudge/Preferences/PreferencesStructure.swift @@ -370,7 +370,8 @@ struct UserExperience: Codable { var loadLaunchAgent: Bool? var maxRandomDelayInSeconds: Int? var noTimers: Bool? - var nudgeEventLaunchDelay: Int? + var nudgeMajorUpgradeEventLaunchDelay: Int? + var nudgeMinorUpdateEventLaunchDelay: Int? var nudgeRefreshCycle: Int? var randomDelay: Bool? } @@ -414,7 +415,8 @@ extension UserExperience { loadLaunchAgent: Bool? = nil, maxRandomDelayInSeconds: Int? = nil, noTimers: Bool? = nil, - nudgeEventLaunchDelay: Int? = nil, + nudgeMajorUpgradeEventLaunchDelay: Int? = nil, + nudgeMinorUpdateEventLaunchDelay: Int? = nil, nudgeRefreshCycle: Int? = nil, randomDelay: Bool? = nil ) -> UserExperience { @@ -439,7 +441,8 @@ extension UserExperience { loadLaunchAgent: loadLaunchAgent ?? self.loadLaunchAgent, maxRandomDelayInSeconds: maxRandomDelayInSeconds ?? self.maxRandomDelayInSeconds, noTimers: noTimers ?? self.noTimers, - nudgeEventLaunchDelay: nudgeEventLaunchDelay ?? self.nudgeEventLaunchDelay, + nudgeMajorUpgradeEventLaunchDelay: nudgeMajorUpgradeEventLaunchDelay ?? self.nudgeMajorUpgradeEventLaunchDelay, + nudgeMinorUpdateEventLaunchDelay: nudgeMinorUpdateEventLaunchDelay ?? self.nudgeMinorUpdateEventLaunchDelay, nudgeRefreshCycle: nudgeRefreshCycle ?? self.nudgeRefreshCycle, randomDelay: randomDelay ?? self.randomDelay ) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 8b453ad3..6f24756b 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -185,15 +185,21 @@ struct AppStateManager { } func delayNudgeEventLogic(currentDate: Date = DateManager().getCurrentDate(), testFileDate: Date? = nil) -> Date { - if UserExperienceVariables.nudgeEventLaunchDelay == 0 { + let isMajorUpgradeRequired = AppStateManager().requireMajorUpgrade() + let launchDelay = isMajorUpgradeRequired ? UserExperienceVariables.nudgeMajorUpgradeEventLaunchDelay : UserExperienceVariables.nudgeMinorUpdateEventLaunchDelay + + if launchDelay == 0 { return PrefsWrapper.requiredInstallationDate } - if releaseDate.addingTimeInterval(TimeInterval(UserExperienceVariables.nudgeEventLaunchDelay * 86400)) > currentDate { - LogManager.info("Device within nudgeEventLaunchDelay, exiting Nudge", logger: uiLog) + + if releaseDate.addingTimeInterval(TimeInterval(launchDelay * 86400)) > currentDate { + let eventType = isMajorUpgradeRequired ? "nudgeMajorUpgradeEventLaunchDelay" : "nudgeMinorUpdateEventLaunchDelay" + LogManager.info("Device within \(eventType)", logger: uiLog) nudgePrimaryState.shouldExit = true return currentDate } else { - LogManager.info("Device outside nudgeEventLaunchDelay", logger: uiLog) + let eventType = isMajorUpgradeRequired ? "nudgeMajorUpgradeEventLaunchDelay" : "nudgeMinorUpdateEventLaunchDelay" + LogManager.info("Device outside \(eventType)", logger: uiLog) return PrefsWrapper.requiredInstallationDate } } diff --git a/Schema/jamf/com.github.macadmins.Nudge.json b/Schema/jamf/com.github.macadmins.Nudge.json index 2639be7e..9f718656 100644 --- a/Schema/jamf/com.github.macadmins.Nudge.json +++ b/Schema/jamf/com.github.macadmins.Nudge.json @@ -1083,8 +1083,27 @@ } ] }, - "nudgeEventLaunchDelay": { - "description": "When a new update is posted to the SOFA feed, this can artificially delay the SOFA nudge events by x amount of days. (Note: This key is only used with Nudge v2.0 and higher)", + "nudgeMajorUpgradeEventLaunchDelay": { + "description": "When a new major upgrade is posted to the SOFA feed, this can artificially delay the SOFA nudge events by x amount of days. (Note: This key is only used with Nudge v2.0 and higher)", + "anyOf": [ + { + "title": "Not Configured", + "type": "null" + }, + { + "title": "Configured", + "default": 0, + "type": "integer", + "options": { + "inputAttributes": { + "placeholder": "0" + } + } + } + ] + }, + "nudgeMinorUpdateEventLaunchDelay": { + "description": "When a new minor update is posted to the SOFA feed, this can artificially delay the SOFA nudge events by x amount of days. (Note: This key is only used with Nudge v2.0 and higher)", "anyOf": [ { "title": "Not Configured", From 9871cd25b23c6ae20590c06515a0efcfda42a207 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Thu, 11 Jul 2024 09:53:50 -0500 Subject: [PATCH 122/141] add compression to potentially reduce SOFA bw --- Nudge/3rd Party Assets/sofa.swift | 1 + 1 file changed, 1 insertion(+) diff --git a/Nudge/3rd Party Assets/sofa.swift b/Nudge/3rd Party Assets/sofa.swift index cee37fde..5e1bf548 100644 --- a/Nudge/3rd Party Assets/sofa.swift +++ b/Nudge/3rd Party Assets/sofa.swift @@ -232,6 +232,7 @@ class SOFA: NSObject, URLSessionDelegate { request.addValue("\(Globals.bundleID)/\(VersionManager.getNudgeVersion())", forHTTPHeaderField: "User-Agent") request.setValue(lastEtag, forHTTPHeaderField: "If-None-Match") // TODO: I'm saving the Etag and sending it, but due to forcing this into a syncronous call, it is always returning a 200 code. When using this in an asycronous method, it eventually returns the 304 response. I'm not sure how to fix this bug. + request.addValue("gzip, deflate, br", forHTTPHeaderField: "Accept-Encoding") // Force compression for JSON var attempts = 0 var responseData: Data? From 5f7e24c9208d57e924806e83c775b4d8d201056c Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Thu, 11 Jul 2024 22:03:22 -0500 Subject: [PATCH 123/141] Update CHANGELOG.md --- CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 58321686..a97e9642 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,7 +4,7 @@ All notable changes to this project will be documented in this file. The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). -## [2.0.0] - 2024-05-21 +## [2.0.0] - 2024-07-17 Requires macOS 12.0 and higher. Further releases and feature requests may make this macOS 13 and higher depending on code complexity. ### Changed From 3bf57e6aa93c245169a64dc67d57b0bcbeb1220c Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Thu, 11 Jul 2024 22:04:12 -0500 Subject: [PATCH 124/141] Update CHANGELOG.md --- CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index a97e9642..0a89a490 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -5,7 +5,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). ## [2.0.0] - 2024-07-17 -Requires macOS 12.0 and higher. Further releases and feature requests may make this macOS 13 and higher depending on code complexity. +Requires macOS 12.0 and higher. ### Changed - Now built on Swift 5.10, Xcode 15.4 and macOS 14 From 43e62e043ad0d17ff1245fe09f4424b9f0628340 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 15 Jul 2024 08:58:40 -0500 Subject: [PATCH 125/141] add breaking change notes --- CHANGELOG.md | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 0a89a490..3257de97 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -7,6 +7,11 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 ## [2.0.0] - 2024-07-17 Requires macOS 12.0 and higher. +### Breaking Changes +- Due to implementing markdown support, many of the elements within Nudge may no longer be in **bold** if you customize them. + - To work around this please add `**` elements to these customizations + - For example: The `mainContentNote` value of `Important Notes` would become `**Important Notes**` + ### Changed - Now built on Swift 5.10, Xcode 15.4 and macOS 14 - New Xcode Scheme `-bundle-mode-profile` to test profile logic From 6101b301bffb8bd50a8422e58cae033b1e9fa9bd Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 15 Jul 2024 09:02:47 -0500 Subject: [PATCH 126/141] add another note about sofa feed --- CHANGELOG.md | 1 + 1 file changed, 1 insertion(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 3257de97..f1754ca9 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -11,6 +11,7 @@ Requires macOS 12.0 and higher. - Due to implementing markdown support, many of the elements within Nudge may no longer be in **bold** if you customize them. - To work around this please add `**` elements to these customizations - For example: The `mainContentNote` value of `Important Notes` would become `**Important Notes**` +- The SOFA feed is **opt-out**, which included the new `Unsupported UI`. If you do not want the Unsupported UI features, you will need to actively opt-out of these options. ### Changed - Now built on Swift 5.10, Xcode 15.4 and macOS 14 From 11b6adfd0dd0a343ed18c4527d13ddd5fc3f52e0 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 15 Jul 2024 09:18:27 -0500 Subject: [PATCH 127/141] retry when json is malformed --- Nudge/Utilities/Utils.swift | 44 ++++++++++++++++++++++++++++++++----- 1 file changed, 38 insertions(+), 6 deletions(-) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 6f24756b..5e04fc38 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -996,7 +996,7 @@ struct NetworkFileManager { if sofaJSONExists { let sofaPathCreationDate = AppStateManager().getModifiedDateForPath(sofaPath.path, testFileDate: nil) - // Use Cache as it is within time inverval + // Use Cache as it is within time interval if TimeInterval(OptionalFeatureVariables.refreshSOFAFeedTime) >= Date().timeIntervalSince(sofaPathCreationDate!) { LogManager.info("Utilizing previously cached SOFA json", logger: sofaLog) do { @@ -1006,7 +1006,8 @@ struct NetworkFileManager { } catch { LogManager.error("Failed to decode previously cached local sofa JSON: \(error.localizedDescription)", logger: sofaLog) LogManager.error("Failed to decode sofa JSON: \(error)", logger: sofaLog) - return nil + // Attempt to redownload and reprocess the file + return redownloadAndReprocessSOFA(url: URL(string: OptionalFeatureVariables.customSOFAFeedURL)!) } } else { LogManager.info("Previously cached SOFA json has expired", logger: sofaLog) @@ -1024,8 +1025,8 @@ struct NetworkFileManager { if let url = URL(string: OptionalFeatureVariables.customSOFAFeedURL) { let sofaData = SOFA().URLSync(url: url) - if (sofaData.responseCode != nil) { - if sofaData.responseCode == 304 && sofaJSONExists { + if let responseCode = sofaData.responseCode { + if responseCode == 304 && sofaJSONExists { LogManager.info("Utilizing previously cached SOFA json due to Etag not changing", logger: sofaLog) do { let sofaData = try Data(contentsOf: sofaPath) @@ -1034,7 +1035,8 @@ struct NetworkFileManager { } catch { LogManager.error("Failed to decode previously cached (Etag) local sofa JSON: \(error.localizedDescription)", logger: sofaLog) LogManager.error("Failed to decode sofa JSON: \(error)", logger: sofaLog) - return nil + // Attempt to redownload and reprocess the file + return redownloadAndReprocessSOFA(url: url) } } else { do { @@ -1052,7 +1054,8 @@ struct NetworkFileManager { } LogManager.error("Failed to decode sofa JSON: \(error.localizedDescription)", logger: sofaLog) LogManager.error("Failed to decode sofa JSON: \(error)", logger: sofaLog) - return nil + // Attempt to redownload and reprocess the file + return redownloadAndReprocessSOFA(url: url) } } } else { @@ -1125,6 +1128,35 @@ struct NetworkFileManager { } return false } + + func redownloadAndReprocessSOFA(url: URL) -> MacOSDataFeed? { + let sofaData = SOFA().URLSync(url: url) + if let responseCode = sofaData.responseCode, responseCode == 200 { + let fileManager = FileManager.default + let appSupportDirectory = fileManager.urls(for: .applicationSupportDirectory, in: .userDomainMask).first! + let appDirectory = appSupportDirectory.appendingPathComponent(Globals.bundleID) + let sofaFile = "sofa-macos_data_feed.json" + let sofaPath = appDirectory.appendingPathComponent(sofaFile) + + do { + if fileManager.fileExists(atPath: appDirectory.path) { + try sofaData.data!.write(to: sofaPath) + } + let assetInfo = try MacOSDataFeed(data: sofaData.data!) + return assetInfo + } catch { + LogManager.error("Failed to decode sofa JSON after redownload: \(error.localizedDescription)", logger: sofaLog) + LogManager.error("Failed to decode sofa JSON after redownload: \(error)", logger: sofaLog) + } + } else { + if sofaData.responseCode == nil { + LogManager.error("Failed to fetch sofa JSON: Device likely has no network connectivity.", logger: sofaLog) + } else { + LogManager.error("Failed to fetch sofa JSON: \(sofaData.error!.localizedDescription)", logger: sofaLog) + } + } + return nil + } } struct SubProcessUtilities { From fee26760c818418fa4343640125e24cee4f584ad Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 15 Jul 2024 13:43:02 -0500 Subject: [PATCH 128/141] preparing adds - move to globs and capture username of process now also true support for macOS 15 --- Nudge/Utilities/UILogic.swift | 72 +++++++++++++++++++++++------------ 1 file changed, 48 insertions(+), 24 deletions(-) diff --git a/Nudge/Utilities/UILogic.swift b/Nudge/Utilities/UILogic.swift index b9531658..312655a4 100644 --- a/Nudge/Utilities/UILogic.swift +++ b/Nudge/Utilities/UILogic.swift @@ -12,9 +12,10 @@ import IOKit.pwr_mgt // Asertions import SwiftUI struct ProcessInfoStruct { - var pid: Int32 - var command: String - var arguments: [String] + let pid: Int32 + let command: String + let arguments: [String] + let username: String } func initialLaunchLogic() { @@ -186,22 +187,22 @@ private func logDeferralStates() { func getAllProcesses() -> [ProcessInfoStruct] { var processes = [ProcessInfoStruct]() - + // Get the number of processes var mib: [Int32] = [CTL_KERN, KERN_PROC, KERN_PROC_ALL] var size = 0 sysctl(&mib, u_int(mib.count), nil, &size, nil, 0) - + let processCount = size / MemoryLayout.size var processList = [kinfo_proc](repeating: kinfo_proc(), count: processCount) - + // Get the list of processes sysctl(&mib, u_int(mib.count), &processList, &size, nil, 0) - + // Extract process info for process in processList { let pid = process.kp_proc.p_pid - + // Get full command path var pathBuffer = [CChar](repeating: 0, count: Int(PATH_MAX)) let result = proc_pidpath(pid, &pathBuffer, UInt32(PATH_MAX)) @@ -210,11 +211,13 @@ func getAllProcesses() -> [ProcessInfoStruct] { String(cString: $0) } } - + let arguments = getArgumentsForPID(pid: pid) - processes.append(ProcessInfoStruct(pid: pid, command: command, arguments: arguments)) + let username = getUsernameForPID(pid: pid) + + processes.append(ProcessInfoStruct(pid: pid, command: command, arguments: arguments, username: username)) } - + return processes } @@ -244,15 +247,36 @@ func getArgumentsForPID(pid: Int32) -> [String] { return args } -func isAnyProcessRunning(commandsWithArgs: [(processRelativeName: String, arguments: [String]?)]) -> Bool { +func getUsernameForPID(pid: Int32) -> String { + var uid: uid_t = 0 + var mib: [Int32] = [CTL_KERN, KERN_PROC, KERN_PROC_PID, pid] + var kp = kinfo_proc() + var size = MemoryLayout.size + + sysctl(&mib, u_int(mib.count), &kp, &size, nil, 0) + + uid = kp.kp_eproc.e_ucred.cr_uid + var pwd = passwd() + var pwdPtr: UnsafeMutablePointer? = nil + getpwuid_r(uid, &pwd, nil, 0, &pwdPtr) + + if let pwdPtr = pwdPtr { + return String(cString: pwdPtr.pointee.pw_name) + } else { + return "unknown" + } +} + + +func isAnyProcessRunning(commandsWithArgs: [(commandPattern: String, arguments: [String]?, username: String?)]) -> Bool { let processes = getAllProcesses() - for (processRelativeName, arguments) in commandsWithArgs { + for (commandPattern, arguments, username) in commandsWithArgs { let matchingProcesses = processes.filter { process in - // Check if the command pattern is a substring of the full command path - process.command.lowercased().contains(processRelativeName.lowercased()) && + fnmatch(commandPattern, process.command, FNM_CASEFOLD) == 0 && (arguments == nil || arguments!.allSatisfy { arg in process.arguments.contains(where: { $0.contains(arg) }) - }) + }) && + (username == nil || process.username == username) } if !matchingProcesses.isEmpty { return true @@ -262,14 +286,14 @@ func isAnyProcessRunning(commandsWithArgs: [(processRelativeName: String, argume } func isDownloadingOrPreparingSoftwareUpdate() -> Bool { - let commandsWithArgs: [(processRelativeName: String, arguments: [String]?)] = [ - ("softwareupdated", ["/System/Library/PrivateFrameworks/MobileSoftwareUpdate.framework/Support/softwareupdated"]), // When downloading a minor update, this process is running. - ("installcoordinationd", ["/System/Library/PrivateFrameworks/InstallCoordination.framework/Support/installcoordinationd"]), // When preparing a minor update, this process is running. Unfortunately, after preparing the update, this process appears to stay running. - ("softwareupdate", ["/usr/bin/softwareupdate", "--fetch-full-installer"]), // When downloading a major upgrade via SoftwareUpdate prefpane, it triggers a --fetch-full-installer run. Nudge also performs this method. - ("softwareupdate", ["/usr/sbin/softwareupdate", "--fetch-full-installer"]), // When downloading a major upgrade via softwareupdate cli, it triggers a --fetch-full-installer run. Nudge also performs this method. - ("osinstallersetupd", ["/Applications/*Install macOS *.app/Contents/Frameworks/OSInstallerSetup.framework/Resources/osinstallersetupd"]), // When installing a major upgrade, this process is running. - ("installerauthagent", ["/System/Library/PrivateFrameworks/IASUtilities.framework/Versions/A/Resources/installerauthagent", "/System/Library/PrivateFrameworks/IASUtilities.framework/Versions/A/Resources/installerauthagent"]), // Possibly on macOS 15, this is running when preparing an update. - // /System/Library/PrivateFrameworks/PackageKit.framework/Resources/installd||system_installd - system_installd may be interesting, but I think installd is being used for any package + let commandsWithArgs: [(commandPattern: String, arguments: [String]?, username: String?)] = [ + ("*softwareupdated", ["/System/Library/PrivateFrameworks/MobileSoftwareUpdate.framework/Support/softwareupdated"], nil), // When downloading a minor update, this process is running. + ("*installcoordinationd", ["/System/Library/PrivateFrameworks/InstallCoordination.framework/Support/installcoordinationd"], nil), // When preparing a minor update, this process is running. Unfortunately, after preparing the update, this process appears to stay running. + ("*softwareupdate", ["/usr/bin/softwareupdate", "--fetch-full-installer"], nil), // When downloading a major upgrade via SoftwareUpdate prefpane, it triggers a --fetch-full-installer run. Nudge also performs this method. + ("*softwareupdate", ["/usr/sbin/softwareupdate", "--fetch-full-installer"], nil), // When downloading a major upgrade via softwareupdate cli, it triggers a --fetch-full-installer run. Nudge also performs this method. + ("*osinstallersetupd", ["/Applications/*Install macOS *.app/Contents/Frameworks/OSInstallerSetup.framework/Resources/osinstallersetupd"], nil), // When installing a major upgrade, this process is running. + ("*com.apple.MobileSoftwareUpdate.UpdateBrainService", [], nil), // On macOS 15, this process is running when preparing an update. + ("*com.apple.StreamingUnzipService.privileged", nil, "_nsurlsessiond"), // When preparing an update on macOS 15, this process is running to extract the OS update for preparation. ] return isAnyProcessRunning(commandsWithArgs: commandsWithArgs) } From 0fd3b7ea16be50ffe219b0c48f4386fafe1cd6b9 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 15 Jul 2024 17:21:44 -0500 Subject: [PATCH 129/141] acceptableUpdatePreparingUsage is now opt-out Logic is tested on macOS 12.0.1 to 12.7.5 update and macOS 15 Beta 3 to macOS 15 Public Beta. All logic works correctly. --- CHANGELOG.md | 5 +++-- .../com.github.macadmins.Nudge.tester.json | 1 - .../Preferences/DefaultPreferencesNudge.swift | 2 +- Nudge/Utilities/UILogic.swift | 22 ++++++++++++++----- Schema/jamf/com.github.macadmins.Nudge.json | 2 +- 5 files changed, 21 insertions(+), 11 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index f1754ca9..9a973ef2 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -8,6 +8,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0 Requires macOS 12.0 and higher. ### Breaking Changes +- **macOS 11 is now unsupported** + - Please use Nudge 1.x releases for macOS 11 - Due to implementing markdown support, many of the elements within Nudge may no longer be in **bold** if you customize them. - To work around this please add `**` elements to these customizations - For example: The `mainContentNote` value of `Important Notes` would become `**Important Notes**` @@ -88,8 +90,7 @@ Requires macOS 12.0 and higher. - When the device is running macOS 12.3 or higher, Nudge uses the delta logic for macOS Upgrades - [Issue 417](https://github.com/macadmins/nudge/issues/417) - Nudge can now bypass activations and re-activations when a macOS update is `Downloading`, `Preparing` or `Staged` for installation. - - To enable this, please configure the `acceptableUpdatePreparingUsage` key under `optionalFeatures` to true - - Be aware that the current logic used for this **cannot differentiate** when an update has completed preparing and is in the `Staged` phase, waiting for a user to reboot. This is due to an Apple process staying in memory. This will result in a reduction in Nudge re-activations + - To disable this, please configure the `acceptableUpdatePreparingUsage` key under `optionalFeatures` to false - Issue [555](https://github.com/macadmins/nudge/issues/555) and [571](https://github.com/macadmins/nudge/issues/571) - Nudge can now attempt to honor DoNotDisturb/Focus times - To enable this, please configure the `honorFocusModes` key in `optionalFeatures` to true diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index cf6a71cb..f4b8a199 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -7,7 +7,6 @@ "Safari" ], "acceptableAssertionUsage": false, - "acceptableUpdatePreparingUsage": true, "acceptableCameraUsage": true, "acceptableScreenSharingUsage": true, "attemptToBlockApplicationLaunches": true, diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index 049196f8..abb8721a 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -86,7 +86,7 @@ struct OptionalFeatureVariables { static var acceptableUpdatePreparingUsage: Bool { optionalFeaturesProfile?["acceptableUpdatePreparingUsage"] as? Bool ?? optionalFeaturesJSON?.acceptableUpdatePreparingUsage ?? - false + true } static var acceptableScreenSharingUsage: Bool { diff --git a/Nudge/Utilities/UILogic.swift b/Nudge/Utilities/UILogic.swift index 312655a4..c06255d6 100644 --- a/Nudge/Utilities/UILogic.swift +++ b/Nudge/Utilities/UILogic.swift @@ -267,7 +267,6 @@ func getUsernameForPID(pid: Int32) -> String { } } - func isAnyProcessRunning(commandsWithArgs: [(commandPattern: String, arguments: [String]?, username: String?)]) -> Bool { let processes = getAllProcesses() for (commandPattern, arguments, username) in commandsWithArgs { @@ -287,15 +286,26 @@ func isAnyProcessRunning(commandsWithArgs: [(commandPattern: String, arguments: func isDownloadingOrPreparingSoftwareUpdate() -> Bool { let commandsWithArgs: [(commandPattern: String, arguments: [String]?, username: String?)] = [ + ("*com.apple.StreamingUnzipService", nil, "_nsurlsessiond"), // When downloading a minor update on macOS 12, this process is running to extract the OS update for preparation. + ("*com.apple.StreamingUnzipService.privileged", nil, "_nsurlsessiond"), // When downloading a minor update on macOS 15, this process is running to extract the OS update for preparation. ("*softwareupdated", ["/System/Library/PrivateFrameworks/MobileSoftwareUpdate.framework/Support/softwareupdated"], nil), // When downloading a minor update, this process is running. - ("*installcoordinationd", ["/System/Library/PrivateFrameworks/InstallCoordination.framework/Support/installcoordinationd"], nil), // When preparing a minor update, this process is running. Unfortunately, after preparing the update, this process appears to stay running. ("*softwareupdate", ["/usr/bin/softwareupdate", "--fetch-full-installer"], nil), // When downloading a major upgrade via SoftwareUpdate prefpane, it triggers a --fetch-full-installer run. Nudge also performs this method. ("*softwareupdate", ["/usr/sbin/softwareupdate", "--fetch-full-installer"], nil), // When downloading a major upgrade via softwareupdate cli, it triggers a --fetch-full-installer run. Nudge also performs this method. - ("*osinstallersetupd", ["/Applications/*Install macOS *.app/Contents/Frameworks/OSInstallerSetup.framework/Resources/osinstallersetupd"], nil), // When installing a major upgrade, this process is running. - ("*com.apple.MobileSoftwareUpdate.UpdateBrainService", [], nil), // On macOS 15, this process is running when preparing an update. - ("*com.apple.StreamingUnzipService.privileged", nil, "_nsurlsessiond"), // When preparing an update on macOS 15, this process is running to extract the OS update for preparation. + ("*com.apple.MobileSoftwareUpdate.UpdateBrainService", [], nil), // When preparing a minor update on macOS 12-15, this process is running when preparing an update. This is the same process for Intel and Apple Silicon devices. ] - return isAnyProcessRunning(commandsWithArgs: commandsWithArgs) + return isAnyProcessRunning(commandsWithArgs: commandsWithArgs) && !isSnapshotPresent(snapshotName: "com.apple.os.update-MSUPrepareUpdate") +} + +func isSnapshotPresent(snapshotName: String) -> Bool { + let subProcessUtilities = SubProcessUtilities() + let result = subProcessUtilities.runProcess(launchPath: "/usr/sbin/diskutil", arguments: ["apfs", "listSnapshots", "/"]) + + if result.exitCode != 0 { + print("Error: \(result.error)") + return false + } + + return result.output.contains(snapshotName) } func needToActivateNudge() -> Bool { diff --git a/Schema/jamf/com.github.macadmins.Nudge.json b/Schema/jamf/com.github.macadmins.Nudge.json index 9f718656..c92da84c 100644 --- a/Schema/jamf/com.github.macadmins.Nudge.json +++ b/Schema/jamf/com.github.macadmins.Nudge.json @@ -95,7 +95,7 @@ }, { "title": "Configured", - "default": false, + "default": true, "type": "boolean" } ] From 6ebb5922b67cb8f15fd530c5eb9d75465dda57e4 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 15 Jul 2024 18:49:25 -0500 Subject: [PATCH 130/141] moved to gzipped data feed by default --- Nudge/Preferences/DefaultPreferencesNudge.swift | 2 +- Schema/jamf/com.github.macadmins.Nudge.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index abb8721a..dd11fae4 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -140,7 +140,7 @@ struct OptionalFeatureVariables { static var customSOFAFeedURL: String { optionalFeaturesProfile?["customSOFAFeedURL"] as? String ?? optionalFeaturesJSON?.customSOFAFeedURL ?? - "https://sofa.macadmins.io/v1/macos_data_feed.json" + "https://sofafeed.macadmins.io/macos_data_feed.json" } static var disableSoftwareUpdateWorkflow: Bool { diff --git a/Schema/jamf/com.github.macadmins.Nudge.json b/Schema/jamf/com.github.macadmins.Nudge.json index c92da84c..e3a14f74 100644 --- a/Schema/jamf/com.github.macadmins.Nudge.json +++ b/Schema/jamf/com.github.macadmins.Nudge.json @@ -232,7 +232,7 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "https://sofa.macadmins.io/v1/macos_data_feed.json" + "placeholder": "https://sofafeed.macadmins.io/macos_data_feed.json" } } } From a6f4acc159625a61551ea7ddc13250319646418b Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 15 Jul 2024 19:47:39 -0500 Subject: [PATCH 131/141] add safety if we cannot save sofa json --- Nudge/Utilities/Utils.swift | 28 +++++++++++++++++++--------- 1 file changed, 19 insertions(+), 9 deletions(-) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 5e04fc38..248ed800 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -1040,11 +1040,16 @@ struct NetworkFileManager { } } else { do { - if fileManager.fileExists(atPath: appDirectory.path) { - try sofaData.data!.write(to: sofaPath) + if let data = sofaData.data { + if fileManager.fileExists(atPath: appDirectory.path) { + try data.write(to: sofaPath) + } + let assetInfo = try MacOSDataFeed(data: data) + return assetInfo + } else { + LogManager.error("Failed to fetch sofa JSON: No data received.", logger: sofaLog) + return redownloadAndReprocessSOFA(url: url) } - let assetInfo = try MacOSDataFeed(data: sofaData.data!) - return assetInfo } catch { do { try fileManager.removeItem(atPath: sofaPath.path) @@ -1139,11 +1144,16 @@ struct NetworkFileManager { let sofaPath = appDirectory.appendingPathComponent(sofaFile) do { - if fileManager.fileExists(atPath: appDirectory.path) { - try sofaData.data!.write(to: sofaPath) + if let data = sofaData.data { + if fileManager.fileExists(atPath: appDirectory.path) { + try data.write(to: sofaPath) + } + let assetInfo = try MacOSDataFeed(data: data) + return assetInfo + } else { + LogManager.error("Failed to fetch sofa JSON: No data received.", logger: sofaLog) + return redownloadAndReprocessSOFA(url: url) } - let assetInfo = try MacOSDataFeed(data: sofaData.data!) - return assetInfo } catch { LogManager.error("Failed to decode sofa JSON after redownload: \(error.localizedDescription)", logger: sofaLog) LogManager.error("Failed to decode sofa JSON after redownload: \(error)", logger: sofaLog) @@ -1152,7 +1162,7 @@ struct NetworkFileManager { if sofaData.responseCode == nil { LogManager.error("Failed to fetch sofa JSON: Device likely has no network connectivity.", logger: sofaLog) } else { - LogManager.error("Failed to fetch sofa JSON: \(sofaData.error!.localizedDescription)", logger: sofaLog) + LogManager.error("Failed to fetch sofa JSON: \(sofaData.error?.localizedDescription ?? "Unknown error")", logger: sofaLog) } } return nil From f938ea1d45d7977cdf678bb653ba7608fc5d7f60 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 15 Jul 2024 19:48:23 -0500 Subject: [PATCH 132/141] move logic for etag later in the processing path --- Nudge/3rd Party Assets/sofa.swift | 7 ++++--- Nudge/Utilities/Utils.swift | 1 + 2 files changed, 5 insertions(+), 3 deletions(-) diff --git a/Nudge/3rd Party Assets/sofa.swift b/Nudge/3rd Party Assets/sofa.swift index 5e1bf548..584fc0e5 100644 --- a/Nudge/3rd Party Assets/sofa.swift +++ b/Nudge/3rd Party Assets/sofa.swift @@ -222,7 +222,7 @@ extension MacOSDataFeed { } class SOFA: NSObject, URLSessionDelegate { - func URLSync(url: URL, maxRetries: Int = 3) -> (data: Data?, response: URLResponse?, error: Error?, responseCode: Int?) { + func URLSync(url: URL, maxRetries: Int = 3) -> (data: Data?, response: URLResponse?, error: Error?, responseCode: Int?, eTag: String?) { let semaphore = DispatchSemaphore(value: 0) let lastEtag = Globals.nudgeDefaults.string(forKey: "LastEtag") ?? "" var request = URLRequest(url: url) @@ -239,6 +239,7 @@ class SOFA: NSObject, URLSessionDelegate { var response: URLResponse? var responseError: Error? var responseCode: Int? + var eTag: String? var successfulQuery = false // Retry loop @@ -254,7 +255,7 @@ class SOFA: NSObject, URLSessionDelegate { responseCode = httpResponse.statusCode if responseCode == 200 { if let etag = httpResponse.allHeaderFields["Etag"] as? String { - Globals.nudgeDefaults.set(etag, forKey: "LastEtag") + eTag = etag } successfulQuery = true } else if responseCode == 304 { @@ -282,6 +283,6 @@ class SOFA: NSObject, URLSessionDelegate { } } - return (responseData, response, responseError, responseCode) + return (responseData, response, responseError, responseCode, eTag) } } diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 248ed800..3a74ba90 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -1045,6 +1045,7 @@ struct NetworkFileManager { try data.write(to: sofaPath) } let assetInfo = try MacOSDataFeed(data: data) + Globals.nudgeDefaults.set(sofaData.eTag, forKey: "LastEtag") return assetInfo } else { LogManager.error("Failed to fetch sofa JSON: No data received.", logger: sofaLog) From b8a73e2562502d5287e81be149f9b3a11df556b6 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 15 Jul 2024 19:48:38 -0500 Subject: [PATCH 133/141] add gzip decompression logic --- Nudge/3rd Party Assets/sofa.swift | 59 +++++++++++++++++++++++++++++-- 1 file changed, 56 insertions(+), 3 deletions(-) diff --git a/Nudge/3rd Party Assets/sofa.swift b/Nudge/3rd Party Assets/sofa.swift index 584fc0e5..7c0953bf 100644 --- a/Nudge/3rd Party Assets/sofa.swift +++ b/Nudge/3rd Party Assets/sofa.swift @@ -222,6 +222,39 @@ extension MacOSDataFeed { } class SOFA: NSObject, URLSessionDelegate { + func decompressGzip(data: Data) -> Data? { + let tempDir = FileManager.default.temporaryDirectory + let compressedFileURL = tempDir.appendingPathComponent(UUID().uuidString) + let decompressedFileURL = tempDir.appendingPathComponent(UUID().uuidString) + + do { + // Write compressed data to a temporary file + try data.write(to: compressedFileURL) + + // Use gzip to decompress the file + let result = SubProcessUtilities().runProcess(launchPath: "/usr/bin/gzip", arguments: ["--decompress", "--to-stdout", compressedFileURL.path]) + + // Clean up temporary files + try FileManager.default.removeItem(at: compressedFileURL) + + if result.exitCode != 0 { + LogManager.error("gzip failed with status \(result.exitCode): \(result.error)", logger: utilsLog) + return nil + } + + // Convert the output string back to Data + guard let decompressedData = result.output.data(using: .utf8) else { + LogManager.error("Failed to convert decompressed output to Data", logger: utilsLog) + return nil + } + + return decompressedData + } catch { + LogManager.error("Failed to decompress gzip data using command: \(error.localizedDescription)", logger: utilsLog) + return nil + } + } + func URLSync(url: URL, maxRetries: Int = 3) -> (data: Data?, response: URLResponse?, error: Error?, responseCode: Int?, eTag: String?) { let semaphore = DispatchSemaphore(value: 0) let lastEtag = Globals.nudgeDefaults.string(forKey: "LastEtag") ?? "" @@ -245,9 +278,10 @@ class SOFA: NSObject, URLSessionDelegate { // Retry loop while attempts < maxRetries { attempts += 1 - let task = session.dataTask(with: request) { data, resp, error in + let task = session.dataTask(with: request) { [weak self] data, resp, error in + guard let self = self else { return } // To handle self being nil guard let httpResponse = resp as? HTTPURLResponse else { - print("Error receiving response: \(error?.localizedDescription ?? "No error information")") + LogManager.error("Error receiving response: \(error?.localizedDescription ?? "No error information")", logger: utilsLog) semaphore.signal() return } @@ -258,11 +292,30 @@ class SOFA: NSObject, URLSessionDelegate { eTag = etag } successfulQuery = true + + if let encoding = httpResponse.allHeaderFields["Content-Encoding"] as? String { + LogManager.debug("Content-Encoding: \(encoding)", logger: utilsLog) + + if encoding == "gzip", let compressedData = data { + responseData = self.decompressGzip(data: compressedData) + + if responseData == nil { + LogManager.error("Failed to decompress gzip data", logger: utilsLog) + responseData = data // Fall back to using the original data + } else { + LogManager.debug("Successfully decompressed gzip data", logger: utilsLog) + } + } else { + responseData = data + } + } else { + responseData = data + } + } else if responseCode == 304 { successfulQuery = true } - responseData = data response = resp responseError = error semaphore.signal() From ac94ef8ef292f962a14829f6d603edb4cacc7739 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Mon, 15 Jul 2024 21:24:17 -0500 Subject: [PATCH 134/141] add v1 to api call --- Nudge/Preferences/DefaultPreferencesNudge.swift | 2 +- Schema/jamf/com.github.macadmins.Nudge.json | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/Nudge/Preferences/DefaultPreferencesNudge.swift b/Nudge/Preferences/DefaultPreferencesNudge.swift index dd11fae4..f013ec68 100644 --- a/Nudge/Preferences/DefaultPreferencesNudge.swift +++ b/Nudge/Preferences/DefaultPreferencesNudge.swift @@ -140,7 +140,7 @@ struct OptionalFeatureVariables { static var customSOFAFeedURL: String { optionalFeaturesProfile?["customSOFAFeedURL"] as? String ?? optionalFeaturesJSON?.customSOFAFeedURL ?? - "https://sofafeed.macadmins.io/macos_data_feed.json" + "https://sofafeed.macadmins.io/v1/macos_data_feed.json" } static var disableSoftwareUpdateWorkflow: Bool { diff --git a/Schema/jamf/com.github.macadmins.Nudge.json b/Schema/jamf/com.github.macadmins.Nudge.json index e3a14f74..358d5eb6 100644 --- a/Schema/jamf/com.github.macadmins.Nudge.json +++ b/Schema/jamf/com.github.macadmins.Nudge.json @@ -232,7 +232,7 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "https://sofafeed.macadmins.io/macos_data_feed.json" + "placeholder": "https://sofafeed.macadmins.io/v1/macos_data_feed.json" } } } From d1dcb150d89e24655c6b9853f0aa9f5efeae82b1 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 16 Jul 2024 09:28:35 -0500 Subject: [PATCH 135/141] move 0byte check to reduce erroneous log errors --- Nudge/Utilities/Utils.swift | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 3a74ba90..93d2d0b4 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -979,12 +979,11 @@ struct NetworkFileManager { let appDirectory = appSupportDirectory.appendingPathComponent(Globals.bundleID) let sofaFile = "sofa-macos_data_feed.json" let sofaPath = appDirectory.appendingPathComponent(sofaFile) - let sofaJSON0Bytes = isFileEmpty(atPath: sofaPath.path) var sofaJSONExists = fileManager.fileExists(atPath: sofaPath.path) // Force delete if bad if sofaJSONExists { - if sofaJSON0Bytes { + if isFileEmpty(atPath: sofaPath.path) { do { try fileManager.removeItem(atPath: sofaPath.path) sofaJSONExists = false From a6f01773fda60bba8b4e5e1b591acbfd9f559dfe Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 16 Jul 2024 09:28:57 -0500 Subject: [PATCH 136/141] customfeedurl to test json to use both sofa feeds --- Example Assets/com.github.macadmins.Nudge.tester.json | 2 ++ 1 file changed, 2 insertions(+) diff --git a/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index f4b8a199..e075c85a 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -10,6 +10,8 @@ "acceptableCameraUsage": true, "acceptableScreenSharingUsage": true, "attemptToBlockApplicationLaunches": true, + "customSOFAFeedURL": "https://sofafeed.macadmins.io/v1/macos_data_feed.json", + "_customSOFAFeedURL": "https://sofa.macadmins.io/v1/macos_data_feed.json", "blockedApplicationBundleIDs": [ "com.apple.Safari" ], From 325e9f71d1a65bd7ffa7f8209d8aa2e11a9e6d52 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 16 Jul 2024 09:29:50 -0500 Subject: [PATCH 137/141] gzip deflate is not needed and add timeout logic sometimes the sofa feed hangs. It could be my device, but this will ensure after 10 seconds, we retry. --- Nudge/3rd Party Assets/sofa.swift | 69 +++++++------------------------ 1 file changed, 16 insertions(+), 53 deletions(-) diff --git a/Nudge/3rd Party Assets/sofa.swift b/Nudge/3rd Party Assets/sofa.swift index 7c0953bf..6d8f3919 100644 --- a/Nudge/3rd Party Assets/sofa.swift +++ b/Nudge/3rd Party Assets/sofa.swift @@ -222,39 +222,6 @@ extension MacOSDataFeed { } class SOFA: NSObject, URLSessionDelegate { - func decompressGzip(data: Data) -> Data? { - let tempDir = FileManager.default.temporaryDirectory - let compressedFileURL = tempDir.appendingPathComponent(UUID().uuidString) - let decompressedFileURL = tempDir.appendingPathComponent(UUID().uuidString) - - do { - // Write compressed data to a temporary file - try data.write(to: compressedFileURL) - - // Use gzip to decompress the file - let result = SubProcessUtilities().runProcess(launchPath: "/usr/bin/gzip", arguments: ["--decompress", "--to-stdout", compressedFileURL.path]) - - // Clean up temporary files - try FileManager.default.removeItem(at: compressedFileURL) - - if result.exitCode != 0 { - LogManager.error("gzip failed with status \(result.exitCode): \(result.error)", logger: utilsLog) - return nil - } - - // Convert the output string back to Data - guard let decompressedData = result.output.data(using: .utf8) else { - LogManager.error("Failed to convert decompressed output to Data", logger: utilsLog) - return nil - } - - return decompressedData - } catch { - LogManager.error("Failed to decompress gzip data using command: \(error.localizedDescription)", logger: utilsLog) - return nil - } - } - func URLSync(url: URL, maxRetries: Int = 3) -> (data: Data?, response: URLResponse?, error: Error?, responseCode: Int?, eTag: String?) { let semaphore = DispatchSemaphore(value: 0) let lastEtag = Globals.nudgeDefaults.string(forKey: "LastEtag") ?? "" @@ -262,12 +229,13 @@ class SOFA: NSObject, URLSessionDelegate { let config = URLSessionConfiguration.default config.requestCachePolicy = .useProtocolCachePolicy let session = URLSession(configuration: config, delegate: self, delegateQueue: nil) + request.addValue("\(Globals.bundleID)/\(VersionManager.getNudgeVersion())", forHTTPHeaderField: "User-Agent") request.setValue(lastEtag, forHTTPHeaderField: "If-None-Match") // TODO: I'm saving the Etag and sending it, but due to forcing this into a syncronous call, it is always returning a 200 code. When using this in an asycronous method, it eventually returns the 304 response. I'm not sure how to fix this bug. request.addValue("gzip, deflate, br", forHTTPHeaderField: "Accept-Encoding") // Force compression for JSON - var attempts = 0 + var attempts = 0 var responseData: Data? var response: URLResponse? var responseError: Error? @@ -278,8 +246,7 @@ class SOFA: NSObject, URLSessionDelegate { // Retry loop while attempts < maxRetries { attempts += 1 - let task = session.dataTask(with: request) { [weak self] data, resp, error in - guard let self = self else { return } // To handle self being nil + let task = session.dataTask(with: request) { data, resp, error in guard let httpResponse = resp as? HTTPURLResponse else { LogManager.error("Error receiving response: \(error?.localizedDescription ?? "No error information")", logger: utilsLog) semaphore.signal() @@ -287,6 +254,9 @@ class SOFA: NSObject, URLSessionDelegate { } responseCode = httpResponse.statusCode + response = resp + responseError = error + if responseCode == 200 { if let etag = httpResponse.allHeaderFields["Etag"] as? String { eTag = etag @@ -295,33 +265,26 @@ class SOFA: NSObject, URLSessionDelegate { if let encoding = httpResponse.allHeaderFields["Content-Encoding"] as? String { LogManager.debug("Content-Encoding: \(encoding)", logger: utilsLog) - - if encoding == "gzip", let compressedData = data { - responseData = self.decompressGzip(data: compressedData) - - if responseData == nil { - LogManager.error("Failed to decompress gzip data", logger: utilsLog) - responseData = data // Fall back to using the original data - } else { - LogManager.debug("Successfully decompressed gzip data", logger: utilsLog) - } - } else { - responseData = data - } - } else { - responseData = data } + responseData = data + } else if responseCode == 304 { successfulQuery = true } - response = resp - responseError = error semaphore.signal() } + + let timeout = DispatchWorkItem { + task.cancel() + semaphore.signal() + } + + DispatchQueue.global().asyncAfter(deadline: .now() + 10, execute: timeout) task.resume() semaphore.wait() + timeout.cancel() if successfulQuery { break From b8255f7d7e44bce5e7c16bd75095adbb2a3a5c99 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 16 Jul 2024 09:36:31 -0500 Subject: [PATCH 138/141] bump release date 07-19 --- CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 9a973ef2..3d4ee793 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,7 +4,7 @@ All notable changes to this project will be documented in this file. The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). -## [2.0.0] - 2024-07-17 +## [2.0.0] - 2024-07-19 Requires macOS 12.0 and higher. ### Breaking Changes From bb0fa25acf25e5b24bf77e1af051d3e8a26ef74b Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 16 Jul 2024 17:32:55 -0500 Subject: [PATCH 139/141] remove bad log event when using latest* values --- Nudge/Utilities/Utils.swift | 15 +++++++++++++-- 1 file changed, 13 insertions(+), 2 deletions(-) diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 93d2d0b4..3b9c3d6b 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -1433,8 +1433,19 @@ struct VersionManager { } static func getMajorRequiredNudgeOSVersion() -> Int { - guard let majorVersion = Int(nudgePrimaryState.requiredMinimumOSVersion.split(separator: ".").first ?? "") else { - LogManager.error("Invalid format for requiredMinimumOSVersion - value is \(nudgePrimaryState.requiredMinimumOSVersion)", logger: utilsLog) + let requiredVersion = nudgePrimaryState.requiredMinimumOSVersion + + // Handle new string values directly + switch requiredVersion { + case "latest", "latest-minor", "latest-supported": + return 0 + default: + break + } + + // Existing logic for version numbers + guard let majorVersion = Int(requiredVersion.split(separator: ".").first ?? "") else { + LogManager.error("Invalid format for requiredMinimumOSVersion - value is \(requiredVersion)", logger: utilsLog) return 0 } logOSVersion(majorVersion, for: "Major required OS version") From bb0f9e0baf684e58cb445b4a3766ee39d48a09f0 Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Tue, 16 Jul 2024 17:33:14 -0500 Subject: [PATCH 140/141] get rid of all prints and log properly --- Nudge/UI/Main.swift | 10 +++++----- Nudge/Utilities/Preferences.swift | 12 ++++++------ Nudge/Utilities/UILogic.swift | 2 +- Nudge/Utilities/Utils.swift | 8 ++++---- 4 files changed, 16 insertions(+), 16 deletions(-) diff --git a/Nudge/UI/Main.swift b/Nudge/UI/Main.swift index 4c704f41..a64ec104 100644 --- a/Nudge/UI/Main.swift +++ b/Nudge/UI/Main.swift @@ -439,7 +439,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { } } } else { - print("applicationTerminatedNotificationImagePath does not exist on disk, skipping notification image.") + LogManager.error("applicationTerminatedNotificationImagePath does not exist on disk, skipping notification image.", logger: uiLog) } return content } @@ -621,7 +621,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { object: NSApplication.shared, queue: .main) { _ in if UserExperienceVariables.allowMovableWindow { return } - print("Window object frame moved - Notification Center") + LogManager.debug("Window object frame moved - Notification Center", logger: utilsLog) UIUtilities().centerNudge() } } @@ -732,17 +732,17 @@ class AppDelegate: NSObject, NSApplicationDelegate { class WindowDelegate: NSObject, NSWindowDelegate { func windowDidMove(_ notification: Notification) { if UserExperienceVariables.allowMovableWindow { return } - print("Window attempted to move - Window Delegate") + LogManager.debug("Window attempted to move - Window Delegate", logger: utilsLog) UIUtilities().centerNudge() } func windowDidChangeScreen(_ notification: Notification) { if UserExperienceVariables.allowMovableWindow { return } - print("Window moved screens - Window Delegate") + LogManager.debug("Window moved screens - Window Delegate", logger: utilsLog) UIUtilities().centerNudge() } func windowDidChangeScreenProfile(_ notification: Notification) { if UserExperienceVariables.allowMovableWindow { return } - print("Display has changed profiles - Window Delegate") + LogManager.debug("Display has changed profiles - Window Delegate", logger: utilsLog) UIUtilities().centerNudge() } } diff --git a/Nudge/Utilities/Preferences.swift b/Nudge/Utilities/Preferences.swift index 9bdef871..fdbd1288 100644 --- a/Nudge/Utilities/Preferences.swift +++ b/Nudge/Utilities/Preferences.swift @@ -34,7 +34,7 @@ func getDesiredLanguage(locale: Locale? = nil) -> String { } } } catch { - print("Failed to decode plist: \(error)") + LogManager.error("Failed to decode plist: \(error)", logger: prefsProfileLog) } } return UserInterfaceVariables.fallbackLanguage @@ -81,7 +81,7 @@ func getOptionalFeaturesProfile() -> [String: Any]? { return nil } } catch { - print("Failed to decode plist: \(error)") + LogManager.error("Failed to decode plist: \(error)", logger: prefsProfileLog) return nil } } @@ -150,7 +150,7 @@ func getOSVersionRequirementsProfile() -> OSVersionRequirement? { return nil } } catch { - print("Failed to decode plist: \(error)") + LogManager.error("Failed to decode plist: \(error)", logger: prefsProfileLog) return nil } } @@ -240,7 +240,7 @@ func getUserExperienceProfile() -> [String: Any]? { return nil } } catch { - print("Failed to decode plist: \(error)") + LogManager.error("Failed to decode plist: \(error)", logger: prefsProfileLog) return nil } } @@ -288,7 +288,7 @@ func getUserInterfaceProfile() -> [String: Any]? { return nil } } catch { - print("Failed to decode plist: \(error)") + LogManager.error("Failed to decode plist: \(error)", logger: prefsProfileLog) return nil } } @@ -332,7 +332,7 @@ func getUserInterfaceUpdateElementsProfile() -> [String: AnyObject]? { return nil } } catch { - print("Failed to decode plist: \(error)") + LogManager.error("Failed to decode plist: \(error)", logger: prefsProfileLog) return nil } } diff --git a/Nudge/Utilities/UILogic.swift b/Nudge/Utilities/UILogic.swift index c06255d6..d84dd827 100644 --- a/Nudge/Utilities/UILogic.swift +++ b/Nudge/Utilities/UILogic.swift @@ -301,7 +301,7 @@ func isSnapshotPresent(snapshotName: String) -> Bool { let result = subProcessUtilities.runProcess(launchPath: "/usr/sbin/diskutil", arguments: ["apfs", "listSnapshots", "/"]) if result.exitCode != 0 { - print("Error: \(result.error)") + LogManager.error("Error: \(result.error)", logger: utilsLog) return false } diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 3b9c3d6b..c9f0fac7 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -225,7 +225,7 @@ struct AppStateManager { if let value = CFPreferencesCopyAppValue(key, appID) as? Bool { return value } else { - print("Key '\(key)' not found in preferences") + LogManager.info("Key '\(key)' not found in preferences", logger: uiLog) return false } @@ -988,7 +988,7 @@ struct NetworkFileManager { try fileManager.removeItem(atPath: sofaPath.path) sofaJSONExists = false } catch { - print("Error deleting file: \(error.localizedDescription)") + LogManager.error("Error deleting file: \(error.localizedDescription)", logger: sofaLog) } } } @@ -1055,7 +1055,7 @@ struct NetworkFileManager { try fileManager.removeItem(atPath: sofaPath.path) sofaJSONExists = false } catch { - print("Error deleting file: \(error.localizedDescription)") + LogManager.error("Error deleting file: \(error.localizedDescription)", logger: sofaLog) } LogManager.error("Failed to decode sofa JSON: \(error.localizedDescription)", logger: sofaLog) LogManager.error("Failed to decode sofa JSON: \(error)", logger: sofaLog) @@ -1129,7 +1129,7 @@ struct NetworkFileManager { return fileSize.intValue == 0 } } catch { - print("Error getting file attributes: \(error.localizedDescription)") + LogManager.error("Error getting file attributes: \(error.localizedDescription)", logger: prefsJSONLog) } return false } From 2cba17605d28d4f1eef573f6f13bb6ec7aaa945b Mon Sep 17 00:00:00 2001 From: Erik Gomez Date: Wed, 17 Jul 2024 22:33:55 -0500 Subject: [PATCH 141/141] Update CHANGELOG.md --- CHANGELOG.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 3d4ee793..5f7b157d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,7 +4,7 @@ All notable changes to this project will be documented in this file. The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/), and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html). -## [2.0.0] - 2024-07-19 +## [2.0.0] - 2024-07-18 Requires macOS 12.0 and higher. ### Breaking Changes