diff --git a/.github/workflows/build_nudge_pr.yml b/.github/workflows/build_nudge_pr.yml index f297cd0b..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-13 + 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 6a43d70c..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-13 + 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 ed3b64cb..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-13 + 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 4a045ba7..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-13 + 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 91bbddc2..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-13 + runs-on: macos-14 steps: - name: Checkout nudge repo diff --git a/CHANGELOG.md b/CHANGELOG.md index 939d5fe7..5f7b157d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,6 +4,100 @@ 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-18 +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**` +- 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 +- 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 + - `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` + - `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 `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 + +### 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) +- 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 +- 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 `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 +- 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. +- 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 + - 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` +- "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: `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. + - 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) +- 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 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 + - 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 the 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. @@ -197,7 +291,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/Example Assets/com.github.macadmins.Nudge.tester.json b/Example Assets/com.github.macadmins.Nudge.tester.json index 400583c4..e075c85a 100644 --- a/Example Assets/com.github.macadmins.Nudge.tester.json +++ b/Example Assets/com.github.macadmins.Nudge.tester.json @@ -6,21 +6,25 @@ "Google Chrome", "Safari" ], - "acceptableAssertionUsage": true, + "acceptableAssertionUsage": false, "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.microsoft.VSCode", - "us.zoom.xos" + "com.apple.Safari" ], - "terminateApplicationsOnLaunch": false + "disableNudgeForStandardInstalls": true, + "honorFocusModes": true, + "terminateApplicationsOnLaunch": true }, "osVersionRequirements": [ { "aboutUpdateURL": "https://apple.com", - "requiredInstallationDate": "2025-01-01T00:00:00Z", - "requiredMinimumOSVersion": "14.99.99" + "requiredMinimumOSVersion": "15.99", + "requiredInstallationDate": "2025-01-01T00:00:00", + "unsupportedURL": "https://google.com" } ], "userExperience": { @@ -31,28 +35,37 @@ "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", + "applicationTerminatedNotificationImagePath": "/Library/Application Support/Nudge/logoLight.png", + "showRequiredDate": true, "simpleMode": false, - "screenShotDarkPath": "", - "screenShotLightPath": "", "updateElements": [ { "_language": "en", "actionButtonText": "actionButtonText", + "actionButtonTextUnsupported": "actionButtonTextUnsupported", "customDeferralButtonText": "customDeferralButtonText", "customDeferralDropdownText": "customDeferralDropdownText", "informationButtonText": "informationButtonText", "mainContentHeader": "mainContentHeader", + "mainContentHeaderUnsupported": "mainContentHeaderUnsupported", "mainContentNote": "mainContentNote", + "mainContentNoteUnsupported1": "mainContentNoteUnsupported", "mainContentSubHeader": "mainContentSubHeader", + "mainContentSubHeaderUnsupported1": "mainContentSubHeaderUnsupported", "mainContentText": "mainContentText", + "mainContentTextUnsupported1": "mainContentTextUnsupported", "mainHeader": "mainHeader", + "mainHeaderUnsupported": "mainHeaderUnsupported", "oneDayDeferralButtonText": "oneDayDeferralButtonText", "oneHourDeferralButtonText": "oneHourDeferralButtonText", "primaryQuitButtonText": "primaryQuitButtonText", "secondaryQuitButtonText": "secondaryQuitButtonText", "subHeader": "subHeader", + "subHeaderUnsupported1": "subHeaderUnsupported", "screenShotAltText": "Click to zoom into screenshot" } ] 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..01368c3b --- /dev/null +++ b/Example Assets/com.github.macadmins.Nudge.tester.plist @@ -0,0 +1,121 @@ + + + + + optionalFeatures + + acceptableAssertionApplicationNames + + zoom.us + Meeting Center + Google Chrome + Safari + + acceptableAssertionUsage + + acceptableCameraUsage + + acceptableScreenSharingUsage + + attemptToBlockApplicationLaunches + + 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 + actionButtonTextUnsupported + actionButtonTextUnsupported + 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/Localizable.xcstrings b/Localizable.xcstrings index 798d0ac7..4cc958c3 100644 --- a/Localizable.xcstrings +++ b/Localizable.xcstrings @@ -2,224 +2,755 @@ "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", + "extractionState" : "manual", + "localizations" : { + "da" : { + "stringUnit" : { + "state" : "translated", + "value" : "**En venligt ment påmindelse fra din lokale IT afdeling**" + } + }, + "de" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Eine freundliche Erinnerung deines IT-Teams**" + } + }, + "en" : { + "stringUnit" : { + "state" : "translated", + "value" : "**A friendly reminder from your local IT team**" + } + }, + "es" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Un recordatorio amistoso de tu equipo IT local**" + } + }, + "fr" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Un rappel amical de votre équipe informatique**" + } + }, + "hi" : { + "stringUnit" : { + "state" : "translated", + "value" : "**आपके स्थानीय IT से मित्रवत अनुस्मारक**" + } + }, + "it" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Un cortese promemoria dal tuo team IT locale**" + } + }, + "ja" : { + "stringUnit" : { + "state" : "translated", + "value" : "**ローカルITチームからのリマインドです**" + } + }, + "ko" : { + "stringUnit" : { + "state" : "translated", + "value" : "**로컬 IT 팀에서 알려드립니다**" + } + }, + "nb" : { + "stringUnit" : { + "state" : "translated", + "value" : "**En påminnelse fra IT**" + } + }, + "nl" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Een herinnering van uw IT-team**" + } + }, + "pl" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Przypomnienie IT**" + } + }, + "pt" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Um lembrete rápido da sua equipe de TI local**" + } + }, + "ru" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Напоминание от команды IT**" + } + }, + "sv" : { + "stringUnit" : { + "state" : "translated", + "value" : "**En påminnelse från IT**" + } + }, + "uk" : { + "stringUnit" : { + "state" : "translated", + "value" : "**Дружнє нагадування від вашої IT-команди**" + } + }, + "zh" : { + "stringUnit" : { + "state" : "translated", + "value" : "**IT团队的友好提醒**" + } + } + } + }, + "**Important Notes**" : { + "comment" : "mainContentNote / mainContentNoteUnsupported", + "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" : "**重要信息**" + } + } + } + }, + "**Your device is no longer capable of receving critical security updates**" : { + "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" : "**您的设备不再能够接收关键安全更新**" + } + } + } + }, + "**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 friendly reminder from your local IT team" : { + "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" : { "da" : { "stringUnit" : { "state" : "translated", - "value" : "En venligt ment påmindelse fra din lokale IT afdeling" + "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" : "Eine freundliche Erinnerung deines IT-Teams" + "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 friendly reminder from your local IT team" + "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" : "Un recordatorio amistoso de tu equipo IT local" + "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 rappel amical de votre équipe informatique" + "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" : "आपके स्थानीय IT से मित्रवत अनुस्मारक" + "value" : "यह सुनिश्चित करने के लिए कि आईटी आपके डिवाइस की सटीक सुरक्षा कर सके, एक पूरी तरह से अद्यतित डिवाइस की आवश्यकता है।\n\nयदि आपको प्रतिस्थापन उपकरण नहीं मिलता है, तो आप अपने दैनिक कार्यों के लिए आवश्यक कुछ वस्तुओं तक पहुंच खो देंगे।\n\nइसके बारे में अधिक जानकारी के लिए कृपया **अपना डिवाइस बदलें** बटन पर क्लिक करें।" } }, "it" : { "stringUnit" : { "state" : "translated", - "value" : "Un cortese promemoria dal tuo team IT locale" + "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チームからのリマインドです" + "value" : "IT 部門がデバイスを正確に保護するには、完全に最新のデバイスが必要です。\n\n交換用デバイスを入手しない場合、日常業務に必要な一部のアイテムにアクセスできなくなります。\n\n詳細については、[**デバイスを交換する**] ボタンをクリックしてください。" } }, "ko" : { "stringUnit" : { "state" : "translated", - "value" : "로컬 IT 팀에서 알려드립니다" + "value" : "IT가 귀하의 장치를 정확하게 보호하려면 완전히 최신 장치가 필요합니다.\n\n교체 장치를 구입하지 않으면 일상적인 작업에 필요한 일부 항목에 액세스할 수 없게 됩니다.\n\n이에 대한 자세한 내용을 보려면 **장치 교체** 버튼을 클릭하세요." } }, "nb" : { "stringUnit" : { "state" : "translated", - "value" : "En påminnelse fra IT" + "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 herinnering van uw IT-team" + "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" : "Przypomnienie IT" + "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 lembrete rápido da sua equipe de TI local" + "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" : "Напоминание от команды IT" + "value" : "Требуется полностью обновленное устройство, чтобы ИТ-специалисты могли точно защитить ваше устройство.\n\nЕсли вы не получите устройство на замену, вы потеряете доступ к некоторым элементам, необходимым для выполнения повседневных задач.\n\nДля получения дополнительной информации об этом нажмите кнопку **Заменить устройство**." } }, "sv" : { "stringUnit" : { "state" : "translated", - "value" : "En påminnelse från IT" + "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" : "Дружнє нагадування від вашої IT-команди" + "value" : "Потрібен повністю оновлений пристрій, щоб ІТ-спеціалісти могли точно захистити ваш пристрій.\n\nЯкщо ви не отримаєте пристрій на заміну, ви втратите доступ до деяких предметів, необхідних для виконання повсякденних завдань.\n\nЩоб дізнатися більше про це, натисніть кнопку **Замінити свій пристрій**." } }, "zh" : { "stringUnit" : { "state" : "translated", - "value" : "IT团队的友好提醒" + "value" : "需要完全更新的设备才能确保 IT 能够准确保护您的设备。\n\n如果您没有获得替换设备,您将无法访问日常任务所需的某些物品。\n\n有关详细信息,请单击\"**更换您的设备**\"按钮。" } } } }, - "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." : { + "comment" : "mainContentText", "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" : "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" : "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" : "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." + "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" : "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" : "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" : "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" : "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" : "translated", + "state" : "needs_review", "value" : "पुरी तरह अपडेटेड उपकरण आवश्यक है IT द्वारा आपके उपकरण को सुरक्षित रखने के लिए \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 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" : "È 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" : "translated", + "state" : "needs_review", "value" : "ITが正確にあなたのデバイスを保護するためには、デバイスを完全に最新にすることが必要です。\n\nデバイスを更新しない場合、日々のタスクに必要なアイテムへのアクセスができなくなる可能性があります。\n\n更新を開始するにはデバイスの更新ボタンをクリックし、指示されたステップに従ってください。" } }, "ko" : { "stringUnit" : { "state" : "translated", - "value" : "IT 팀이 귀하의 기기를 정밀하게 보호하기 위하여 최신 업데이트가 기기에 항시 탑재될 것을 요청드립니다.\n\n귀하께서 기기를 업데이트 하지 않는다면, 일상의 업무에 필요한 아이템들의 접근 권한이 상실될 수 있습니다.\n\n기기를 업데이트 하시려면, 기기 업데이트 버튼을 클릭하시고 제공되는 안내를 따르시면 됩니다." + "value" : "IT 팀이 귀하의 기기를 정밀하게 보호하기 위하여 최신 업데이트가 기기에 항시 탑재될 것을 요청드립니다.\n\n귀하께서 기기를 업데이트 하지 않는다면, 일상의 업무에 필요한 아이템들의 접근 권한이 상실될 수 있습니다.\n\n기기를 업데이트 하시려면, **기기 업데이트** 버튼을 클릭하시고 제공되는 안내를 따르시면 됩니다." } }, "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" : "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" : "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" : "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" : "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" : "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" : "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" : "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" : "translated", + "state" : "needs_review", "value" : "Обновления нужны, чтобы IT-команда могла защитить Mac и данные.\n\nЕсли вы не обновите устройство, есть риск потерять доступ к важным для работы сервисам.\n\nЧтобы начать обновление, просто нажмите кнопку установки." } }, "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" : "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" : "translated", - "value" : "Потрібен повністю оновлений пристрій щоб бути впевненим що IT зможе його надійно захистити.\n\nЯкщо ви не оновите пристрій, ви можете втратити доступ до деяких речей, необхідних для виконання повсякденних завдань.\n\nЩоб почати оновлення, просто клацніть кнопку Оновити пристрій та виконайте запропоновані кроки." + "value" : "Потрібен повністю оновлений пристрій щоб бути впевненим що IT зможе його надійно захистити.\n\nЯкщо ви не оновите пристрій, ви можете втратити доступ до деяких речей, необхідних для виконання повсякденних завдань.\n\nЩоб почати оновлення, просто клацніть кнопку **Оновити пристрій** та виконайте запропоновані кроки." } }, "zh" : { "stringUnit" : { "state" : "translated", - "value" : "更新至最新的软件以确保 IT 可以精准的保护设备安全。\n\n如未及时完成更新,您将会无法访问日常工作所需的网页或程序。\n \n立刻更新,请点击【更新设备】并根据提供的步骤完成更新。" + "value" : "更新至最新的软件以确保 IT 可以精准的保护设备安全。\n\n如未及时完成更新,您将会无法访问日常工作所需的网页或程序。\n \n立刻更新,请点击\"**更新设备**\"并根据提供的步骤完成更新。" } } } }, "Additional Device Information" : { - "comment" : "// Additional Device Information", + "comment" : "Additional Device Information", "extractionState" : "manual", "localizations" : { "da" : { @@ -327,7 +858,7 @@ } }, "Application terminated" : { - "comment" : "User Notification", + "comment" : "Application terminated", "extractionState" : "manual", "localizations" : { "da" : { @@ -435,7 +966,7 @@ } }, "Architecture:" : { - "comment" : "Architecture", + "comment" : "Architecture:", "extractionState" : "manual", "localizations" : { "da" : { @@ -543,6 +1074,7 @@ } }, "Click for additional device information" : { + "comment" : "Click for additional device information", "extractionState" : "manual", "localizations" : { "da" : { @@ -650,6 +1182,7 @@ } }, "Click for more information about the security update" : { + "comment" : "Click for more information about the security update", "extractionState" : "manual", "localizations" : { "da" : { @@ -757,7 +1290,7 @@ } }, "Click to close" : { - "comment" : "Help buttons", + "comment" : "Click to close", "extractionState" : "manual", "localizations" : { "da" : { @@ -865,6 +1398,7 @@ } }, "Click to zoom into screenshot" : { + "comment" : "screenShotAltText", "extractionState" : "manual", "localizations" : { "da" : { @@ -972,7 +1506,7 @@ } }, "Current OS Version:" : { - "comment" : "Current OS Version", + "comment" : "Current OS Version:", "extractionState" : "manual", "localizations" : { "da" : { @@ -1080,6 +1614,7 @@ } }, "Custom" : { + "comment" : "customDeferralButtonText", "extractionState" : "manual", "localizations" : { "da" : { @@ -1187,7 +1722,7 @@ } }, "Days Remaining To Update:" : { - "comment" : "Days Remaining To Update", + "comment" : "Days Remaining To Update:", "extractionState" : "manual", "localizations" : { "da" : { @@ -1295,6 +1830,7 @@ } }, "Defer" : { + "comment" : "customDeferralDropdownText", "extractionState" : "manual", "localizations" : { "da" : { @@ -1402,7 +1938,7 @@ } }, "Deferred Count:" : { - "comment" : "Deferred Count", + "comment" : "Deferred Count:", "extractionState" : "manual", "localizations" : { "da" : { @@ -1510,7 +2046,7 @@ } }, "Hours Remaining To Update:" : { - "comment" : "Hours Remaining To Update", + "comment" : "Hours Remaining To Update:", "extractionState" : "manual", "localizations" : { "da" : { @@ -1618,6 +2154,7 @@ } }, "I understand" : { + "comment" : "secondaryQuitButtonText", "extractionState" : "manual", "localizations" : { "da" : { @@ -1724,759 +2261,980 @@ } } }, - "Important Notes" : { + "Language:" : { + "comment" : "Language:", + "extractionState" : "manual", + "localizations" : { + "da" : { + "stringUnit" : { + "state" : "translated", + "value" : "Sprog:" + } + }, + "de" : { + "stringUnit" : { + "state" : "translated", + "value" : "Sprache:" + } + }, + "en" : { + "stringUnit" : { + "state" : "translated", + "value" : "Language:" + } + }, + "es" : { + "stringUnit" : { + "state" : "translated", + "value" : "Idioma:" + } + }, + "fr" : { + "stringUnit" : { + "state" : "translated", + "value" : "Langue :" + } + }, + "hi" : { + "stringUnit" : { + "state" : "translated", + "value" : "भाषा:" + } + }, + "it" : { + "stringUnit" : { + "state" : "translated", + "value" : "Lingua:" + } + }, + "ja" : { + "stringUnit" : { + "state" : "translated", + "value" : "言語:" + } + }, + "ko" : { + "stringUnit" : { + "state" : "translated", + "value" : "언어:" + } + }, + "nb" : { + "stringUnit" : { + "state" : "translated", + "value" : "Språk:" + } + }, + "nl" : { + "stringUnit" : { + "state" : "translated", + "value" : "Taal:" + } + }, + "pl" : { + "stringUnit" : { + "state" : "translated", + "value" : "Język:" + } + }, + "pt" : { + "stringUnit" : { + "state" : "translated", + "value" : "Idioma:" + } + }, + "ru" : { + "stringUnit" : { + "state" : "translated", + "value" : "Язык:" + } + }, + "sv" : { + "stringUnit" : { + "state" : "translated", + "value" : "Språk:" + } + }, + "uk" : { + "stringUnit" : { + "state" : "translated", + "value" : "Мова:" + } + }, + "zh" : { + "stringUnit" : { + "state" : "translated", + "value" : "语言:" + } + } + } + }, + "Later" : { + "comment" : "primaryQuitButtonText", + "extractionState" : "manual", + "localizations" : { + "da" : { + "stringUnit" : { + "state" : "translated", + "value" : "Senere" + } + }, + "de" : { + "stringUnit" : { + "state" : "translated", + "value" : "Später" + } + }, + "en" : { + "stringUnit" : { + "state" : "translated", + "value" : "Later" + } + }, + "es" : { + "stringUnit" : { + "state" : "translated", + "value" : "Más Tarde" + } + }, + "fr" : { + "stringUnit" : { + "state" : "translated", + "value" : "Plus tard" + } + }, + "hi" : { + "stringUnit" : { + "state" : "translated", + "value" : "बाद मे" + } + }, + "it" : { + "stringUnit" : { + "state" : "translated", + "value" : "Dopo" + } + }, + "ja" : { + "stringUnit" : { + "state" : "translated", + "value" : "後で" + } + }, + "ko" : { + "stringUnit" : { + "state" : "translated", + "value" : "나중에" + } + }, + "nb" : { + "stringUnit" : { + "state" : "translated", + "value" : "Senere" + } + }, + "nl" : { + "stringUnit" : { + "state" : "translated", + "value" : "Later" + } + }, + "pl" : { + "stringUnit" : { + "state" : "translated", + "value" : "Później" + } + }, + "pt" : { + "stringUnit" : { + "state" : "translated", + "value" : "Depois" + } + }, + "ru" : { + "stringUnit" : { + "state" : "translated", + "value" : "Потом" + } + }, + "sv" : { + "stringUnit" : { + "state" : "translated", + "value" : "Senare" + } + }, + "uk" : { + "stringUnit" : { + "state" : "translated", + "value" : "Пізніше" + } + }, + "zh" : { + "stringUnit" : { + "state" : "translated", + "value" : "我同意" + } + } + } + }, + "More Info" : { + "comment" : "informationButtonText", "extractionState" : "manual", "localizations" : { "da" : { "stringUnit" : { "state" : "translated", - "value" : "Vigtig information" + "value" : "Mere information" } }, "de" : { "stringUnit" : { "state" : "translated", - "value" : "Wichtige Hinweise" + "value" : "Mehr Informationen" } }, "en" : { "stringUnit" : { "state" : "translated", - "value" : "Important Notes" + "value" : "More Info" } }, "es" : { "stringUnit" : { "state" : "translated", - "value" : "Notas importantes" + "value" : "Más Información" } }, "fr" : { "stringUnit" : { "state" : "translated", - "value" : "Informations importantes" + "value" : "Plus d'informations" } }, "hi" : { "stringUnit" : { "state" : "translated", - "value" : "आवश्यक जानकारी" + "value" : "अधिकतर जानकारी" } }, "it" : { "stringUnit" : { "state" : "translated", - "value" : "Note importanti" + "value" : "Informazioni ulteriori" } }, "ja" : { "stringUnit" : { "state" : "translated", - "value" : "重要" + "value" : "詳細情報" } }, "ko" : { "stringUnit" : { "state" : "translated", - "value" : "중요 사항" + "value" : "자세한 정보" } }, "nb" : { "stringUnit" : { "state" : "translated", - "value" : "Viktig informasjon" + "value" : "Mer informasjon" } }, "nl" : { "stringUnit" : { "state" : "translated", - "value" : "Belangrijke informatie" + "value" : "Meer Informatie" } }, "pl" : { "stringUnit" : { "state" : "translated", - "value" : "Ważne informacje" + "value" : "Więcej informacji" } }, "pt" : { "stringUnit" : { "state" : "translated", - "value" : "Notas Importantes" + "value" : "Mais Informações" } }, "ru" : { "stringUnit" : { "state" : "translated", - "value" : "Важные примечания" + "value" : "Информация" } }, "sv" : { "stringUnit" : { "state" : "translated", - "value" : "Viktig information" + "value" : "Mer information" } }, "uk" : { "stringUnit" : { "state" : "translated", - "value" : "Important Notes" + "value" : "Більше інформації" } }, "zh" : { "stringUnit" : { "state" : "translated", - "value" : "重要信息" + "value" : "更多信息" } } } }, - "Language:" : { - "comment" : "Language", + "One Day" : { + "comment" : "oneDayDeferralButtonText", "extractionState" : "manual", "localizations" : { "da" : { "stringUnit" : { "state" : "translated", - "value" : "Sprog:" + "value" : "En dag" } }, "de" : { "stringUnit" : { "state" : "translated", - "value" : "Sprache:" + "value" : "Ein Tag" } }, "en" : { "stringUnit" : { "state" : "translated", - "value" : "Language:" + "value" : "One Day" } }, "es" : { "stringUnit" : { "state" : "translated", - "value" : "Idioma:" + "value" : "Un Día" } }, "fr" : { "stringUnit" : { "state" : "translated", - "value" : "Langue :" + "value" : "Un jour" } }, "hi" : { "stringUnit" : { "state" : "translated", - "value" : "भाषा:" + "value" : "एक दिन" } }, "it" : { "stringUnit" : { "state" : "translated", - "value" : "Lingua:" + "value" : "Un giorno" } }, "ja" : { "stringUnit" : { "state" : "translated", - "value" : "言語:" + "value" : "1日" } }, "ko" : { "stringUnit" : { "state" : "translated", - "value" : "언어:" + "value" : "하루" } }, "nb" : { "stringUnit" : { "state" : "translated", - "value" : "Språk:" + "value" : "En dag" } }, "nl" : { "stringUnit" : { "state" : "translated", - "value" : "Taal:" + "value" : "Op een dag" } }, "pl" : { "stringUnit" : { "state" : "translated", - "value" : "Język:" + "value" : "1 dzień" } }, "pt" : { "stringUnit" : { "state" : "translated", - "value" : "Idioma:" + "value" : "Um Dia" } }, "ru" : { "stringUnit" : { "state" : "translated", - "value" : "Язык:" + "value" : "На день" } }, "sv" : { "stringUnit" : { "state" : "translated", - "value" : "Språk:" + "value" : "En dag" } }, "uk" : { "stringUnit" : { "state" : "translated", - "value" : "Мова:" + "value" : "Один день" } }, "zh" : { "stringUnit" : { "state" : "translated", - "value" : "语言:" + "value" : "一天" } } } }, - "Later" : { + "One Hour" : { + "comment" : "oneHourDeferralButtonText", "extractionState" : "manual", "localizations" : { "da" : { "stringUnit" : { "state" : "translated", - "value" : "Senere" + "value" : "En time" } }, "de" : { "stringUnit" : { "state" : "translated", - "value" : "Später" + "value" : "Eine Stunde" } }, "en" : { "stringUnit" : { "state" : "translated", - "value" : "Later" + "value" : "One Hour" } }, "es" : { "stringUnit" : { "state" : "translated", - "value" : "Más Tarde" + "value" : "Una Hora" } }, "fr" : { "stringUnit" : { "state" : "translated", - "value" : "Plus tard" + "value" : "Une heure" } }, "hi" : { "stringUnit" : { "state" : "translated", - "value" : "बाद मे" + "value" : "एक घंटा" } }, "it" : { "stringUnit" : { "state" : "translated", - "value" : "Dopo" + "value" : "Un'ora" } }, "ja" : { "stringUnit" : { "state" : "translated", - "value" : "後で" + "value" : "1時間" } }, "ko" : { "stringUnit" : { "state" : "translated", - "value" : "나중에" + "value" : "한 시간" } }, "nb" : { "stringUnit" : { "state" : "translated", - "value" : "Senere" + "value" : "En time" } }, "nl" : { "stringUnit" : { "state" : "translated", - "value" : "Later" + "value" : "Een uur" } }, "pl" : { "stringUnit" : { "state" : "translated", - "value" : "Później" + "value" : "1 godzina" } }, "pt" : { "stringUnit" : { "state" : "translated", - "value" : "Depois" + "value" : "Uma Hora" } }, "ru" : { "stringUnit" : { "state" : "translated", - "value" : "Потом" + "value" : "На час" } }, "sv" : { "stringUnit" : { "state" : "translated", - "value" : "Senare" + "value" : "En timme" } }, "uk" : { "stringUnit" : { "state" : "translated", - "value" : "Пізніше" + "value" : "Одна година" } }, "zh" : { "stringUnit" : { "state" : "translated", - "value" : "我同意" + "value" : "一小时" } } } }, - "More Info" : { - "comment" : "More Info", + "Please update your device to use this application" : { + "comment" : "Please update your device to use this application", "extractionState" : "manual", "localizations" : { "da" : { "stringUnit" : { "state" : "translated", - "value" : "Mere information" + "value" : "Opdater din Mac for at bruge denne app" } }, "de" : { "stringUnit" : { "state" : "translated", - "value" : "Mehr Informationen" + "value" : "Bitte aktualisiere dein Gerät, um dieses Programm zu verwenden." } }, "en" : { "stringUnit" : { "state" : "translated", - "value" : "More Info" + "value" : "Please update your device to use this application" } }, "es" : { "stringUnit" : { "state" : "translated", - "value" : "Más Información" + "value" : "Actualiza tu dispositivo para poder usar esta aplicación" } }, "fr" : { "stringUnit" : { "state" : "translated", - "value" : "Plus d'informations" + "value" : "Veuillez mettre à jour votre appareil pour utiliser cette application" } }, "hi" : { "stringUnit" : { "state" : "translated", - "value" : "अधिकतर जानकारी" + "value" : "कृपया इस ऐप का उपयोग करने के लिए अपने डिवाइस को अपडेट करें" } }, "it" : { "stringUnit" : { "state" : "translated", - "value" : "Informazioni ulteriori" + "value" : "Aggiorna il tuo dispositivo per utilizzare questa app" } }, "ja" : { "stringUnit" : { "state" : "translated", - "value" : "詳細情報" + "value" : "このアプリを使用するには、デバイスを更新してください" } }, "ko" : { "stringUnit" : { "state" : "translated", - "value" : "자세한 정보" + "value" : "이 앱을 사용하려면 기기를 업데이트하세요." } }, "nb" : { "stringUnit" : { "state" : "translated", - "value" : "Mer informasjon" + "value" : "Oppdater enheten din for å bruke denne appen" } }, "nl" : { "stringUnit" : { "state" : "translated", - "value" : "Meer Informatie" + "value" : "Werk uw apparaat bij om deze app te gebruiken" } }, "pl" : { "stringUnit" : { "state" : "translated", - "value" : "Więcej informacji" + "value" : "Zaktualizuj swoje urządzenie, aby używać tej aplikacji" } }, "pt" : { "stringUnit" : { "state" : "translated", - "value" : "Mais Informações" + "value" : "Atualize seu dispositivo para usar este aplicativo" } }, "ru" : { "stringUnit" : { "state" : "translated", - "value" : "Информация" + "value" : "Пожалуйста, обновите свое устройство, чтобы использовать это приложение" } }, "sv" : { "stringUnit" : { "state" : "translated", - "value" : "Mer information" + "value" : "Uppdatera din enhet för att använda den här appen" } }, "uk" : { "stringUnit" : { "state" : "translated", - "value" : "Більше інформації" + "value" : "Будь ласка, оновіть ваш пристрій щоб використовувати цю програму" } }, "zh" : { "stringUnit" : { "state" : "translated", - "value" : "更多信息" + "value" : "请更新您的设备以使用此应用" } } } }, - "One Day" : { + "Please work with your local IT team to obtain a replacement device" : { + "comment" : "mainContentSubHeaderUnsupported", "extractionState" : "manual", "localizations" : { "da" : { "stringUnit" : { "state" : "translated", - "value" : "En dag" + "value" : "Arbejd med dit lokale it-team for at få en erstatningsenhed" } }, "de" : { "stringUnit" : { "state" : "translated", - "value" : "Ein Tag" + "value" : "Bitte arbeiten Sie mit Ihrem lokalen IT-Team zusammen, um ein Ersatzgerät zu erhalten" } }, "en" : { "stringUnit" : { "state" : "translated", - "value" : "One Day" + "value" : "Please work with your local IT team to obtain a replacement device" } }, "es" : { "stringUnit" : { "state" : "translated", - "value" : "Un Día" + "value" : "Trabaje con su equipo de TI local para obtener un dispositivo de reemplazo." } }, "fr" : { "stringUnit" : { "state" : "translated", - "value" : "Un jour" + "value" : "Veuillez travailler avec votre équipe informatique locale pour obtenir un appareil de remplacement." } }, "hi" : { "stringUnit" : { "state" : "translated", - "value" : "एक दिन" + "value" : "प्रतिस्थापन उपकरण प्राप्त करने के लिए कृपया अपनी स्थानीय आईटी टीम के साथ काम करें" } }, "it" : { "stringUnit" : { "state" : "translated", - "value" : "Un giorno" + "value" : "Collabora con il team IT locale per ottenere un dispositivo sostitutivo" } }, "ja" : { "stringUnit" : { "state" : "translated", - "value" : "1日" + "value" : "地元の IT チームと協力して交換用デバイスを入手してください" } }, "ko" : { "stringUnit" : { "state" : "translated", - "value" : "하루" + "value" : "교체 장치를 얻으려면 현지 IT 팀과 협력하십시오." } }, "nb" : { "stringUnit" : { "state" : "translated", - "value" : "En dag" + "value" : "Samarbeid med ditt lokale IT-team for å få en erstatningsenhet" } }, "nl" : { "stringUnit" : { "state" : "translated", - "value" : "Op een dag" + "value" : "Werk samen met uw lokale IT-team om een vervangend apparaat te verkrijgen" } }, "pl" : { "stringUnit" : { "state" : "translated", - "value" : "1 dzień" + "value" : "Skontaktuj się z lokalnym zespołem IT, aby uzyskać urządzenie zastępcze" } }, "pt" : { "stringUnit" : { "state" : "translated", - "value" : "Um Dia" + "value" : "Trabalhe com sua equipe de TI local para obter um dispositivo de substituição" } }, "ru" : { "stringUnit" : { "state" : "translated", - "value" : "На день" + "value" : "Свяжитесь с местной ИТ-отделом, чтобы получить устройство на замену." } }, "sv" : { "stringUnit" : { "state" : "translated", - "value" : "En dag" + "value" : "Vänligen samarbeta med ditt lokala IT-team för att skaffa en ersättningsenhet" } }, "uk" : { "stringUnit" : { "state" : "translated", - "value" : "Один день" + "value" : "Зверніться до місцевої ІТ-служби, щоб отримати пристрій на заміну" } }, "zh" : { "stringUnit" : { "state" : "translated", - "value" : "一天" + "value" : "请与您当地的 IT 团队合作获取更换设备" } } } }, - "One Hour" : { + "Replace Your Device" : { + "comment" : "informationButtonTextUnsupported", "extractionState" : "manual", "localizations" : { "da" : { "stringUnit" : { "state" : "translated", - "value" : "En time" + "value" : "Udskift din enhed" } }, "de" : { "stringUnit" : { "state" : "translated", - "value" : "Eine Stunde" + "value" : "Ersetzen Sie Ihr Gerät" } }, "en" : { "stringUnit" : { "state" : "translated", - "value" : "One Hour" + "value" : "Replace Your Device" } }, "es" : { "stringUnit" : { "state" : "translated", - "value" : "Una Hora" + "value" : "Reemplace su dispositivo" } }, "fr" : { "stringUnit" : { "state" : "translated", - "value" : "Une heure" + "value" : "Remplacez votre appareil" } }, "hi" : { "stringUnit" : { "state" : "translated", - "value" : "एक घंटा" + "value" : "अपना उपकरण बदलें" } }, "it" : { "stringUnit" : { "state" : "translated", - "value" : "Un'ora" + "value" : "Sostituisci il tuo dispositivo" } }, "ja" : { "stringUnit" : { "state" : "translated", - "value" : "1時間" + "value" : "デバイスを交換してください" } }, "ko" : { "stringUnit" : { "state" : "translated", - "value" : "한 시간" + "value" : "장치 교체" } }, "nb" : { "stringUnit" : { "state" : "translated", - "value" : "En time" + "value" : "Bytt ut enheten din" } }, "nl" : { "stringUnit" : { "state" : "translated", - "value" : "Een uur" + "value" : "Vervang uw apparaat" } }, "pl" : { "stringUnit" : { "state" : "translated", - "value" : "1 godzina" + "value" : "Wymień swoje urządzenie" } }, "pt" : { "stringUnit" : { "state" : "translated", - "value" : "Uma Hora" + "value" : "Substitua seu dispositivo" } }, "ru" : { "stringUnit" : { "state" : "translated", - "value" : "На час" + "value" : "Замените ваше устройство" } }, "sv" : { "stringUnit" : { "state" : "translated", - "value" : "En timme" + "value" : "Byt ut din enhet" } }, "uk" : { "stringUnit" : { "state" : "translated", - "value" : "Одна година" + "value" : "Замініть свій пристрій" } }, "zh" : { "stringUnit" : { "state" : "translated", - "value" : "一小时" + "value" : "更换您的设备" } } } }, - "Please update your device to use this application" : { + "Required Date:" : { + "comment" : "Required Date:", "extractionState" : "manual", "localizations" : { "da" : { "stringUnit" : { "state" : "translated", - "value" : "Opdater din Mac for at bruge denne app" + "value" : "Påkrævet Dato:" } }, "de" : { "stringUnit" : { "state" : "translated", - "value" : "Bitte aktualisiere dein Gerät, um dieses Programm zu verwenden." + "value" : "Erforderliches Datum:" } }, "en" : { "stringUnit" : { "state" : "translated", - "value" : "Please update your device to use this application" + "value" : "Required Date:" } }, "es" : { "stringUnit" : { "state" : "translated", - "value" : "Actualiza tu dispositivo para poder usar esta aplicación" + "value" : "Fecha Requerida:" } }, "fr" : { "stringUnit" : { "state" : "translated", - "value" : "Veuillez mettre à jour votre appareil pour utiliser cette application" + "value" : "Date Requise:" } }, "hi" : { "stringUnit" : { "state" : "translated", - "value" : "कृपया इस ऐप का उपयोग करने के लिए अपने डिवाइस को अपडेट करें" + "value" : "तारीख चाहिए:" } }, "it" : { "stringUnit" : { "state" : "translated", - "value" : "Aggiorna il tuo dispositivo per utilizzare questa app" + "value" : "Data Richiesta:" } }, "ja" : { "stringUnit" : { "state" : "translated", - "value" : "このアプリを使用するには、デバイスを更新してください" + "value" : "必要な日付:" } }, "ko" : { "stringUnit" : { "state" : "translated", - "value" : "이 앱을 사용하려면 기기를 업데이트하세요." + "value" : "필수 날짜:" } }, "nb" : { "stringUnit" : { "state" : "translated", - "value" : "Oppdater enheten din for å bruke denne appen" + "value" : "Krevd Dato:" } }, "nl" : { "stringUnit" : { "state" : "translated", - "value" : "Werk uw apparaat bij om deze app te gebruiken" + "value" : "Verplichte Datum:" } }, "pl" : { "stringUnit" : { "state" : "translated", - "value" : "Zaktualizuj swoje urządzenie, aby używać tej aplikacji" + "value" : "Wymagana Data:" } }, "pt" : { "stringUnit" : { "state" : "translated", - "value" : "Atualize seu dispositivo para usar este aplicativo" + "value" : "Data Requerida:" } }, "ru" : { "stringUnit" : { "state" : "translated", - "value" : "Пожалуйста, обновите свое устройство, чтобы использовать это приложение" + "value" : "Требуемая дата:" } }, "sv" : { "stringUnit" : { "state" : "translated", - "value" : "Uppdatera din enhet för att använda den här appen" + "value" : "Nödvändiga Datumet:" } }, "uk" : { "stringUnit" : { "state" : "translated", - "value" : "Будь ласка, оновіть ваш пристрій щоб використовувати цю програму" + "value" : "Необхідна дата:" } }, "zh" : { "stringUnit" : { "state" : "translated", - "value" : "请更新您的设备以使用此应用" + "value" : "要求日期:" } } } }, "Required OS Version:" : { - "comment" : "Required OS Version", + "comment" : "Required OS Version:", "extractionState" : "manual", "localizations" : { "da" : { @@ -2584,7 +3342,7 @@ } }, "Serial Number:" : { - "comment" : "Serial Number", + "comment" : "Serial Number:", "extractionState" : "manual", "localizations" : { "da" : { @@ -2692,7 +3450,7 @@ } }, "Update Device" : { - "comment" : "// Right side of Nudge", + "comment" : "actionButtonText", "extractionState" : "manual", "localizations" : { "da" : { @@ -2800,6 +3558,7 @@ } }, "Updates can take around 30 minutes to complete" : { + "comment" : "mainContentSubHeader", "extractionState" : "manual", "localizations" : { "da" : { @@ -2907,7 +3666,7 @@ } }, "Username:" : { - "comment" : "Username", + "comment" : "Username:", "extractionState" : "manual", "localizations" : { "da" : { @@ -3015,7 +3774,7 @@ } }, "Version:" : { - "comment" : "Nudge Version", + "comment" : "Version:", "extractionState" : "manual", "localizations" : { "da" : { @@ -3123,6 +3882,7 @@ } }, "Your device requires a security update" : { + "comment" : "mainHeader", "extractionState" : "manual", "localizations" : { "da" : { @@ -3230,6 +3990,7 @@ } }, "Your device requires a security update (Demo Mode)" : { + "comment" : "mainHeader (Demo Mode)", "extractionState" : "manual", "localizations" : { "da" : { @@ -3335,113 +4096,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.xcodeproj/project.pbxproj b/Nudge.xcodeproj/project.pbxproj index 636edf4f..ece1e934 100644 --- a/Nudge.xcodeproj/project.pbxproj +++ b/Nudge.xcodeproj/project.pbxproj @@ -18,11 +18,17 @@ 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 */; }; + 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 */; }; + 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 */; }; 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 */; }; @@ -90,11 +96,15 @@ 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 = ""; }; + 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 = ""; }; 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 = ""; }; @@ -158,6 +168,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 = ""; @@ -177,6 +188,15 @@ path = Common; sourceTree = ""; }; + 634CE1032BB47433002C26C4 /* 3rd Party Assets */ = { + isa = PBXGroup; + children = ( + 634CE1082BB47480002C26C4 /* gdmf.swift */, + 631A6D752BF2654000DC1EF3 /* sofa.swift */, + ); + path = "3rd Party Assets"; + sourceTree = ""; + }; 639B6B3925DF1FEB00E38EC1 /* Preferences */ = { isa = PBXGroup; children = ( @@ -234,6 +254,7 @@ 63D7D0D625C9E9A400236281 = { isa = PBXGroup; children = ( + 633C4C702C3F6465005720F0 /* CHANGELOG.md */, 031B0F2125D8AE3200E68A28 /* Example Assets */, 6316F0E62832CA0700E1354D /* Schema */, 637CEBC02A30C9E700EFA3E9 /* Localizable.xcstrings */, @@ -257,6 +278,7 @@ 63D7D0E125C9E9A400236281 /* Nudge */ = { isa = PBXGroup; children = ( + 634CE1032BB47433002C26C4 /* 3rd Party Assets */, 73CC1D7729B81EE500FBF8E2 /* com.github.macadmins.Nudge.SMAppService.plist */, 639B6B5425DF374600E38EC1 /* UI */, 639B6B3925DF1FEB00E38EC1 /* Preferences */, @@ -435,6 +457,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 */, @@ -454,6 +477,7 @@ isa = PBXResourcesBuildPhase; buildActionMask = 2147483647; files = ( + 633C4C712C3F6465005720F0 /* CHANGELOG.md in Resources */, ); runOnlyForDeploymentPostprocessing = 0; }; @@ -467,6 +491,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 */, @@ -481,6 +506,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 */, @@ -497,6 +523,7 @@ files = ( 0BC9972C25CE2DFC0019FC8F /* OSVersionTests.swift in Sources */, 63D7D0F625C9E9A500236281 /* NudgeTests.swift in Sources */, + 634CE10A2BB47480002C26C4 /* gdmf.swift in Sources */, ); runOnlyForDeploymentPostprocessing = 0; }; @@ -505,6 +532,7 @@ buildActionMask = 2147483647; files = ( 63D7D10125C9E9A500236281 /* NudgeUITests.swift in Sources */, + 634CE10B2BB47480002C26C4 /* gdmf.swift in Sources */, ); runOnlyForDeploymentPostprocessing = 0; }; @@ -580,7 +608,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 +668,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 +697,8 @@ "$(inherited)", "@executable_path/../Frameworks", ); - MACOSX_DEPLOYMENT_TARGET = 11.0; - MARKETING_VERSION = 1.1.12; + MACOSX_DEPLOYMENT_TARGET = 12.0; + MARKETING_VERSION = 2.0.0; PRODUCT_BUNDLE_IDENTIFIER = com.github.macadmins.Nudge; PRODUCT_NAME = "$(TARGET_NAME)"; PROVISIONING_PROFILE_SPECIFIER = ""; @@ -700,8 +728,8 @@ "$(inherited)", "@executable_path/../Frameworks", ); - MACOSX_DEPLOYMENT_TARGET = 11.0; - MARKETING_VERSION = 1.1.12; + MACOSX_DEPLOYMENT_TARGET = 12.0; + MARKETING_VERSION = 2.0.0; PRODUCT_BUNDLE_IDENTIFIER = com.github.macadmins.Nudge; PRODUCT_NAME = "$(TARGET_NAME)"; PROVISIONING_PROFILE_SPECIFIER = ""; @@ -728,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)"; @@ -756,7 +784,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)"; 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/3rd Party Assets/gdmf.swift b/Nudge/3rd Party Assets/gdmf.swift new file mode 100644 index 00000000..1baeb122 --- /dev/null +++ b/Nudge/3rd Party Assets/gdmf.swift @@ -0,0 +1,176 @@ +// +// 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 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 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 { + 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/3rd Party Assets/sofa.swift b/Nudge/3rd Party Assets/sofa.swift new file mode 100644 index 00000000..6d8f3919 --- /dev/null +++ b/Nudge/3rd Party Assets/sofa.swift @@ -0,0 +1,304 @@ +// +// 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" + } +} + +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? + 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" + case build = "Build" + case releaseDate = "ReleaseDate" + case expirationDate = "ExpirationDate" + case supportedDevices = "SupportedDevices" + case securityInfo = "SecurityInfo" + case cves = "CVEs" + case activelyExploitedCVEs = "ActivelyExploitedCVEs" + case uniqueCVEsCount = "UniqueCVEsCount" + } +} + +extension LatestOS: OSInformation { + // All required properties are already implemented +} + +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 + + enum CodingKeys: String, CodingKey { + case updateName = "UpdateName" + case productVersion = "ProductVersion" + case releaseDate = "ReleaseDate" + case securityInfo = "SecurityInfo" + case supportedDevices = "SupportedDevices" + case cves = "CVEs" + case activelyExploitedCVEs = "ActivelyExploitedCVEs" + case uniqueCVEsCount = "UniqueCVEsCount" + case daysSincePreviousRelease = "DaysSincePreviousRelease" + } +} + +extension SecurityRelease: OSInformation { + var build: String { + "" + } // fake out build for now +} + +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 = .iso8601 // Use ISO 8601 date format + 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 + ) + } +} + +class SOFA: NSObject, URLSessionDelegate { + 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) + 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 responseData: Data? + var response: URLResponse? + var responseError: Error? + var responseCode: Int? + var eTag: String? + var successfulQuery = false + + // Retry loop + while attempts < maxRetries { + attempts += 1 + 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() + return + } + + responseCode = httpResponse.statusCode + response = resp + responseError = error + + if responseCode == 200 { + if let etag = httpResponse.allHeaderFields["Etag"] as? String { + eTag = etag + } + successfulQuery = true + + if let encoding = httpResponse.allHeaderFields["Content-Encoding"] as? String { + LogManager.debug("Content-Encoding: \(encoding)", logger: utilsLog) + } + + responseData = data + + } else if responseCode == 304 { + successfulQuery = true + } + + 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 + } + + // 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, responseCode, eTag) + } +} diff --git a/Nudge/Info.plist b/Nudge/Info.plist index c7214285..9eee977e 100644 --- a/Nudge/Info.plist +++ b/Nudge/Info.plist @@ -15,9 +15,9 @@ CFBundlePackageType $(PRODUCT_BUNDLE_PACKAGE_TYPE) CFBundleShortVersionString - 1.1.16 + 2.0.0 CFBundleVersion - 1.1.16 + 2.0.0 LSApplicationCategoryType public.app-category.utilities LSMinimumSystemVersion 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 10092ca9..f013ec68 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 } @@ -82,6 +82,12 @@ struct OptionalFeatureVariables { optionalFeaturesJSON?.acceptableCameraUsage ?? false } + + static var acceptableUpdatePreparingUsage: Bool { + optionalFeaturesProfile?["acceptableUpdatePreparingUsage"] as? Bool ?? + optionalFeaturesJSON?.acceptableUpdatePreparingUsage ?? + true + } static var acceptableScreenSharingUsage: Bool { optionalFeaturesProfile?["acceptableScreenSharingUsage"] as? Bool ?? @@ -113,6 +119,12 @@ struct OptionalFeatureVariables { false } + static var attemptToCheckForSupportedDevice: Bool { + optionalFeaturesProfile?["attemptToCheckForSupportedDevice"] as? Bool ?? + optionalFeaturesJSON?.attemptToCheckForSupportedDevice ?? + true + } + static var attemptToFetchMajorUpgrade: Bool { optionalFeaturesProfile?["attemptToFetchMajorUpgrade"] as? Bool ?? optionalFeaturesJSON?.attemptToFetchMajorUpgrade ?? @@ -125,29 +137,66 @@ struct OptionalFeatureVariables { [String]() } + static var customSOFAFeedURL: String { + optionalFeaturesProfile?["customSOFAFeedURL"] as? String ?? + optionalFeaturesJSON?.customSOFAFeedURL ?? + "https://sofafeed.macadmins.io/v1/macos_data_feed.json" + } + + 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 disableSoftwareUpdateWorkflow: Bool { - optionalFeaturesProfile?["disableSoftwareUpdateWorkflow"] as? Bool ?? - optionalFeaturesJSON?.disableSoftwareUpdateWorkflow ?? + static var honorFocusModes: Bool { + optionalFeaturesProfile?["honorFocusModes"] as? Bool ?? + optionalFeaturesJSON?.honorFocusModes ?? + false + } + + static var honorCycleTimersOnExit: Bool { + optionalFeaturesProfile?["honorCycleTimersOnExit"] as? Bool ?? + optionalFeaturesJSON?.honorCycleTimersOnExit ?? false } + static var refreshSOFAFeedTime: Int { + optionalFeaturesProfile?["refreshSOFAFeedTime"] as? Int ?? + optionalFeaturesJSON?.refreshSOFAFeedTime ?? + 86400 + } + static var terminateApplicationsOnLaunch: Bool { optionalFeaturesProfile?["terminateApplicationsOnLaunch"] as? Bool ?? optionalFeaturesJSON?.terminateApplicationsOnLaunch ?? false } + + static var utilizeSOFAFeed: Bool { + optionalFeaturesProfile?["utilizeSOFAFeed"] as? Bool ?? + optionalFeaturesJSON?.utilizeSOFAFeed ?? + true + } } // OS Version Requirements 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() @@ -157,15 +206,61 @@ struct OSVersionRequirementVariables { getAboutUpdateURL(OSVerReq: osVersionRequirementsJSON) ?? "" } - + + static var activelyExploitedCVEsMajorUpgradeSLA: Int { + osVersionRequirementsProfile?.activelyExploitedCVEsMajorUpgradeSLA ?? + osVersionRequirementsJSON?.activelyExploitedCVEsMajorUpgradeSLA ?? + 14 + } + + static var activelyExploitedCVEsMinorUpdateSLA: Int { + osVersionRequirementsProfile?.activelyExploitedCVEsMinorUpdateSLA ?? + osVersionRequirementsJSON?.activelyExploitedCVEsMinorUpdateSLA ?? + 14 + } + static var majorUpgradeAppPath: String { osVersionRequirementsProfile?.majorUpgradeAppPath ?? osVersionRequirementsJSON?.majorUpgradeAppPath ?? "" } - + + static var nonActivelyExploitedCVEsMajorUpgradeSLA: Int { + osVersionRequirementsProfile?.nonActivelyExploitedCVEsMajorUpgradeSLA ?? + osVersionRequirementsJSON?.nonActivelyExploitedCVEsMajorUpgradeSLA ?? + 21 + } + + static var nonActivelyExploitedCVEsMinorUpdateSLA: Int { + osVersionRequirementsProfile?.nonActivelyExploitedCVEsMinorUpdateSLA ?? + osVersionRequirementsJSON?.nonActivelyExploitedCVEsMinorUpdateSLA ?? + 21 + } + static var requiredMinimumOSVersion: String { - try! OSVersion(PrefsWrapper.requiredMinimumOSVersion).description + if ["latest", "latest-supported", "latest-minor"].contains(PrefsWrapper.requiredMinimumOSVersion) { + PrefsWrapper.requiredMinimumOSVersion + } else { + try! OSVersion(PrefsWrapper.requiredMinimumOSVersion).description + } + } + + static var standardMajorUpgradeSLA: Int { + osVersionRequirementsProfile?.standardMajorUpgradeSLA ?? + osVersionRequirementsJSON?.standardMajorUpgradeSLA ?? + 28 + } + + static var standardMinorUpdateSLA: Int { + osVersionRequirementsProfile?.standardMinorUpdateSLA ?? + osVersionRequirementsJSON?.standardMinorUpdateSLA ?? + 28 + } + + static var unsupportedURL: String { + getUnsupportedURL(OSVerReq: osVersionRequirementsProfile) ?? + getUnsupportedURL(OSVerReq: osVersionRequirementsJSON) ?? + "" } } @@ -178,13 +273,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 ?? @@ -287,6 +388,18 @@ struct UserExperienceVariables { false } + static var nudgeMinorUpdateEventLaunchDelay: Int { + userExperienceProfile?["nudgeMinorUpdateEventLaunchDelay"] as? Int ?? + userExperienceJSON?.nudgeMinorUpdateEventLaunchDelay ?? + 0 + } + + static var nudgeMajorUpgradeEventLaunchDelay: Int { + userExperienceProfile?["nudgeMajorUpgradeEventLaunchDelay"] as? Int ?? + userExperienceJSON?.nudgeMajorUpgradeEventLaunchDelay ?? + 0 + } + static var nudgeRefreshCycle: Int { userExperienceProfile?["nudgeRefreshCycle"] as? Int ?? userExperienceJSON?.nudgeRefreshCycle ?? @@ -306,7 +419,49 @@ struct UserInterfaceVariables { static var userInterfaceJSON: UserInterface? = getUserInterfaceJSON() 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 actionButtonTextUnsupported: String { + userInterfaceUpdateElementsProfile?["actionButtonTextUnsupported"] as? String ?? + userInterfaceUpdateElementsJSON?.actionButtonTextUnsupported ?? + "Replace Your 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 ?? userInterfaceJSON?.fallbackLanguage ?? @@ -331,52 +486,70 @@ 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 informationButtonText: String { userInterfaceUpdateElementsProfile?["informationButtonText"] as? String ?? userInterfaceUpdateElementsJSON?.informationButtonText ?? "More Info" } - + 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 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**" + } + 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." + "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 { @@ -384,6 +557,30 @@ struct UserInterfaceVariables { 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 ?? @@ -391,12 +588,30 @@ 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 ?? true } + static var showDaysRemainingToUpdate: Bool { + userInterfaceProfile?["showDaysRemainingToUpdate"] as? Bool ?? + userInterfaceJSON?.showDaysRemainingToUpdate ?? + true + } + + static var showRequiredDate: Bool { + userInterfaceProfile?["showRequiredDate"] as? Bool ?? + userInterfaceJSON?.showRequiredDate ?? + false + } + static var singleQuitButton: Bool { userInterfaceProfile?["singleQuitButton"] as? Bool ?? userInterfaceJSON?.singleQuitButton ?? @@ -406,37 +621,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 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" + static var subHeaderUnsupported: String { + userInterfaceUpdateElementsProfile?["subHeaderUnsupported"] as? String ?? + userInterfaceUpdateElementsJSON?.subHeaderUnsupported ?? + "**A friendly reminder from your local IT team**" } } @@ -447,6 +638,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", @@ -459,6 +651,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/Preferences/PreferencesStructure.swift b/Nudge/Preferences/PreferencesStructure.swift index d4f64507..085b6e8b 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 { @@ -51,9 +66,12 @@ extension NudgePreferences { // MARK: - OptionalFeatures struct OptionalFeatures: Codable { var acceptableApplicationBundleIDs, acceptableAssertionApplicationNames: [String]? - var acceptableAssertionUsage, acceptableCameraUsage, acceptableScreenSharingUsage, aggressiveUserExperience, aggressiveUserFullScreenExperience, asynchronousSoftwareUpdate, attemptToBlockApplicationLaunches, attemptToFetchMajorUpgrade: Bool? + var acceptableAssertionUsage, acceptableCameraUsage, acceptableUpdatePreparingUsage, acceptableScreenSharingUsage, aggressiveUserExperience, aggressiveUserFullScreenExperience, asynchronousSoftwareUpdate, attemptToBlockApplicationLaunches, attemptToCheckForSupportedDevice, attemptToFetchMajorUpgrade: Bool? var blockedApplicationBundleIDs: [String]? - var disableSoftwareUpdateWorkflow, enforceMinorUpdates, terminateApplicationsOnLaunch: Bool? + var customSOFAFeedURL: String? + var disableNudgeForStandardInstalls, disableSoftwareUpdateWorkflow, enforceMinorUpdates, honorFocusModes, honorCycleTimersOnExit: Bool? + var refreshSOFAFeedTime: Int? + var terminateApplicationsOnLaunch, utilizeSOFAFeed: Bool? } // MARK: OptionalFeatures convenience initializers and mutators @@ -79,32 +97,48 @@ extension OptionalFeatures { acceptableAssertionApplicationNames: [String]? = nil, acceptableAssertionUsage: Bool? = nil, acceptableCameraUsage: Bool? = nil, + acceptableUpdatePreparingUsage: Bool? = nil, acceptableScreenSharingUsage: Bool? = nil, aggressiveUserExperience: Bool? = nil, aggressiveUserFullScreenExperience: Bool? = nil, asynchronousSoftwareUpdate: Bool? = nil, attemptToBlockApplicationLaunches: Bool? = nil, + attemptToCheckForSupportedDevice: Bool? = nil, attemptToFetchMajorUpgrade: Bool? = nil, blockedApplicationBundleIDs: [String]? = nil, + customSOFAFeedURL: String? = nil, + disableNudgeForStandardInstalls: Bool? = nil, disableSoftwareUpdateWorkflow: Bool? = nil, enforceMinorUpdates: Bool? = nil, - terminateApplicationsOnLaunch: Bool? = nil + honorFocusModes: Bool? = nil, + honorCycleTimersOnExit: Bool? = nil, + refreshSOFAFeedTime: Int? = nil, + terminateApplicationsOnLaunch: Bool? = nil, + utilizeSOFAFeed: Bool? = nil ) -> OptionalFeatures { return OptionalFeatures( acceptableApplicationBundleIDs: acceptableApplicationBundleIDs ?? self.acceptableApplicationBundleIDs, 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, asynchronousSoftwareUpdate: asynchronousSoftwareUpdate ?? self.asynchronousSoftwareUpdate, attemptToBlockApplicationLaunches: attemptToBlockApplicationLaunches ?? self.attemptToBlockApplicationLaunches, + 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, - terminateApplicationsOnLaunch: terminateApplicationsOnLaunch ?? self.terminateApplicationsOnLaunch + honorFocusModes: honorFocusModes ?? self.honorFocusModes, + honorCycleTimersOnExit: honorCycleTimersOnExit ?? self.honorCycleTimersOnExit, + refreshSOFAFeedTime: refreshSOFAFeedTime ?? self.refreshSOFAFeedTime, + terminateApplicationsOnLaunch: terminateApplicationsOnLaunch ?? self.terminateApplicationsOnLaunch, + utilizeSOFAFeed: utilizeSOFAFeed ?? self.utilizeSOFAFeed ) } } @@ -113,9 +147,19 @@ extension OptionalFeatures { struct OSVersionRequirement: Codable { var aboutUpdateURL: String? var aboutUpdateURLs: [AboutUpdateURL]? - var actionButtonPath, majorUpgradeAppPath: String? + var actionButtonPath: String? + var activelyExploitedCVEsMajorUpgradeSLA: Int? + var activelyExploitedCVEsMinorUpdateSLA: Int? + var majorUpgradeAppPath: String? + var nonActivelyExploitedCVEsMajorUpgradeSLA: Int? + var nonActivelyExploitedCVEsMinorUpdateSLA: Int? var requiredInstallationDate: Date? - var requiredMinimumOSVersion, targetedOSVersionsRule: String? + var requiredMinimumOSVersion: String? + var standardMajorUpgradeSLA: Int? + var standardMinorUpdateSLA: Int? + var targetedOSVersionsRule: String? + var unsupportedURL: String? + var unsupportedURLs: [UnsupportedURL]? } // MARK: OSVersionRequirement convenience initializers and mutators @@ -123,11 +167,18 @@ extension OSVersionRequirement { init(fromDictionary: [String: AnyObject]) { self.aboutUpdateURL = fromDictionary["aboutUpdateURL"] as? String self.actionButtonPath = fromDictionary["actionButtonPath"] as? String + self.activelyExploitedCVEsMajorUpgradeSLA = fromDictionary["activelyExploitedCVEsMajorUpgradeSLA"] as? Int + self.activelyExploitedCVEsMinorUpdateSLA = fromDictionary["activelyExploitedCVEsMinorUpdateSLA"] as? Int self.majorUpgradeAppPath = fromDictionary["majorUpgradeAppPath"] as? String + self.nonActivelyExploitedCVEsMajorUpgradeSLA = fromDictionary["nonActivelyExploitedCVEsMajorUpgradeSLA"] as? Int + self.nonActivelyExploitedCVEsMinorUpdateSLA = fromDictionary["nonActivelyExploitedCVEsMinorUpdateSLA"] as? Int self.requiredMinimumOSVersion = fromDictionary["requiredMinimumOSVersion"] as? String + self.standardMajorUpgradeSLA = fromDictionary["standardMajorUpgradeSLA"] as? Int + self.standardMinorUpdateSLA = fromDictionary["standardMinorUpdateSLA"] as? Int 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 } @@ -137,6 +188,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 @@ -150,8 +210,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 { @@ -170,19 +245,35 @@ extension OSVersionRequirement { aboutUpdateURL: String? = nil, aboutUpdateURLs: [AboutUpdateURL]? = nil, actionButtonPath: String? = nil, + activelyExploitedCVEsMajorUpgradeSLA: Int? = nil, + activelyExploitedCVEsMinorUpdateSLA: Int? = nil, majorUpgradeAppPath: String? = nil, + nonActivelyExploitedCVEsMajorUpgradeSLA: Int? = nil, + nonActivelyExploitedCVEsMinorUpdateSLA: Int? = nil, requiredInstallationDate: Date? = nil, requiredMinimumOSVersion: String? = nil, - targetedOSVersionsRule: String? = nil + standardMajorUpgradeSLA: Int? = nil, + standardMinorUpdateSLA: Int? = nil, + targetedOSVersionsRule: String? = nil, + unsupportedURL: String? = nil, + unsupportedURLs: [UnsupportedURL]? = nil ) -> OSVersionRequirement { return OSVersionRequirement( aboutUpdateURL: aboutUpdateURL ?? self.aboutUpdateURL, aboutUpdateURLs: aboutUpdateURLs ?? self.aboutUpdateURLs, actionButtonPath: actionButtonPath ?? self.actionButtonPath, + activelyExploitedCVEsMajorUpgradeSLA: activelyExploitedCVEsMajorUpgradeSLA ?? self.activelyExploitedCVEsMajorUpgradeSLA, + activelyExploitedCVEsMinorUpdateSLA: activelyExploitedCVEsMinorUpdateSLA ?? self.activelyExploitedCVEsMinorUpdateSLA, majorUpgradeAppPath: majorUpgradeAppPath ?? self.majorUpgradeAppPath, + nonActivelyExploitedCVEsMajorUpgradeSLA: nonActivelyExploitedCVEsMajorUpgradeSLA ?? self.nonActivelyExploitedCVEsMajorUpgradeSLA, + nonActivelyExploitedCVEsMinorUpdateSLA: nonActivelyExploitedCVEsMinorUpdateSLA ?? self.nonActivelyExploitedCVEsMinorUpdateSLA, requiredInstallationDate: requiredInstallationDate ?? self.requiredInstallationDate, requiredMinimumOSVersion: requiredMinimumOSVersion ?? self.requiredMinimumOSVersion, - targetedOSVersionsRule: targetedOSVersionsRule ?? self.targetedOSVersionsRule + standardMajorUpgradeSLA: standardMajorUpgradeSLA ?? self.standardMajorUpgradeSLA, + standardMinorUpdateSLA: standardMinorUpdateSLA ?? self.standardMinorUpdateSLA, + targetedOSVersionsRule: targetedOSVersionsRule ?? self.targetedOSVersionsRule, + unsupportedURL: unsupportedURL ?? self.unsupportedURL, + unsupportedURLs: unsupportedURLs ?? self.unsupportedURLs ) } } @@ -227,9 +318,49 @@ 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, allowUserQuitDeferrals: Bool? + var allowGracePeriods, allowLaterDeferralButton, allowMovableWindow, allowUserQuitDeferrals: Bool? var allowedDeferrals, allowedDeferralsUntilForcedSecondaryQuitButton, approachingRefreshCycle, approachingWindowTime: Int? var calendarDeferralUnit: String? var elapsedRefreshCycle, gracePeriodInstallDelay, gracePeriodLaunchDelay: Int? @@ -239,6 +370,8 @@ struct UserExperience: Codable { var loadLaunchAgent: Bool? var maxRandomDelayInSeconds: Int? var noTimers: Bool? + var nudgeMajorUpgradeEventLaunchDelay: Int? + var nudgeMinorUpdateEventLaunchDelay: Int? var nudgeRefreshCycle: Int? var randomDelay: Bool? } @@ -264,6 +397,7 @@ extension UserExperience { func with( allowGracePeriods: Bool? = nil, allowLaterDeferralButton: Bool? = nil, + allowMovableWindow: Bool? = nil, allowUserQuitDeferrals: Bool? = nil, allowedDeferrals: Int? = nil, allowedDeferralsUntilForcedSecondaryQuitButton: Int? = nil, @@ -281,12 +415,15 @@ extension UserExperience { loadLaunchAgent: Bool? = nil, maxRandomDelayInSeconds: Int? = nil, noTimers: Bool? = nil, + nudgeMajorUpgradeEventLaunchDelay: Int? = nil, + nudgeMinorUpdateEventLaunchDelay: Int? = nil, nudgeRefreshCycle: Int? = nil, randomDelay: Bool? = nil ) -> 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, @@ -304,6 +441,8 @@ extension UserExperience { loadLaunchAgent: loadLaunchAgent ?? self.loadLaunchAgent, maxRandomDelayInSeconds: maxRandomDelayInSeconds ?? self.maxRandomDelayInSeconds, noTimers: noTimers ?? self.noTimers, + nudgeMajorUpgradeEventLaunchDelay: nudgeMajorUpgradeEventLaunchDelay ?? self.nudgeMajorUpgradeEventLaunchDelay, + nudgeMinorUpdateEventLaunchDelay: nudgeMinorUpdateEventLaunchDelay ?? self.nudgeMinorUpdateEventLaunchDelay, nudgeRefreshCycle: nudgeRefreshCycle ?? self.nudgeRefreshCycle, randomDelay: randomDelay ?? self.randomDelay ) @@ -312,10 +451,10 @@ 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 showDeferralCount, simpleMode, singleQuitButton: Bool? + var iconDarkPath, iconLightPath, requiredInstallationDisplayFormat, screenShotDarkPath, screenShotLightPath: String? + var showActivelyExploitedCVEs, showDeferralCount, showDaysRemainingToUpdate, showRequiredDate, simpleMode, singleQuitButton: Bool? var updateElements: [UpdateElement]? } @@ -339,28 +478,38 @@ extension UserInterface { func with( actionButtonPath: String? = nil, + applicationTerminatedNotificationImagePath: String? = nil, fallbackLanguage: String? = nil, forceFallbackLanguage: Bool? = nil, forceScreenShotIcon: Bool? = nil, iconDarkPath: String? = nil, iconLightPath: String? = nil, + requiredInstallationDisplayFormat: String? = nil, screenShotDarkPath: String? = nil, screenShotLightPath: String? = nil, + showActivelyExploitedCVEs: Bool? = nil, showDeferralCount: Bool? = nil, + showDaysRemainingToUpdate: Bool? = nil, + showRequiredDate: Bool? = nil, simpleMode: Bool? = nil, singleQuitButton: Bool? = nil, updateElements: [UpdateElement]? = nil ) -> UserInterface { return UserInterface( actionButtonPath: actionButtonPath ?? self.actionButtonPath, + applicationTerminatedNotificationImagePath: applicationTerminatedNotificationImagePath ?? self.applicationTerminatedNotificationImagePath, fallbackLanguage: fallbackLanguage ?? self.fallbackLanguage, forceFallbackLanguage: forceFallbackLanguage ?? self.forceFallbackLanguage, 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, + showDaysRemainingToUpdate: showDaysRemainingToUpdate ?? self.showDaysRemainingToUpdate, + showRequiredDate: showRequiredDate ?? self.showRequiredDate, simpleMode: simpleMode ?? self.simpleMode, singleQuitButton: singleQuitButton ?? self.singleQuitButton, updateElements: updateElements ?? self.updateElements @@ -370,13 +519,14 @@ extension UserInterface { // MARK: - UpdateElement 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 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, customDeferralButtonText, customDeferralDropdownText, informationButtonText, mainContentHeader, mainContentNote, mainContentSubHeader, mainContentText, mainHeader, oneDayDeferralButtonText, oneHourDeferralButtonText, primaryQuitButtonText, secondaryQuitButtonText, subHeader, screenShotAltText + 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 } } @@ -401,38 +551,56 @@ 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, mainContentHeader: 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, primaryQuitButtonText: String? = nil, + screenShotAltText: String? = nil, secondaryQuitButtonText: String? = nil, subHeader: String? = nil, - screenShotAltText: String? = nil + subHeaderUnsupported: String? = nil ) -> 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, mainContentHeader: mainContentHeader ?? self.mainContentHeader, + 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, primaryQuitButtonText: primaryQuitButtonText ?? self.primaryQuitButtonText, + screenShotAltText: screenShotAltText ?? self.screenShotAltText, secondaryQuitButtonText: secondaryQuitButtonText ?? self.secondaryQuitButtonText, subHeader: subHeader ?? self.subHeader, - screenShotAltText: screenShotAltText ?? self.screenShotAltText + subHeaderUnsupported: subHeaderUnsupported ?? self.subHeaderUnsupported ) } } diff --git a/Nudge/UI/Common/CompanyLogo.swift b/Nudge/UI/Common/CompanyLogo.swift index 2c7c1699..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 { @@ -28,8 +29,35 @@ 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 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 { @@ -53,7 +81,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/Common/InformationButton.swift b/Nudge/UI/Common/InformationButton.swift index 3b9bfeb0..80dd4ea6 100644 --- a/Nudge/UI/Common/InformationButton.swift +++ b/Nudge/UI/Common/InformationButton.swift @@ -20,14 +20,14 @@ 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))) + 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))) + .help("Click for more information about the security update.".localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .onHoverEffect() ) } @@ -37,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/Common/QuitButtons.swift b/Nudge/UI/Common/QuitButtons.swift index 4c200d7a..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 { @@ -59,13 +66,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 @@ -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 } @@ -106,14 +119,16 @@ 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() } 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() { diff --git a/Nudge/UI/Defaults.swift b/Nudge/UI/Defaults.swift index 11314570..7f750d07 100644 --- a/Nudge/UI/Defaults.swift +++ b/Nudge/UI/Defaults.swift @@ -23,9 +23,13 @@ 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 + static let gdmfAssets = NetworkFileManager().getGDMFAssets() + static let sofaAssets = NetworkFileManager().getSOFAAssets() + static let hardwareModelIDs = DeviceManager().getHardwareModelIDs() } struct Intervals { @@ -61,18 +65,23 @@ struct UIConstants { } class AppState: ObservableObject { + @Published var activelyExploitedCVEs = false @Published var afterFirstStateChange = false @Published var allowButtons = true @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 @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 @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 9714903f..a64ec104 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 } @@ -110,6 +110,7 @@ struct ContentView: View { } appState.daysRemaining = DateManager().getNumberOfDaysBetween() appState.hoursRemaining = DateManager().getNumberOfHoursRemaining() + appState.secondsRemaining = DateManager().getNumberOfSecondsRemaining() } } @@ -171,6 +172,98 @@ class AppDelegate: NSObject, NSApplicationDelegate { // print("applicationWillBecomeActive") } + func sofaPreLaunchLogic() { + // TODO: Add more logging to "unsupported devices" UI. + if OptionalFeatureVariables.utilizeSOFAFeed { + var selectedOS: OSInformation? + var foundMatch = false + if let macOSSOFAAssets = Globals.sofaAssets?.osVersions { + for osVersion in macOSSOFAAssets { + 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 + } + } else if PrefsWrapper.requiredMinimumOSVersion == "latest-supported" { + if OptionalFeatureVariables.attemptToCheckForSupportedDevice { + selectedOS = osVersion.securityReleases.first + if !selectedOS!.supportedDevices.contains(where: { supportedDevice in Globals.hardwareModelIDs.contains { $0.uppercased() == supportedDevice.uppercased() } }) { + continue + } + } else { + LogManager.notice("Attempting to use latest-supported without supported device UI features. Please set attemptToCheckForSupportedDevice to true", logger: sofaLog) + break + } + } else { + if osVersion.securityReleases.first(where: { $0.productVersion == nudgePrimaryState.requiredMinimumOSVersion }) != nil { + selectedOS = osVersion.securityReleases.first(where: { $0.productVersion == nudgePrimaryState.requiredMinimumOSVersion }) + } else { + continue + } + } + let activelyExploitedCVEs = selectedOS!.activelyExploitedCVEs.count > 0 + let presentCVEs = selectedOS!.cves.count > 0 + let slaExtension: TimeInterval + 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 { + 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 + releaseDate = selectedOS!.releaseDate ?? Date() + 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) + LogManager.notice("SOFA Assets: \(selectedOS!.supportedDevices)", logger: sofaLog) + LogManager.notice("SOFA CVEs: \(selectedOS!.cves)", logger: sofaLog) + + if OptionalFeatureVariables.attemptToCheckForSupportedDevice { + 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 // false + } + 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.error("Could not fetch SOFA feed", logger: sofaLog) + nudgePrimaryState.shouldExit = true + exit(1) + } + } + } + // Pre-Launch Logic func applicationWillFinishLaunching(_ notification: Notification) { // print("applicationWillFinishLaunching") @@ -178,6 +271,8 @@ class AppDelegate: NSObject, NSApplicationDelegate { checkForBadProfilePath() handleCommandLineArguments() applyGracePeriodLogic() + sofaPreLaunchLogic() + applydelayNudgeEventLogic() applyRandomDelayIfNecessary() updateNudgeState() handleSoftwareUpdateRequirements() @@ -215,16 +310,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 @@ -232,7 +330,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() { @@ -243,13 +341,20 @@ 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)) } } + private func applydelayNudgeEventLogic() { + _ = AppStateManager().delayNudgeEventLogic() + 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) { @@ -273,11 +378,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 { + LogManager.error("applicationTerminatedNotificationImagePath does not exist on disk, skipping notification image.", logger: uiLog) + } return content } @@ -343,9 +506,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) + } } } @@ -445,6 +612,7 @@ class AppDelegate: NSObject, NSApplicationDelegate { setupScreenChangeObservers() setupScreenLockObservers() setupWorkspaceNotificationCenterObservers() + setupUserDefaultsObservers() } private func setupNotificationCenterObservers() { @@ -452,11 +620,21 @@ class AppDelegate: NSObject, NSApplicationDelegate { forName: NSWindow.didChangeScreenNotification, object: NSApplication.shared, queue: .main) { _ in - print("Window object frame moved - Notification Center") + if UserExperienceVariables.allowMovableWindow { return } + LogManager.debug("Window object frame moved - Notification Center", logger: utilsLog) UIUtilities().centerNudge() } } + 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, @@ -514,10 +692,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 ?? "" @@ -528,8 +707,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() { @@ -548,15 +731,18 @@ class AppDelegate: NSObject, NSApplicationDelegate { class WindowDelegate: NSObject, NSWindowDelegate { func windowDidMove(_ notification: Notification) { - print("Window attempted to move - Window Delegate") + if UserExperienceVariables.allowMovableWindow { return } + LogManager.debug("Window attempted to move - Window Delegate", logger: utilsLog) UIUtilities().centerNudge() } func windowDidChangeScreen(_ notification: Notification) { - print("Window moved screens - Window Delegate") + if UserExperienceVariables.allowMovableWindow { return } + LogManager.debug("Window moved screens - Window Delegate", logger: utilsLog) UIUtilities().centerNudge() } func windowDidChangeScreenProfile(_ notification: Notification) { - print("Display has changed profiles - Window Delegate") + if UserExperienceVariables.allowMovableWindow { return } + LogManager.debug("Display has changed profiles - Window Delegate", logger: utilsLog) UIUtilities().centerNudge() } } diff --git a/Nudge/UI/SimpleMode/SimpleMode.swift b/Nudge/UI/SimpleMode/SimpleMode.swift index 4673fe32..26cadae0 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 ? getMainHeader().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)) : getMainHeaderUnsupported().localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) .font(.title) remainingTimeView @@ -40,7 +40,17 @@ struct SimpleMode: View { } Spacer() - updateButton + if appState.deviceSupportedByOSVersion { + Button(action: { + UIUtilities().updateDevice() + }) { + Text(UserInterfaceVariables.actionButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale))) + .frame(minWidth: 120) + } + .keyboardShortcut(.defaultAction) + } else { + InformationButtonAsAction() + } Spacer() } } @@ -76,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/LeftSide.swift b/Nudge/UI/StandardMode/LeftSide.swift index a57a9c05..3c429933 100644 --- a/Nudge/UI/StandardMode/LeftSide.swift +++ b/Nudge/UI/StandardMode/LeftSide.swift @@ -36,9 +36,17 @@ 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) + 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)) } @@ -48,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) @@ -56,7 +64,7 @@ struct StandardModeLeftSide: View { } } - private var shouldShowDaysRemaining: Bool { + private var shouldshowDaysRemainingToUpdate: Bool { ((appState.daysRemaining > 0 || 0 > appState.hoursRemaining) && !CommandLineUtilities().demoModeEnabled()) || CommandLineUtilities().demoModeEnabled() } } @@ -79,12 +87,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/UI/StandardMode/RightSide.swift b/Nudge/UI/StandardMode/RightSide.swift index 60d71f5b..00c31283 100644 --- a/Nudge/UI/StandardMode/RightSide.swift +++ b/Nudge/UI/StandardMode/RightSide.swift @@ -38,17 +38,16 @@ struct StandardModeRightSide: View { HStack { VStack(alignment: .leading, spacing: 5) { HStack { - Text(getMainHeader().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) .lineLimit(1) } - + HStack { - Text(UserInterfaceVariables.subHeader.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,33 +63,35 @@ struct StandardModeRightSide: View { HStack(alignment: .center) { VStack(alignment: .leading, spacing: 1) { HStack { - Text(UserInterfaceVariables.mainContentHeader.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 { - Text(UserInterfaceVariables.mainContentSubHeader.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(.init(UserInterfaceVariables.actionButtonText.localized(desiredLanguage: getDesiredLanguage(locale: appState.locale)))) + } + .keyboardShortcut(.defaultAction) + } else { + InformationButtonAsAction() } - .keyboardShortcut(.defaultAction) } Divider() HStack { - Text(UserInterfaceVariables.mainContentNote.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() } @@ -98,7 +99,7 @@ struct StandardModeRightSide: View { ScrollView(.vertical) { VStack { HStack { - Text(UserInterfaceVariables.mainContentText.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() @@ -112,7 +113,7 @@ struct StandardModeRightSide: View { private var screenshotDisplay: some View { Group { - if shouldShowScreenshot { + if shouldShowScreenshot() { screenshotButton } else { EmptyView() @@ -122,8 +123,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 +150,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..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 { @@ -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 + .customResizable() + @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/Logger.swift b/Nudge/Utilities/Logger.swift index 9bd156f5..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) @@ -44,8 +44,10 @@ 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") +let sofaLog = LogManager.createLogger(category: "sofa") // Log State class LogState { 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 diff --git a/Nudge/Utilities/Preferences.swift b/Nudge/Utilities/Preferences.swift index 9370d230..fdbd1288 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 { + LogManager.error("Failed to decode plist: \(error)", logger: prefsProfileLog) + } + } return UserInterfaceVariables.fallbackLanguage } @@ -33,19 +60,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 { + LogManager.error("Failed to decode plist: \(error)", logger: prefsProfileLog) + 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 +128,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 } - let requirements = osRequirementsArray.map { OSVersionRequirement(fromDictionary: $0) } - return getOSVersionRequirements(from: requirements) + // Check if the Data is empty + if Globals.configProfile.isEmpty { + logEmptyKey("osVersionRequirements", 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 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 { + LogManager.error("Failed to decode plist: \(error)", logger: prefsProfileLog) + return nil + } } private func getOSVersionRequirements(from requirements: [OSVersionRequirement]?) -> OSVersionRequirement? { @@ -122,6 +185,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? { @@ -135,13 +219,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 { + LogManager.error("Failed to decode plist: \(error)", logger: prefsProfileLog) + return nil + } } // userInterface @@ -166,13 +267,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 + } + + // Check if the Data is empty + if Globals.configProfile.isEmpty { logEmptyKey("userInterface", forJSON: false) return nil } - return userInterface + + 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 { + LogManager.error("Failed to decode plist: \(error)", logger: prefsProfileLog) + return nil + } } // Loop through JSON userInterface -> updateElements preferences and then compare language @@ -185,16 +303,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 { + LogManager.error("Failed to decode plist: \(error)", logger: prefsProfileLog) + return nil + } } private func getMatchingUpdateElements(updateElements: [T]?, languageKey: String, logKey: String) -> T? { @@ -239,6 +379,16 @@ func getMainHeader() -> String { getUserInterfaceUpdateElementsJSON()?.mainHeader ?? "Your device requires a security update" } +func getMainHeaderUnsupported() -> String { + if CommandLineUtilities().demoModeEnabled() { + return "Your device requires a security update (Demo Mode)" + } else if CommandLineUtilities().unitTestingEnabled() { + return "Your device requires a security update (Unit Testing Mode)" + } + return UserInterfaceVariables.userInterfaceUpdateElementsProfile?["mainHeaderUnsupported"] as? String ?? + getUserInterfaceUpdateElementsJSON()?.mainHeaderUnsupported ?? "Your device requires a security update" +} + func simpleMode() -> Bool { return CommandLineUtilities().simpleModeEnabled() || UserInterfaceVariables.userInterfaceProfile?["simpleMode"] as? Bool ?? diff --git a/Nudge/Utilities/SoftwareUpdate.swift b/Nudge/Utilities/SoftwareUpdate.swift index fed909fa..6955edcd 100644 --- a/Nudge/Utilities/SoftwareUpdate.swift +++ b/Nudge/Utilities/SoftwareUpdate.swift @@ -10,7 +10,7 @@ import os class SoftwareUpdate { 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) @@ -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) = 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,13 +54,13 @@ 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 } 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) @@ -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 } @@ -86,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/UILogic.swift b/Nudge/Utilities/UILogic.swift index 4764ec87..d84dd827 100644 --- a/Nudge/Utilities/UILogic.swift +++ b/Nudge/Utilities/UILogic.swift @@ -6,10 +6,18 @@ // import AppKit +import Darwin import Foundation import IOKit.pwr_mgt // Asertions import SwiftUI +struct ProcessInfoStruct { + let pid: Int32 + let command: String + let arguments: [String] + let username: String +} + func initialLaunchLogic() { guard !CommandLineUtilities().unitTestingEnabled() else { LogManager.debug("App being ran in test mode", logger: uiLog) @@ -177,6 +185,129 @@ 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 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 arguments = getArgumentsForPID(pid: pid) + let username = getUsernameForPID(pid: pid) + + processes.append(ProcessInfoStruct(pid: pid, command: command, arguments: arguments, username: username)) + } + + 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 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 (commandPattern, arguments, username) in commandsWithArgs { + let matchingProcesses = processes.filter { process in + 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 + } + } + return false +} + +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. + ("*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. + ("*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) && !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 { + LogManager.error("Error: \(result.error)", logger: utilsLog) + return false + } + + return result.output.contains(snapshotName) +} + func needToActivateNudge() -> Bool { if NSApplication.shared.isActive && nudgeLogState.afterFirstLaunch { LogManager.notice("Nudge is currently the frontmostApplication", logger: uiLog) @@ -245,7 +376,9 @@ private func shouldActivateNudgeBasedOnAggressiveExperience(_ runningApplication } AppStateManager().activateNudge() if !CommandLineUtilities().unitTestingEnabled() { - UIUtilities().updateDevice(userClicked: false) + if nudgePrimaryState.deviceSupportedByOSVersion { + UIUtilities().updateDevice(userClicked: false) + } } return true } else { @@ -263,6 +396,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() @@ -311,6 +445,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 } diff --git a/Nudge/Utilities/Utils.swift b/Nudge/Utilities/Utils.swift index 8af2223f..c9f0fac7 100644 --- a/Nudge/Utilities/Utils.swift +++ b/Nudge/Utilities/Utils.swift @@ -8,6 +8,8 @@ import AppKit import CoreMediaIO import Foundation +import Intents +import IOKit #if canImport(ServiceManagement) import ServiceManagement #endif @@ -16,6 +18,13 @@ import SystemConfiguration struct AppStateManager { func activateNudge() { + if OptionalFeatureVariables.honorFocusModes { + 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) nudgePrimaryState.lastRefreshTime = DateManager().getCurrentDate() guard let mainWindow = NSApp.windows.first else { return } @@ -23,6 +32,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) @@ -102,12 +116,24 @@ 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) + if attributes?[.size] as? Int == 0 && testFileDate == nil { + return DateManager().coerceStringToDate(dateString: "2020-08-06T00:00:00Z") + } let creationDate = attributes?[.creationDate] as? Date return testFileDate ?? creationDate } + func getModifiedDateForPath(_ path: String, testFileDate: Date?) -> Date? { + let attributes = try? FileManager.default.attributesOfItem(atPath: path) + if attributes?[.size] as? Int == 0 && testFileDate == nil { + 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 @@ -158,6 +184,26 @@ struct AppStateManager { return calculateNewRequiredInstallationDateIfNeeded(currentDate: currentDate, gracePeriodPathCreationDate: gracePeriodPathCreationDate) } + func delayNudgeEventLogic(currentDate: Date = DateManager().getCurrentDate(), testFileDate: Date? = nil) -> Date { + let isMajorUpgradeRequired = AppStateManager().requireMajorUpgrade() + let launchDelay = isMajorUpgradeRequired ? UserExperienceVariables.nudgeMajorUpgradeEventLaunchDelay : UserExperienceVariables.nudgeMinorUpdateEventLaunchDelay + + if launchDelay == 0 { + return PrefsWrapper.requiredInstallationDate + } + + 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 { + let eventType = isMajorUpgradeRequired ? "nudgeMajorUpgradeEventLaunchDelay" : "nudgeMinorUpdateEventLaunchDelay" + LogManager.info("Device outside \(eventType)", logger: uiLog) + return PrefsWrapper.requiredInstallationDate + } + } + private func isDeferralAllowed(threshold: Int, logMessage: String) -> Bool { if CommandLineUtilities().demoModeEnabled() { return true @@ -170,6 +216,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 { + LogManager.info("Key '\(key)' not found in preferences", logger: uiLog) + 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) @@ -232,7 +306,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) @@ -254,7 +328,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 @@ -280,10 +354,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-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 argument passed", logger: uiLog) + LogManager.debug("-bundle-mode-profile argument passed", logger: uiLog) nudgeLogState.hasLoggedBundleMode = true } return argumentPassed @@ -353,13 +436,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 @@ -381,10 +464,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), @@ -397,8 +503,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) @@ -418,14 +524,38 @@ struct DateManager { return formatter }() - private let dateFormatterCurrent: DateFormatter = { + let dateFormatterLocalTime: DateFormatter = { let formatter = DateFormatter() - formatter.dateFormat = "yyyy-MM-dd HH:mm:ss Z" + formatter.dateFormat = "yyyy-MM-dd'T'HH:mm:ss" 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() + if dateString.contains("Z") { + dateFormatterISO8601.date(from: dateString) ?? getCurrentDate() + } else { + dateFormatterLocalTime.date(from: dateString) ?? getCurrentDate() + } + } + + 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 { @@ -440,10 +570,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) } } @@ -461,6 +591,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 { @@ -480,6 +616,18 @@ struct DeviceManager { return result == -1 ? -1 : Int(cputype) } + 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 { // https://stackoverflow.com/a/63539782 let type = getCPUTypeInt() @@ -491,17 +639,35 @@ 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" } } + func getHardwareModel() -> String { + getSysctlValue(for: "hw.model") ?? "" + } + + func getHardwareModelIDs() -> [String] { + 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), gestaltModelStringID.trimmingCharacters(in: .whitespacesAndNewlines)] + } + func getHardwareUUID() -> String { guard !CommandLineUtilities().demoModeEnabled(), !CommandLineUtilities().unitTestingEnabled() else { @@ -510,6 +676,66 @@ 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 + 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 + } + } + + 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) @@ -517,7 +743,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, @@ -535,6 +761,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 @@ -597,7 +831,7 @@ struct ImageManager { struct LoggerUtilities { func logRequiredMinimumOSVersion() { if !requiredMinimumOSVersionNil() { - Globals.nudgeDefaults.set(OSVersionRequirementVariables.requiredMinimumOSVersion, forKey: "requiredMinimumOSVersion") + Globals.nudgeDefaults.set(nudgePrimaryState.requiredMinimumOSVersion, forKey: "requiredMinimumOSVersion") } } @@ -686,7 +920,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 } @@ -699,6 +933,149 @@ struct NetworkFileManager { } } + 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 + // 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 { + LogManager.error("Failed to decode gdmf JSON URL string", logger: utilsLog) + } + return nil + } + + func getSOFAAssets() -> MacOSDataFeed? { + if !OptionalFeatureVariables.utilizeSOFAFeed { + return nil + } + 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) + var sofaJSONExists = fileManager.fileExists(atPath: sofaPath.path) + + // Force delete if bad + if sofaJSONExists { + if isFileEmpty(atPath: sofaPath.path) { + do { + try fileManager.removeItem(atPath: sofaPath.path) + sofaJSONExists = false + } catch { + LogManager.error("Error deleting file: \(error.localizedDescription)", logger: sofaLog) + } + } + } + + if sofaJSONExists { + let sofaPathCreationDate = AppStateManager().getModifiedDateForPath(sofaPath.path, testFileDate: nil) + // 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 { + let sofaData = try Data(contentsOf: sofaPath) + let assetInfo = try MacOSDataFeed(data: sofaData) + return assetInfo + } 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) + // 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) + } + } 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: OptionalFeatureVariables.customSOFAFeedURL) { + let sofaData = SOFA().URLSync(url: url) + 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) + let assetInfo = try MacOSDataFeed(data: sofaData) + return assetInfo + } 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) + // Attempt to redownload and reprocess the file + return redownloadAndReprocessSOFA(url: url) + } + } else { + do { + if let data = sofaData.data { + if fileManager.fileExists(atPath: appDirectory.path) { + 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) + return redownloadAndReprocessSOFA(url: url) + } + } catch { + do { + try fileManager.removeItem(atPath: sofaPath.path) + sofaJSONExists = false + } catch { + 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) + // Attempt to redownload and reprocess the file + return redownloadAndReprocessSOFA(url: url) + } + } + } 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) + } + } + } else { + LogManager.error("Failed to decode sofa JSON URL string", logger: sofaLog) + } + return nil + } + func getBackupMajorUpgradeAppPath() -> String { switch VersionManager.getMajorRequiredNudgeOSVersion() { case 12: @@ -707,6 +1084,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" } @@ -723,11 +1102,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("Using embedded plist url: \(bundleUrl)", logger: utilsLog) + return nil + } + if let jsonUrl = URL(string: url) { LogManager.debug("JSON url: \(url)", logger: utilsLog) return decodeNudgePreferences(from: jsonUrl) @@ -736,6 +1120,81 @@ 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 { + LogManager.error("Error getting file attributes: \(error.localizedDescription)", logger: prefsJSONLog) + } + 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 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) + } + } 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 ?? "Unknown error")", logger: sofaLog) + } + } + return nil + } +} + +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 { @@ -800,6 +1259,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 { @@ -875,9 +1342,17 @@ struct UIUtilities { NSWorkspace.shared.open(url) } - private func postUpdateDeviceActions(userClicked: Bool) { + 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) + } + + 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 @@ -888,7 +1363,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) } } @@ -929,7 +1404,7 @@ struct UIUtilities { } } - postUpdateDeviceActions(userClicked: userClicked) + postUpdateDeviceActions(userClicked: userClicked, unSupportedUI: false) } func userInitiatedExit() { @@ -942,7 +1417,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) @@ -958,8 +1433,19 @@ struct VersionManager { } static func getMajorRequiredNudgeOSVersion() -> Int { - guard let majorVersion = Int(OSVersionRequirementVariables.requiredMinimumOSVersion.split(separator: ".").first ?? "") else { - LogManager.error("Invalid format for 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") @@ -981,7 +1467,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 @@ -1011,7 +1497,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 diff --git a/Schema/jamf/com.github.macadmins.Nudge.json b/Schema/jamf/com.github.macadmins.Nudge.json index c5096efe..358d5eb6 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": true, + "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": [ @@ -171,7 +199,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", @@ -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://sofafeed.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 + } + ] } } } @@ -345,6 +452,44 @@ } ] }, + "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", + "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": [ @@ -363,6 +508,44 @@ } ] }, + "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", + "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": [ @@ -399,8 +582,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.", + "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", @@ -408,21 +591,37 @@ }, { "title": "Configured", - "type": "array", - "items": { - "options": { - "inputAttributes": { - "placeholder": "11.5.1" - } - }, - "type": "string", - "title": "targetedOSVersion" + "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", + "type": "null" + }, + { + "title": "Configured", + "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", @@ -438,6 +637,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" + } + } + } + ] + } + } + } + } + ] } } } @@ -484,6 +757,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": [ @@ -796,8 +1083,8 @@ } ] }, - "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.", + "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", @@ -805,18 +1092,18 @@ }, { "title": "Configured", - "default": 60, + "default": 0, "type": "integer", "options": { "inputAttributes": { - "placeholder": "60" + "placeholder": "0" } } } ] }, - "randomDelay": { - "description": "Enables an initial delay Nudge before launching the UI. This is useful if you do not want your users to all receive the Nudge prompt at the exact time of the LaunchAgent.", + "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", @@ -824,11 +1111,49 @@ }, { "title": "Configured", - "default": false, - "type": "boolean", + "default": 0, + "type": "integer", "options": { "inputAttributes": { - "placeholder": "false" + "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": [ + { + "title": "Not Configured", + "type": "null" + }, + { + "title": "Configured", + "default": 60, + "type": "integer", + "options": { + "inputAttributes": { + "placeholder": "60" + } + } + } + ] + }, + "randomDelay": { + "description": "Enables an initial delay Nudge before launching the UI. This is useful if you do not want your users to all receive the Nudge prompt at the exact time of the LaunchAgent.", + "anyOf": [ + { + "title": "Not Configured", + "type": "null" + }, + { + "title": "Configured", + "default": false, + "type": "boolean", + "options": { + "inputAttributes": { + "placeholder": "false" } } } @@ -868,6 +1193,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": [ @@ -951,6 +1294,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": [ @@ -987,6 +1348,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": [ @@ -1001,6 +1376,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": [ @@ -1080,6 +1483,60 @@ } ] }, + "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", + "title": "Not Configured" + }, + { + "title": "Configured", + "type": "string", + "options": { + "inputAttributes": { + "placeholder": "Replace Your Device" + } + } + } + ] + }, + "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": [ @@ -1134,8 +1591,8 @@ } ] }, - "oneDayDeferralButtonText": { - "description": "Modifies the oneDayDeferralButtonText, also known as the \"One Day\" button.", + "mainContentHeader": { + "description": "Modifies the mainContentHeader. This is the \"Your device will restart during this update\" text.", "anyOf": [ { "type": "null", @@ -1146,14 +1603,14 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "One Day" + "placeholder": "**Your device will restart during this update**" } } } ] }, - "oneHourDeferralButtonText": { - "description": "Modifies the oneHourDeferralButtonText, also known as the \"One Hour\" button.", + "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", @@ -1164,14 +1621,14 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "One Hour" + "placeholder": "**Your device is no longer capable of receving critical security updates**" } } } ] }, - "mainContentHeader": { - "description": "Modifies the mainContentHeader. This is the \"Your device will restart during this update\" text.", + "mainContentNote": { + "description": "Modifies the mainContentNote. This is the \"Important Notes\" text.", "anyOf": [ { "type": "null", @@ -1182,14 +1639,14 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "Your device will restart during this update" + "placeholder": "**Important Notes**" } } } ] }, - "mainContentNote": { - "description": "Modifies the mainContentNote. This is the \"Important Notes\" text.", + "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", @@ -1200,7 +1657,7 @@ "type": "string", "options": { "inputAttributes": { - "placeholder": "Important Notes" + "placeholder": "**Important Notes**" } } } @@ -1224,6 +1681,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": [ @@ -1242,6 +1717,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": [ @@ -1260,6 +1753,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": [ @@ -1278,6 +1825,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": [ @@ -1308,7 +1873,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**" } } } 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"