From 8683dac7080845beff793f9dd447d43f307e8238 Mon Sep 17 00:00:00 2001 From: Allen Houchins Date: Thu, 6 Nov 2025 21:55:24 -0600 Subject: [PATCH] Add Privileges as macOS FMA Introduced Privileges as a maintained app for macOS, including input/output metadata, install/uninstall scripts, icon component, and app image. This enables management and display of Privileges in the software catalog. --- .../inputs/homebrew/privileges.json | 8 +++++++ ee/maintained-apps/outputs/apps.json | 9 +++++++- .../outputs/privileges/darwin.json | 21 ++++++++++++++++++ .../components/icons/Privileges.tsx | 14 ++++++++++++ .../images/app-icon-privileges-60x60@2x.png | Bin 0 -> 9190 bytes 5 files changed, 51 insertions(+), 1 deletion(-) create mode 100644 ee/maintained-apps/inputs/homebrew/privileges.json create mode 100644 ee/maintained-apps/outputs/privileges/darwin.json create mode 100644 frontend/pages/SoftwarePage/components/icons/Privileges.tsx create mode 100644 website/assets/images/app-icon-privileges-60x60@2x.png diff --git a/ee/maintained-apps/inputs/homebrew/privileges.json b/ee/maintained-apps/inputs/homebrew/privileges.json new file mode 100644 index 000000000000..1cddb2ed4358 --- /dev/null +++ b/ee/maintained-apps/inputs/homebrew/privileges.json @@ -0,0 +1,8 @@ +{ + "name": "Privileges", + "unique_identifier": "corp.sap.privileges", + "token": "privileges", + "installer_format": "dmg", + "slug": "privileges/darwin", + "default_categories": ["Productivity"] +} diff --git a/ee/maintained-apps/outputs/apps.json b/ee/maintained-apps/outputs/apps.json index 76f0641e55a2..82c8d21328fb 100644 --- a/ee/maintained-apps/outputs/apps.json +++ b/ee/maintained-apps/outputs/apps.json @@ -197,6 +197,13 @@ "unique_identifier": "com.postmanlabs.mac", "description": "Postman is a collaboration platform for API development that simplifies building, testing, and sharing APIs." }, + { + "name": "Privileges", + "slug": "privileges/darwin", + "platform": "darwin", + "unique_identifier": "corp.sap.privileges", + "description": "Privileges is an admin rights switcher for macOS." + }, { "name": "Santa", "slug": "santa/darwin", @@ -268,4 +275,4 @@ "description": "Zoom is a leading video communication platform for meetings, webinars, and collaboration." } ] -} +} \ No newline at end of file diff --git a/ee/maintained-apps/outputs/privileges/darwin.json b/ee/maintained-apps/outputs/privileges/darwin.json new file mode 100644 index 000000000000..216c2a05fe12 --- /dev/null +++ b/ee/maintained-apps/outputs/privileges/darwin.json @@ -0,0 +1,21 @@ +{ + "versions": [ + { + "version": "2.4.2", + "queries": { + "exists": "SELECT 1 FROM apps WHERE bundle_identifier = 'corp.sap.privileges';" + }, + "installer_url": "https://github.com/SAP/macOS-enterprise-privileges/releases/download/2.4.2/Privileges_2.4.2.pkg", + "install_script_ref": "53b579cd", + "uninstall_script_ref": "9b6ae988", + "sha256": "83e0d84ead1fdf35a20fbb1395aebcac3dc1430605c93f55731ad7e6d145f408", + "default_categories": [ + "Productivity" + ] + } + ], + "refs": { + "53b579cd": "#!/bin/sh\n\n# variables\nAPPDIR=\"/Applications/\"\nTMPDIR=$(dirname \"$(realpath $INSTALLER_PATH)\")\n\n# extract contents\nMOUNT_POINT=$(mktemp -d /tmp/dmg_mount_XXXXXX)\nhdiutil attach -plist -nobrowse -readonly -mountpoint \"$MOUNT_POINT\" \"$INSTALLER_PATH\"\nsudo cp -R \"$MOUNT_POINT\"/* \"$TMPDIR\"\nhdiutil detach \"$MOUNT_POINT\"\n# install pkg files\nsudo installer -pkg \"$TMPDIR/Privileges_2.4.2.pkg\" -target /\n", + "9b6ae988": "#!/bin/sh\n\n# variables\nLOGGED_IN_USER=$(scutil \u003c\u003c\u003c \"show State:/Users/ConsoleUser\" | awk '/Name :/ { print $3 }')\n# functions\n\nexpand_pkgid_and_map() {\n local PKGID=\"$1\"\n local FUNC=\"$2\"\n if [[ \"$PKGID\" == *\"*\" ]]; then\n local prefix=\"${PKGID%\\*}\"\n echo \"Expanding wildcard for PKGID: $PKGID\"\n for receipt in $(pkgutil --pkgs | grep \"^${prefix}\"); do\n echo \"Processing $receipt\"\n \"$FUNC\" \"$receipt\"\n done\n else\n \"$FUNC\" \"$PKGID\"\n fi\n}\n\nforget_pkg() {\n local PKGID=\"$1\"\n expand_pkgid_and_map \"$PKGID\" forget_receipt\n}\n\nforget_receipt() {\n local PKGID=\"$1\"\n sudo pkgutil --forget \"$PKGID\"\n}\n\nremove_launchctl_service() {\n local service=\"$1\"\n local booleans=(\"true\" \"false\")\n local plist_status\n local paths\n local should_sudo\n\n echo \"Removing launchctl service ${service}\"\n\n for should_sudo in \"${booleans[@]}\"; do\n plist_status=$(launchctl list \"${service}\" 2\u003e/dev/null)\n\n if [[ $plist_status == \\{* ]]; then\n if [[ $should_sudo == \"true\" ]]; then\n sudo launchctl remove \"${service}\"\n else\n launchctl remove \"${service}\"\n fi\n sleep 1\n fi\n\n paths=(\n \"/Library/LaunchAgents/${service}.plist\"\n \"/Library/LaunchDaemons/${service}.plist\"\n )\n\n # if not using sudo, prepend the home directory to the paths\n if [[ $should_sudo == \"false\" ]]; then\n for i in \"${!paths[@]}\"; do\n paths[i]=\"${HOME}${paths[i]}\"\n done\n fi\n\n for path in \"${paths[@]}\"; do\n if [[ -e \"$path\" ]]; then\n if [[ $should_sudo == \"true\" ]]; then\n sudo rm -f -- \"$path\"\n else\n rm -f -- \"$path\"\n fi\n fi\n done\n done\n}\n\nremove_pkg_files() {\n local PKGID=\"$1\"\n expand_pkgid_and_map \"$PKGID\" remove_receipt_files\n}\n\nremove_receipt_files() {\n local PKGID=\"$1\"\n local PKGINFO VOLUME INSTALL_LOCATION FULL_INSTALL_LOCATION\n\n echo \"pkgutil --pkg-info-plist \\\"$PKGID\\\"\"\n PKGINFO=$(pkgutil --pkg-info-plist \"$PKGID\")\n VOLUME=$(echo \"$PKGINFO\" | awk '/\u003ckey\u003evolume\u003c\\/key\u003e/ {getline; gsub(/.*\u003cstring\u003e|\u003c\\/string\u003e.*/, \"\"); print}')\n INSTALL_LOCATION=$(echo \"$PKGINFO\" | awk '/\u003ckey\u003einstall-location\u003c\\/key\u003e/ {getline; gsub(/.*\u003cstring\u003e|\u003c\\/string\u003e.*/, \"\"); print}')\n\n if [ -z \"$INSTALL_LOCATION\" ] || [ \"$INSTALL_LOCATION\" = \"/\" ]; then\n FULL_INSTALL_LOCATION=\"$VOLUME\"\n else\n FULL_INSTALL_LOCATION=\"$VOLUME/$INSTALL_LOCATION\"\n FULL_INSTALL_LOCATION=$(echo \"$FULL_INSTALL_LOCATION\" | sed 's|//|/|g')\n fi\n\n echo \"sudo pkgutil --only-files --files \\\"$PKGID\\\" | sed \\\"s|^|${FULL_INSTALL_LOCATION}/|\\\" | tr '\\\\\\\\n' '\\\\\\\\0' | /usr/bin/sudo -u root -E -- /usr/bin/xargs -0 -- /bin/rm -rf\"\n sudo pkgutil --only-files --files \"$PKGID\" | sed \"s|^|/${INSTALL_LOCATION}/|\" | tr '\\n' '\\0' | /usr/bin/sudo -u root -E -- /usr/bin/xargs -0 -- /bin/rm -rf\n\n echo \"sudo pkgutil --only-dirs --files \\\"$PKGID\\\" | sed \\\"s|^|${FULL_INSTALL_LOCATION}/|\\\" | grep '\\\\.app$' | tr '\\\\\\\\n' '\\\\\\\\0' | /usr/bin/sudo -u root -E -- /usr/bin/xargs -0 -- /bin/rm -rf\"\n sudo pkgutil --only-dirs --files \"$PKGID\" | sed \"s|^|${FULL_INSTALL_LOCATION}/|\" | grep '\\.app$' | tr '\\n' '\\0' | /usr/bin/sudo -u root -E -- /usr/bin/xargs -0 -- /bin/rm -rf\n\n root_app_dir=$(\n sudo pkgutil --only-dirs --files \"$PKGID\" \\\n | sed \"s|^|${FULL_INSTALL_LOCATION}/|\" \\\n | grep 'Applications' \\\n | awk '{ print length, $0 }' \\\n | sort -n \\\n | head -n1 \\\n | cut -d' ' -f2-\n )\n if [ -n \"$root_app_dir\" ]; then\n echo \"sudo rmdir -p \\\"$root_app_dir\\\" 2\u003e/dev/null || :\"\n sudo rmdir -p \"$root_app_dir\" 2\u003e/dev/null || :\n fi\n}\n\ntrash() {\n local logged_in_user=\"$1\"\n local target_file=\"$2\"\n local timestamp=\"$(date +%Y-%m-%d-%s)\"\n local rand=\"$(jot -r 1 0 99999)\"\n\n # replace ~ with /Users/$logged_in_user\n if [[ \"$target_file\" == ~* ]]; then\n target_file=\"/Users/$logged_in_user${target_file:1}\"\n fi\n\n local trash=\"/Users/$logged_in_user/.Trash\"\n local file_name=\"$(basename \"${target_file}\")\"\n\n if [[ -e \"$target_file\" ]]; then\n echo \"removing $target_file.\"\n mv -f \"$target_file\" \"$trash/${file_name}_${timestamp}_${rand}\"\n else\n echo \"$target_file doesn't exist.\"\n fi\n}\n\nremove_launchctl_service 'corp.sap.privileges.agent'\nremove_launchctl_service 'corp.sap.privileges.daemon'\nremove_launchctl_service 'corp.sap.privileges.watcher'\nremove_pkg_files 'corp.sap.privileges.pkg'\nforget_pkg 'corp.sap.privileges.pkg'\ntrash $LOGGED_IN_USER '~/Library/Application Scripts/corp.sap.privileges'\ntrash $LOGGED_IN_USER '~/Library/Containers/corp.sap.privileges'\ntrash $LOGGED_IN_USER '~/Library/Group Containers/*.corp.sap.privileges'\n" + } +} \ No newline at end of file diff --git a/frontend/pages/SoftwarePage/components/icons/Privileges.tsx b/frontend/pages/SoftwarePage/components/icons/Privileges.tsx new file mode 100644 index 000000000000..0a3f01b45a46 --- /dev/null +++ b/frontend/pages/SoftwarePage/components/icons/Privileges.tsx @@ -0,0 +1,14 @@ +import * as React from "react"; + +import type { SVGProps } from "react"; + +const Privileges = (props: SVGProps) => ( + + + +); +export default Privileges; diff --git a/website/assets/images/app-icon-privileges-60x60@2x.png b/website/assets/images/app-icon-privileges-60x60@2x.png new file mode 100644 index 0000000000000000000000000000000000000000..8c9f002bb042d4a466dd60bc5ee05616b142f2ab GIT binary patch literal 9190 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 literal 0 HcmV?d00001