65 Commits
v0.1 ... v0.2

Author SHA1 Message Date
Armin Briegel
ba22bb639c Update README.md 2020-06-09 11:51:26 +02:00
Armin Briegel
cd96c31ad3 fixed some typos 2020-06-09 11:51:20 +02:00
Armin Briegel
a7e731d1ed commented citrixworkspace until way to determine URL is found #17 2020-06-09 10:39:13 +02:00
Armin Briegel
0515a1c635 added swiftruntimeforcommandlinetools, closes #9 2020-06-09 10:30:23 +02:00
Armin Briegel
9bd79fc3e6 added sfsymbols, special handling for Apple signed apps without teamID 2020-06-09 10:25:14 +02:00
Armin Briegel
50d6df50d7 add ringcentralapp and ringcentralmeeting 2020-06-09 09:18:15 +02:00
Armin Briegel
4fd029e342 switched find commands to newline separated 2020-06-09 09:08:05 +02:00
Armin Briegel
dd925d329b fixed typo in Labels 2020-06-05 16:37:16 +02:00
Armin Briegel
f379ee0eda added wwdcformac 2020-06-05 16:32:02 +02:00
Armin Briegel
e5b5dcc49e updated Labels.txt 2020-06-03 13:58:46 +02:00
Armin Briegel
adecf05f45 added firefoxpkg and firefoxesrpkg 2020-06-03 13:55:43 +02:00
Armin Briegel
3e899ae79c fixed git download url generation 2020-06-02 15:19:13 +02:00
Armin Briegel
0ebf001246 added credits 2020-06-02 12:35:04 +02:00
Armin Briegel
b12bd875cd added variable NOTIFY which controls notifications, closes #24 2020-05-29 16:59:25 +02:00
Armin Briegel
ce88a3f3c8 added powershell and powershell-lts, closes #4 2020-05-29 16:50:01 +02:00
Armin Briegel
594db2c6d6 changed search pattern in installPKGinzip to *.pkg 2020-05-29 16:18:34 +02:00
Armin Briegel
778b4697fc add missing ;; 2020-05-29 16:12:45 +02:00
Armin Briegel
e4e9aec0e2 renamed ScriptLogging to printlog, will only write to log file when running as root 2020-05-29 16:04:11 +02:00
Mischa van der Bent
8c7c58b7d9 Update Notification version error
added a sleep of 10 sec so Spotlight know about the application that jsut installed
2020-05-29 15:25:16 +02:00
Mischa van der Bent
a881df4e81 Merge remote-tracking branch 'upstream/dev' into dev 2020-05-28 09:42:28 +02:00
kenchan0130
fb69bf94f6 Fix downloadurl of umbrella-roaming-client 2020-05-26 11:32:27 +09:00
Armin Briegel
27ef27851b fixed eraseinstall download url 2020-05-25 14:52:41 +02:00
Armin Briegel
9577f3c411 added docker and brave, closes #12 2020-05-22 09:10:31 +02:00
Armin Briegel
1b999a1101 fixed goodgledrivefilestream 2020-05-22 09:02:16 +02:00
Armin Briegel
b67aa97f0d changed Webex Teams from pkgInDmg to dmg 2020-05-22 08:47:51 +02:00
Mischa van der Bent
b7ff0d296f add start end log
add start end log
2020-05-20 11:02:02 +02:00
Mischa van der Bent
ea9a97eb7a Revert "Revert "Update Installomator.sh""
This reverts commit 2203f89612.
2020-05-20 11:00:26 +02:00
Mischa van der Bent
2203f89612 Revert "Update Installomator.sh"
This reverts commit 12a87e718d.
2020-05-20 11:00:02 +02:00
Mischa van der Bent
12a87e718d Update Installomator.sh
added Logging function and change the echo to function
2020-05-20 10:53:44 +02:00
Sebastien Jeanquier
6dea2ac136 Add Brave Label
Add Brave Label
2020-05-20 09:47:52 +02:00
Sebastien Jeanquier
9744b6683a Add Docker & Brave
Add Docker & Brave in (I think) the correct part of the script
2020-05-20 09:43:57 +02:00
Sebastien Jeanquier
84510540c3 Add Docker
Add Docker
2020-05-20 09:12:11 +02:00
Sebastien Jeanquier
73d386bfff Add Docker Desktop
Add Docker Desktop
2020-05-20 07:20:13 +02:00
Armin Briegel
4472a9a2f3 added signal 2020-05-19 13:09:25 +02:00
Armin Briegel
4f396b5191 added adobereaderdc 2020-05-18 20:38:04 +02:00
Armin Briegel
27e54e8ddd fixed a stray typo 2020-05-18 11:45:42 +02:00
Armin Briegel
d45af38636 re-added postman, jamfpppcutility, jamfmigrator, jamfreenroller 2020-05-18 11:43:35 +02:00
Armin Briegel
e99af92196 removed roamingclient and powershell after testing failed 2020-05-18 11:38:21 +02:00
Armin Briegel
b32e143dc0 installPkgInZip now finds first pkg in zip 2020-05-18 10:56:58 +02:00
Armin Briegel
6a55709f40 installPkgInDmg now searches for the first pkg in the dmg 2020-05-18 10:44:42 +02:00
Armin Briegel
a6344a490f fixed executable bit again 2020-05-18 10:08:53 +02:00
Armin Briegel
5a62c45b4c Merge branch 'Karabiner-Elements' of https://github.com/kenchan0130/Installomator into onishi-prs 2020-05-18 10:07:27 +02:00
Armin Briegel
432e8e83e2 updated credits 2020-05-18 10:02:21 +02:00
kenchan0130
14a397d23b Add a label of Karabiner-Elements 2020-05-18 16:55:50 +09:00
Armin Briegel
2b63d6939a Merge branch 'appcleaner' of https://github.com/kenchan0130/Installomator into onishi-prs 2020-05-18 09:48:28 +02:00
Armin Briegel
9de120d734 Merge branch 'umbrella-roaming-client' of https://github.com/kenchan0130/Installomator into onishi-prs 2020-05-18 09:41:12 +02:00
Armin Briegel
f34a144397 Merge branch 'google-japansese-input' of https://github.com/kenchan0130/Installomator into onishi-prs 2020-05-18 09:37:14 +02:00
Armin Briegel
06c42906e0 Merge branch 'santa' of https://github.com/kenchan0130/Installomator into onishi-prs 2020-05-18 09:34:07 +02:00
Armin Briegel
260ad87d4c added credit 2020-05-18 09:33:29 +02:00
Armin Briegel
e0a27139ad Merge branch 'powershell' of https://github.com/kenchan0130/Installomator into onishi-prs 2020-05-18 09:31:30 +02:00
kenchan0130
e5a1956258 Add a label of AppCleaner 2020-05-18 16:05:33 +09:00
kenchan0130
f01a50e428 Add a label of Roaming Client 2020-05-18 15:39:14 +09:00
kenchan0130
b6c18b4438 Add a label of Google Japanese Input 2020-05-18 15:18:58 +09:00
kenchan0130
65c33585fa Add a label of santa 2020-05-18 14:59:05 +09:00
kenchan0130
8ec2179c1c Add a label of powershell 2020-05-18 13:53:29 +09:00
Armin Briegel
eaf7bc2a01 Merge branch 'mischavdbent-Add-Apps' into dev 2020-05-15 15:30:49 +02:00
Armin Briegel
b02ca4781a Merge branch 'Add-Apps' of https://github.com/mischavdbent/Installomator into mischavdbent-Add-Apps 2020-05-15 15:28:18 +02:00
Armin Briegel
6f0029da7a updated version 2020-05-15 15:27:50 +02:00
Armin Briegel
6c3466e983 few more fixes in ReadMe 2020-05-15 15:26:51 +02:00
kenchan0130
9dcacfae91 apply fmt 2020-05-15 19:54:27 +09:00
Mischa van der Bent
26487d0d35 added apps labels
icons, autodmg, postman, sublimetext, jamfpppcutility, jamfmigrator, jamfreenroller
2020-05-15 12:40:32 +02:00
Mischa van der Bent
563c2e7447 Added Apps
icons, autodmg, postman, sublimetext, jamfpppcutility, jamfmigrator, jamfreenroller
2020-05-15 12:40:19 +02:00
Armin Briegel
5c0cb98f69 added FAQ to ReadMe 2020-05-15 11:51:44 +02:00
Armin Briegel
7420292fec added some missing labels 2020-05-14 15:58:13 +02:00
Armin Briegel
552e5cc730 updated ReadMe and License 2020-05-14 11:48:50 +02:00
4 changed files with 476 additions and 132 deletions

View File

@@ -8,8 +8,8 @@
# inspired by the download scripts from William Smith and Sander Schram
# with additional ideas and contribution from Isaac Ordonez, Mann consulting
VERSION='0.1'
VERSIONDATE='20200512'
VERSION='0.2'
VERSIONDATE='20200529'
export PATH=/usr/bin:/bin:/usr/sbin:/sbin
@@ -20,6 +20,14 @@ export PATH=/usr/bin:/bin:/usr/sbin:/sbin
# also no actual installation will be performed
DEBUG=1
# notify behavior
NOTIFY=success
# options:
# - success notify the user on success
# - silent no notifications
# behavior when blocking processes are found
BLOCKING_PROCESS_ACTION=prompt_user
# options:
@@ -30,7 +38,6 @@ BLOCKING_PROCESS_ACTION=prompt_user
# - kill kill process without prompting or giving the user a chance to save
# Each workflow label needs to be listed in the case statement below.
# for each label these variables can be set:
#
@@ -45,7 +52,7 @@ BLOCKING_PROCESS_ACTION=prompt_user
# - zip
# - pkgInDmg
# - pkgInZip (not yet tested)
#
#
# - downloadURL: (required)
# URL to download the dmg.
# Can be generated with a series of commands (see BBEdit for an example).
@@ -61,7 +68,7 @@ BLOCKING_PROCESS_ACTION=prompt_user
# spctl -a -vv -t install ~/Downloads/desktoppr-0.2.pkg
#
# The team ID is the ten-digit ID at the end of the line starting with 'origin='
#
#
# - archiveName: (optional)
# The name of the downloaded file.
# When not given the archiveName is derived from the $name.
@@ -74,7 +81,7 @@ BLOCKING_PROCESS_ACTION=prompt_user
# dmg or zip:
# Applications will be copied to this directory.
# Default value is '/Applications' for dmg and zip installations.
# pkg:
# pkg:
# targetDir is used as the install-location. Default is '/'.
#
# - blockingProcesses: (optional)
@@ -85,7 +92,7 @@ BLOCKING_PROCESS_ACTION=prompt_user
# blockingProcesses=( "Keynote" "Pages" "Numbers" )
# When a workflow has no blocking processes, use
# blockingProcesses=( NONE )
#
#
# - pkgName: (optional, only used for dmgInPkg and dmgInZip)
# File name of the pkg file _inside_ the dmg or zip
# When not given the pkgName is derived from the $name
@@ -106,46 +113,72 @@ BLOCKING_PROCESS_ACTION=prompt_user
# todos:
# TODO: better logging (or, really, any logging other than echo)
# TODO: ?blockingProcesses for SharePointPlugin
# TODO: generic function Sparkle to get latest download
# TODO: ?notify user of errors
# TODO: ?generic function to initiate a SparkleProcess
# TODO: ?generic function to initiate a Sparkle Update
# TODO: better version retrieval and reporting, before and after install
# functions to help with getting info
# Logging
log_location="/private/var/log/Installomator.log"
printlog(){
timestamp=$(date +%F\ %T)
if [[ "$(whoami)" == "root" ]]; then
echo "$timestamp" "$1" | tee -a $log_location
else
echo "$timestamp" "$1"
fi
}
# will get the latest release download from a github repo
downloadURLFromGit() { # $1 git user name, $2 git repo name
gitusername=${1?:"no git user name"}
gitreponame=${2?:"no git repo name"}
if [[ $type == "pkgInDmg" ]]; then
filetype="dmg"
elif [[ $type == "pkgInZip" ]]; then
filetype="zip"
else
filetype=$type
fi
if [ -n "$archiveName" ]; then
downloadURL=$(curl --silent --fail "https://api.github.com/repos/$gitusername/$gitreponame/releases/latest" \
| awk -F '"' "/browser_download_url/ && /$archiveName/ { print \$4 }")
else
downloadURL=$(curl --silent --fail "https://api.github.com/repos/$gitusername/$gitreponame/releases/latest" \
| awk -F '"' "/browser_download_url/ && /$type/ { print \$4 }")
| awk -F '"' "/browser_download_url/ && /$filetype/ { print \$4 }")
fi
if [ -z "$downloadURL" ]; then
cleanupAndExit 9 "could not retrieve download URL for $gitusername/$gitreponame"
echo "could not retrieve download URL for $gitusername/$gitreponame"
exit 9
else
echo "$downloadURL"
return 0
fi
}
printlog "################## Start Installomator"
# get the label
if [[ $# -eq 0 ]]; then
echo "no label provided"
printlog "no label provided"
exit 1
elif [[ $# -gt 3 ]]; then
# jamf uses $4 for the first custom parameter
echo "shifting arguments for Jamf"
printlog "shifting arguments for Jamf"
shift 3
fi
label=${1:?"no label provided"}
printlog "################## $label"
# lowercase the label
label=${label:l}
@@ -158,16 +191,23 @@ currentUser=$(scutil <<< "show State:/Users/ConsoleUser" | awk '/Name :/ { print
case $label in
version)
# print the script VERSION
echo "$VERSION"
printlog "$VERSION"
exit 0
;;
longversion)
# print the script version
echo "Installomater: version $VERSION ($VERSIONDATE)"
printlog "Installomater: version $VERSION ($VERSIONDATE)"
exit 0
;;
# label descriptions start here
autodmg)
# credit: Mischa van der Bent (@mischavdbent)
name="AutoDMG"
type="dmg"
downloadURL=$(downloadURLFromGit MagerValp AutoDMG)
expectedTeamID="5KQ3D3FG5H"
;;
googlechrome)
name="Google Chrome"
type="dmg"
@@ -183,6 +223,21 @@ case $label in
updateToolArguments=( -runMode oneshot -userInitiated YES )
updateToolRunAsCurrentUser=1
;;
googlejapaneseinput)
# credit: Tadayuki Onishi (@kenchan0130)
name="GoogleJapaneseInput"
type="pkgInDmg"
pkgName="GoogleJapaneseInput.pkg"
downloadURL="https://dl.google.com/japanese-ime/latest/GoogleJapaneseInput.dmg"
expectedTeamID="EQHXZ8M8AV"
;;
santa)
# credit: Tadayuki Onishi (@kenchan0130)
name="Santa"
type="pkgInDmg"
downloadURL=$(downloadURLFromGit google santa)
expectedTeamID="EQHXZ8M8AV"
;;
spotify)
name="Spotify"
type="dmg"
@@ -198,7 +253,21 @@ case $label in
firefox)
name="Firefox"
type="dmg"
downloadURL="https://download.mozilla.org/?product=firefox-latest&amp;os=osx&amp;lang=en-US"
downloadURL="https://download.mozilla.org/?product=firefox-latest&os=osx&lang=en-US"
expectedTeamID="43AQ936H96"
blockingProcesses=( firefox )
;;
firefoxpkg)
name="Firefox"
type="pkg"
downloadURL="https://download.mozilla.org/?product=firefox-pkg-latest-ssl&os=osx&lang=en-US"
expectedTeamID="43AQ936H96"
blockingProcesses=( firefox )
;;
firefoxesrpkg)
name="Firefox"
type="pkg"
downloadURL="https://download.mozilla.org/?product=firefox-esr-pkg-latest-ssl&os=osx"
expectedTeamID="43AQ936H96"
blockingProcesses=( firefox )
;;
@@ -222,7 +291,7 @@ case $label in
expectedTeamID="GVZRY6KDKR"
;;
suspiciouspackage)
# credit: Mischa van der Bent
# credit: Mischa van der Bent (@mischavdbent)
name="Suspicious Package"
type="dmg"
downloadURL="https://mothersruin.com/software/downloads/SuspiciousPackage.dmg"
@@ -238,7 +307,7 @@ case $label in
eraseinstall)
name="EraseInstall"
type="pkg"
downloadURL=$(curl -fs https://bitbucket.org/prowarehouse-nl/erase-install/downloads/ | grep pkg | cut -d'"' -f2 | head -n 1)
downloadURL=https://bitbucket.org$(curl -fs https://bitbucket.org/prowarehouse-nl/erase-install/downloads/ | grep pkg | cut -d'"' -f2 | head -n 1)
expectedTeamID="R55HK5K86Y"
;;
omnigraffle7)
@@ -289,44 +358,44 @@ case $label in
| cut -d '"' -f 2 )
expectedTeamID="UPXU4CQZ5P"
;;
boxdrive)
# credit: Isaac Ordonez, Mann consulting
boxdrive)
# credit: Isaac Ordonez, Mann consulting (@mannconsulting)
name="Box"
type="pkg"
downloadURL="https://e3.boxcdn.net/box-installers/desktop/releases/mac/Box.pkg"
expectedTeamID="M683GB7CPW"
;;
aviatrix)
# credit: Isaac Ordonez, Mann consulting
# credit: Isaac Ordonez, Mann consulting (@mannconsulting)
name="Aviatrix VPN Client"
type="pkg"
downloadURL="https://s3-us-west-2.amazonaws.com/aviatrix-download/AviatrixVPNClient/AVPNC_mac.pkg"
expectedTeamID="32953Z7NBN"
;;
zoom)
# credit: Isaac Ordonez, Mann consulting
# credit: Isaac Ordonez, Mann consulting (@mannconsulting)
name="Zoom.us"
type="pkg"
downloadURL="https://zoom.us/client/latest/ZoomInstallerIT.pkg"
expectedTeamID="BJ4HAAB9B3"
expectedTeamID="BJ4HAAB9B3"
blockingProcesses=( zoom.us )
;;
sonos)
# credit: Erik Stam
# credit: Erik Stam (@erikstam)
name="Sonos"
type="dmg"
downloadURL="https://www.sonos.com/redir/controller_software_mac"
expectedTeamID="2G4LW83Q3E"
;;
coderunner)
# credit: Erik Stam
# credit: Erik Stam (@erikstam)
name="CodeRunner"
type="zip"
downloadURL="https://coderunnerapp.com/download"
expectedTeamID="R4GD98AJF9"
;;
openvpnconnect)
# credit: Erik Stam
# credit: Erik Stam (@erikstam)
name="OpenVPN"
type="pkgInDmg"
pkgName="OpenVPN_Connect_Installer_signed.pkg"
@@ -346,38 +415,46 @@ case $label in
expectedTeamID="2BUA8C4S2C"
;;
webexmeetings)
# credit: Erik Stam
# credit: Erik Stam (@erikstam)
name="Cisco Webex Meetings"
type="pkgInDmg"
downloadURL="https://akamaicdn.webex.com/client/webexapp.dmg"
expectedTeamID="DE8Y96K9QP"
;;
webexteams)
# credit: Erik Stam
# credit: Erik Stam (@erikstam)
name="Webex Teams"
type="pkgInDmg"
type="dmg"
downloadURL="https://binaries.webex.com/WebexTeamsDesktop-MACOS-Gold/WebexTeams.dmg"
expectedTeamID="DE8Y96K9QP"
;;
citrixworkspace)
# credit: Erik Stam
name="Citrix Workspace"
type="pkgInDmg"
downloadURL="https://downloads.citrix.com/17596/CitrixWorkspaceApp.dmg?__gda__=1588183500_fc68033aef7d6d163d8b8309b964f1de"
expectedTeamID="S272Y5R93J"
;;
#citrixworkspace)
# credit: Erik Stam (@erikstam)
#name="Citrix Workspace"
#type="pkgInDmg"
#downloadURL="https://downloads.citrix.com/17596/CitrixWorkspaceApp.dmg?__gda__=1588183500_fc68033aef7d6d163d8b8309b964f1de"
#expectedTeamID="S272Y5R93J"
#;;
privileges)
# credit: Erik Stam
# credit: Erik Stam (@erikstam)
name="Privileges"
type="zip"
downloadURL=$(downloadURLFromGit sap macOS-enterprise-privileges )
expectedTeamID="7R5ZEU67FQ"
;;
icons)
# credit: Mischa van der Bent (@mischavdbent)
name="Icons"
type="zip"
downloadURL=$(downloadURLFromGit sap macOS-icon-generator )
expectedTeamID="7R5ZEU67FQ"
;;
googledrivefilestream)
# credit: Isaac Ordonez, Mann consulting
# credit: Isaac Ordonez, Mann consulting (@mannconsulting)
name="Google Drive File Stream"
type="pkgInDmg"
downloadURL="https://dl.google.com/drive-file-stream/GoogleDriveFileStream.dmg"
pkgName="GoogleDriveFileStream.pkg"
expectedTeamID="EQHXZ8M8AV"
;;
plisteditpro)
@@ -385,13 +462,20 @@ case $label in
type="zip"
downloadURL="https://www.fatcatsoftware.com/plisteditpro/PlistEditPro.zip"
expectedTeamID="8NQ43ND65V"
;;
;;
slack)
name="Slack"
type="dmg"
downloadURL="https://slack.com/ssb/download-osx"
expectedTeamID="BQR82RBBHL"
;;
sublimetext)
# credit: Mischa van der Bent (@mischavdbent)
name="Sublime Text"
type="dmg"
downloadURL="https://download.sublimetext.com/latest/stable/osx"
expectedTeamID="Z6D26JE4Y4"
;;
githubdesktop)
name="GitHub Desktop"
type="zip"
@@ -467,7 +551,135 @@ case $label in
downloadURL=$(curl -fs https://royaltsx-v4.royalapps.com/updates_stable | xpath '//rss/channel/item[1]/enclosure/@url' 2>/dev/null | cut -d '"' -f 2)
expectedTeamID="VXP8K9EDP6"
;;
appcleaner)
# credit: Tadayuki Onishi (@kenchan0130)
name="AppCleaner"
type="zip"
downloadURL=$(curl -fs https://freemacsoft.net/appcleaner/Updates.xml | xpath '//rss/channel/*/enclosure/@url' 2>/dev/null | tr " " "\n" | sort | tail -1 | cut -d '"' -f 2)
expectedTeamID="X85ZX835W9"
;;
karabinerelements)
# credit: Tadayuki Onishi (@kenchan0130)
name="Karabiner-Elements"
type="pkgInDmg"
downloadURL=$(downloadURLFromGit pqrs-org Karabiner-Elements)
expectedTeamID="G43BCU2T37"
;;
postman)
# credit: Mischa van der Bent
name="Postman"
type="zip"
downloadURL="https://dl.pstmn.io/download/latest/osx"
expectedTeamID="H7H8Q7M5CK"
;;
jamfpppcutility)
# credit: Mischa van der Bent
name="PPPC Utility"
type="zip"
downloadURL=$(downloadURLFromGit jamf PPPC-Utility)
expectedTeamID="483DWKW443"
;;
jamfmigrator)
# credit: Mischa van der Bent
name="jamf-migrator"
type="zip"
downloadURL=$(downloadURLFromGit jamf JamfMigrator)
expectedTeamID="PS2F6S478M"
;;
jamfreenroller)
# credit: Mischa van der Bent
name="ReEnroller"
type="zip"
downloadURL=$(downloadURLFromGit jamf ReEnroller)
expectedTeamID="PS2F6S478M"
;;
adobereaderdc)
name="Adobe Acrobat Reader DC"
type="pkgInDmg"
downloadURL=$(adobecurrent=`curl -s https://armmf.adobe.com/arm-manifests/mac/AcrobatDC/reader/current_version.txt | tr -d '.'` && echo http://ardownload.adobe.com/pub/adobe/reader/mac/AcrobatDC/"$adobecurrent"/AcroRdrDC_"$adobecurrent"_MUI.dmg)
expectedTeamID="JQ525L2MZD"
blockingProcesses=( "AdobeReader" )
;;
signal)
# credit: Søren Theilgaard
name="Signal"
type="dmg"
downloadURL=https://updates.signal.org/desktop/$(curl -fs https://updates.signal.org/desktop/latest-mac.yml | awk '/url/ && /dmg/ {print $3}')
expectedTeamID="U68MSDN6DR"
;;
docker)
# credit: @securitygeneration
name="Docker"
type="dmg"
downloadURL="https://download.docker.com/mac/stable/Docker.dmg"
expectedTeamID="9BNSXJN65R"
;;
brave)
# credit: @securitygeneration
name="Brave Browser"
type="dmg"
downloadURL="https://laptop-updates.brave.com/latest/osx"
expectedTeamID="9BNSXJN65R"
;;
umbrellaroamingclient)
# credit: Tadayuki Onishi (@kenchan0130)
name="Umbrella Roaming Client"
type="pkgInZip"
downloadURL=https://disthost.umbrella.com/roaming/upgrade/mac/production/$( curl -fsL https://disthost.umbrella.com/roaming/upgrade/mac/production/manifest.json | awk -F '"' '/"downloadFilename"/ { print $4 }' )
expectedTeamID="7P7HQ8H646"
;;
powershell)
# credit: Tadayuki Onishi (@kenchan0130)
name="PowerShell"
type="pkg"
downloadURL=$(curl -fs "https://api.github.com/repos/Powershell/Powershell/releases/latest" \
| awk -F '"' '/browser_download_url/ && /pkg/ { print $4 }' | grep -v lts )
expectedTeamID="UBF8T346G9"
;;
powershell-lts)
# credit: Tadayuki Onishi (@kenchan0130)
name="PowerShell"
type="pkg"
downloadURL=$(curl -fs "https://api.github.com/repos/Powershell/Powershell/releases/latest" \
| awk -F '"' '/browser_download_url/ && /pkg/ { print $4 }' | grep lts)
expectedTeamID="UBF8T346G9"
;;
wwdcformac)
name="WWDC"
type="zip"
downloadURL="https://cdn.wwdc.io/WWDC_latest.zip"
expectedTeamID="8C7439RJLG"
;;
ringcentralmeetings)
# credit: Isaac Ordonez, Mann consulting (@mannconsulting)
name="Ring Central Meetings"
type="pkg"
downloadURL="http://dn.ringcentral.com/data/web/download/RCMeetings/1210/RCMeetingsClientSetup.pkg"
expectedTeamID="M932RC5J66"
blockingProcesses=( "RingCentral Meetings" )
;;
ringcentralapp)
# credit: Isaac Ordonez, Mann consulting (@mannconsulting)
name="Glip"
type="dmg"
downloadURL="https://downloads.ringcentral.com/glip/rc/GlipForMac"
expectedTeamID="M932RC5J66"
blockingProcesses=( "Glip" )
;;
sfsymbols)
name="SF Symbols"
type="pkgInDmg"
downloadURL="https://developer.apple.com/design/downloads/SF-Symbols.dmg"
expectedTeamID="Software Update"
;;
swiftruntimeforcommandlinetools)
# Note: this installer will error on macOS versions later than 10.14.3
name="SwiftRuntimeForCommandLineTools"
type="pkgInDmg"
downloadURL="https://updates.cdn-apple.com/2019/cert/061-41823-20191025-5efc5a59-d7dc-46d3-9096-396bb8cb4a73/SwiftRuntimeForCommandLineTools.dmg"
expectedTeamID="Software Update"
;;
# Note: Packages is signed but _not_ notarized, so spctl will reject it
# packages)
@@ -482,7 +694,7 @@ case $label in
# https://docs.microsoft.com/en-us/deployoffice/mac/update-office-for-mac-using-msupdate
# download link IDs from: https://macadmin.software
microsoftoffice365)
name="MicrosoftOffice365"
type="pkg"
@@ -493,7 +705,7 @@ case $label in
blockingProcesses=( "Microsoft AutoUpdate" "Microsoft Word" "Microsoft PowerPoint" "Microsoft Excel" "Microsoft OneNote" "Microsoft Outlook" "OneDrive" )
updateTool="/Library/Application Support/Microsoft/MAU2.0/Microsoft AutoUpdate.app/Contents/MacOS/msupdate"
updateToolArguments=( --install )
;;
;;
microsoftofficebusinesspro)
name="MicrosoftOfficeBusinessPro"
type="pkg"
@@ -504,7 +716,7 @@ case $label in
blockingProcesses=( "Microsoft AutoUpdate" "Microsoft Word" "Microsoft PowerPoint" "Microsoft Excel" "Microsoft OneNote" "Microsoft Outlook" "OneDrive" "Teams")
updateTool="/Library/Application Support/Microsoft/MAU2.0/Microsoft AutoUpdate.app/Contents/MacOS/msupdate"
updateToolArguments=( --install )
;;
;;
microsoftedgeconsumerstable)
name="Microsoft Edge"
type="pkg"
@@ -513,7 +725,7 @@ case $label in
updateTool="/Library/Application Support/Microsoft/MAU2.0/Microsoft AutoUpdate.app/Contents/MacOS/msupdate"
updateToolArguments=( --install --apps EDGE01 )
;;
microsoftcompanyportal)
microsoftcompanyportal)
name="Company Portal"
type="pkg"
downloadURL="https://go.microsoft.com/fwlink/?linkid=869655"
@@ -521,7 +733,7 @@ case $label in
updateTool="/Library/Application Support/Microsoft/MAU2.0/Microsoft AutoUpdate.app/Contents/MacOS/msupdate"
updateToolArguments=( --install --apps IMCP01 )
;;
microsoftskypeforbusiness)
microsoftskypeforbusiness)
name="Skype for Business"
type="pkg"
downloadURL="https://go.microsoft.com/fwlink/?linkid=832978"
@@ -529,7 +741,7 @@ case $label in
updateTool="/Library/Application Support/Microsoft/MAU2.0/Microsoft AutoUpdate.app/Contents/MacOS/msupdate"
updateToolArguments=( --install --apps MSFB16 )
;;
microsoftremotedesktop)
microsoftremotedesktop)
name="Microsoft Remote Desktop"
type="pkg"
downloadURL="https://go.microsoft.com/fwlink/?linkid=868963"
@@ -537,7 +749,7 @@ case $label in
updateTool="/Library/Application Support/Microsoft/MAU2.0/Microsoft AutoUpdate.app/Contents/MacOS/msupdate"
updateToolArguments=( --install --apps MSRD10 )
;;
microsoftteams)
microsoftteams)
name="Microsoft Teams"
type="pkg"
downloadURL="https://go.microsoft.com/fwlink/?linkid=869428"
@@ -637,7 +849,6 @@ case $label in
;;
# these descriptions exist for testing and are intentionally broken
brokendownloadurl)
name="Google Chrome"
@@ -659,7 +870,7 @@ case $label in
;;
*)
# unknown label
echo "unknown label $label"
printlog "unknown label $label"
exit 1
;;
esac
@@ -667,23 +878,24 @@ esac
# functions
cleanupAndExit() { # $1 = exit code, $2 message
if [[ -n $2 && $1 -ne 0 ]]; then
echo "ERROR: $2"
printlog "ERROR: $2"
fi
if [ "$DEBUG" -eq 0 ]; then
# remove the temporary working directory when done
echo "Deleting $tmpDir"
printlog "Deleting $tmpDir"
rm -Rf "$tmpDir"
fi
if [ -n "$dmgmount" ]; then
# unmount disk image
echo "Unmounting $dmgmount"
printlog "Unmounting $dmgmount"
hdiutil detach "$dmgmount"
fi
printlog "################## End Installomator \n\n"
exit "$1"
}
runAsUser() {
runAsUser() {
if [[ $currentUser != "loginwindow" ]]; then
uid=$(id -u "$currentUser")
launchctl asuser $uid sudo -u $currentUser "$@"
@@ -699,7 +911,7 @@ displaynotification() { # $1: message $2: title
message=${1:-"Message"}
title=${2:-"Notification"}
manageaction="/Library/Application Support/JAMF/bin/Management Action.app/Contents/MacOS/Management Action"
if [[ -x "$manageaction" ]]; then
"$manageaction" -message "$message" -title "$title"
else
@@ -718,32 +930,32 @@ getAppVersion() {
if [[ ${#filteredAppPaths} -eq 1 ]]; then
installedAppPath=$filteredAppPaths[1]
appversion=$(mdls -name kMDItemVersion -raw $installedAppPath )
echo "found app at $installedAppPath, version $appversion"
printlog "found app at $installedAppPath, version $appversion"
else
echo "could not determine location of $appName"
printlog "could not determine location of $appName"
fi
else
echo "could not find $appName"
printlog "could not find $appName"
fi
}
checkRunningProcesses() {
# don't check in DEBUG mode
if [[ $DEBUG -ne 0 ]]; then
echo "DEBUG mode, not checking for blocking processes"
printlog "DEBUG mode, not checking for blocking processes"
return
fi
# try at most 3 times
for i in {1..3}; do
countedProcesses=0
for x in ${blockingProcesses}; do
if pgrep -xq "$x"; then
echo "found blocking process $x"
case $BLOCKING_PROCESS_ACTION in
kill)
echo "killing process $x"
printlog "killing process $x"
pkill $x
;;
prompt_user)
@@ -758,7 +970,7 @@ checkRunningProcesses() {
cleanupAndExit 12 "blocking process '$x' found, aborting"
;;
esac
countedProcesses=$((countedProcesses + 1))
fi
done
@@ -768,7 +980,7 @@ checkRunningProcesses() {
break
else
# give the user a bit of time to quit apps
echo "waiting 30 seconds for processes to quit"
printlog "waiting 30 seconds for processes to quit"
sleep 30
fi
done
@@ -777,24 +989,24 @@ checkRunningProcesses() {
cleanupAndExit 11 "could not quit all processes, aborting..."
fi
echo "no more blocking processes, continue with update"
printlog "no more blocking processes, continue with update"
}
installAppWithPath() { # $1: path to app to install in $targetDir
appPath=${1?:"no path to app"}
# check if app exists
if [ ! -e "$appPath" ]; then
cleanupAndExit 8 "could not find: $appPath"
fi
# verify with spctl
echo "Verifying: $appPath"
printlog "Verifying: $appPath"
if ! teamID=$(spctl -a -vv "$appPath" 2>&1 | awk '/origin=/ {print $NF }' | tr -d '()' ); then
cleanupAndExit 4 "Error verifying $appPath"
fi
echo "Team ID: $teamID (expected: $expectedTeamID )"
printlog "Team ID: $teamID (expected: $expectedTeamID )"
if [ "$expectedTeamID" != "$teamID" ]; then
cleanupAndExit 5 "Team IDs do not match"
@@ -806,19 +1018,19 @@ installAppWithPath() { # $1: path to app to install in $targetDir
if [ "$DEBUG" -eq 0 ]; then
cleanupAndExit 6 "not running as root, exiting"
fi
echo "DEBUG enabled, skipping copy and chown steps"
return 0
fi
# remove existing application
if [ -e "$targetDir/$appName" ]; then
echo "Removing existing $targetDir/$appName"
printlog "Removing existing $targetDir/$appName"
rm -Rf "$targetDir/$appName"
fi
# copy app to /Applications
echo "Copy $appPath to $targetDir"
printlog "Copy $appPath to $targetDir"
if ! ditto "$appPath" "$targetDir/$appName"; then
cleanupAndExit 7 "Error while copying"
fi
@@ -827,76 +1039,85 @@ installAppWithPath() { # $1: path to app to install in $targetDir
# set ownership to current user
if [ "$currentUser" != "loginwindow" ]; then
echo "Changing owner to $currentUser"
chown -R "$currentUser" "$targetDir/$appName"
chown -R "$currentUser" "$targetDir/$appName"
else
echo "No user logged in, not changing user"
printlog "No user logged in, not changing user"
fi
}
mountDMG() {
# mount the dmg
echo "Mounting $tmpDir/$archiveName"
printlog "Mounting $tmpDir/$archiveName"
# always pipe 'Y\n' in case the dmg requires an agreement
if ! dmgmount=$(echo 'Y'$'\n' | hdiutil attach "$tmpDir/$archiveName" -nobrowse -readonly | tail -n 1 | cut -c 54- ); then
if ! dmgmount=$(printlog 'Y'$'\n' | hdiutil attach "$tmpDir/$archiveName" -nobrowse -readonly | tail -n 1 | cut -c 54- ); then
cleanupAndExit 3 "Error mounting $tmpDir/$archiveName"
fi
if [[ ! -e $dmgmount ]]; then
echo "Error mounting $tmpDir/$archiveName"
printlog "Error mounting $tmpDir/$archiveName"
cleanupAndExit 3
fi
echo "Mounted: $dmgmount"
}
installFromDMG() {
mountDMG
installAppWithPath "$dmgmount/$appName"
}
installFromPKG() {
# verify with spctl
echo "Verifying: $archiveName"
if ! teamID=$(spctl -a -vv -t install "$archiveName" 2>&1 | awk '/origin=/ {print $NF }' | tr -d '()' ); then
echo "Error verifying $archiveName"
printlog "Verifying: $archiveName"
if ! spctlout=$(spctl -a -vv -t install "$archiveName" 2>&1 ); then
printlog "Error verifying $archiveName"
cleanupAndExit 4
fi
teamID=$(echo $spctlout | awk -F '(' '/origin=/ {print $2 }' | tr -d '()' )
echo $teamID
# Apple signed software has no teamID, grab entire origin instead
if [[ -z $teamID ]]; then
teamID=$(echo $spctlout | awk -F '=' '/origin=/ {print $NF }')
fi
echo "Team ID: $teamID (expected: $expectedTeamID )"
printlog "Team ID: $teamID (expected: $expectedTeamID )"
if [ "$expectedTeamID" != "$teamID" ]; then
echo "Team IDs do not match!"
printlog "Team IDs do not match!"
cleanupAndExit 5
fi
# skip install for DEBUG
if [ "$DEBUG" -ne 0 ]; then
echo "DEBUG enabled, skipping installation"
printlog "DEBUG enabled, skipping installation"
return 0
fi
# check for root
if [ "$(whoami)" != "root" ]; then
# not running as root
echo "not running as root, exiting"
cleanupAndExit 6
cleanupAndExit 6
fi
# install pkg
echo "Installing $archiveName to $targetDir"
printlog "Installing $archiveName to $targetDir"
if ! installer -pkg "$archiveName" -tgt "$targetDir" ; then
echo "error installing $archiveName"
printlog "error installing $archiveName"
cleanupAndExit 9
fi
}
installFromZIP() {
# unzip the archive
echo "Unzipping $archiveName"
printlog "Unzipping $archiveName"
tar -xf "$archiveName"
installAppWithPath "$tmpDir/$appName"
}
@@ -904,36 +1125,51 @@ installPkgInDmg() {
mountDMG
# locate pkg in dmg
if [[ -z $pkgName ]]; then
pkgName="$name.pkg"
# find first file ending with 'pkg'
findfiles=$(find "$dmgmount" -iname "*.pkg" -maxdepth 1 )
filearray=( ${(f)findfiles} )
if [[ ${#filearray} -eq 0 ]]; then
cleanupAndExit 20 "couldn't find pkg in dmg $archiveName"
fi
archiveName="${filearray[1]}"
printlog "found pkg: $archiveName"
else
# it is now safe to overwrite archiveName for installFromPKG
archiveName="$dmgmount/$pkgName"
fi
# it is now safe to overwrite archiveName for installFromPKG
archiveName="$dmgmount/$pkgName"
# installFromPkgs
installFromPKG
}
installPkgInZip() {
# unzip the archive
echo "Unzipping $archiveName"
printlog "Unzipping $archiveName"
tar -xf "$archiveName"
# locate pkg in zip
if [[ -z $pkgName ]]; then
pkgName="$name.pkg"
# find first file starting with $name and ending with 'pkg'
findfiles=$(find "$tmpDir" -iname "*.pkg" -maxdepth 1 )
filearray=( ${(f)findfiles} )
if [[ ${#filearray} -eq 0 ]]; then
cleanupAndExit 20 "couldn't find pkg in zip $archiveName"
fi
archiveName="${filearray[1]}"
# it is now safe to overwrite archiveName for installFromPKG
printlog "found pkg: $archiveName"
else
# it is now safe to overwrite archiveName for installFromPKG
archiveName="$tmpDir/$pkgName"
fi
# it is now safe to overwrite archiveName for installFromPKG
archiveName="$tmpDir/$pkgName"
# installFromPkgs
installFromPKG
}
runUpdateTool() {
if [[ -x $updateTool ]]; then
echo "running $updateTool $updateToolArguments"
printlog "running $updateTool $updateToolArguments"
if [[ -n $updateToolRunAsCurrentUser ]]; then
runAsUser $updateTool ${updateToolArguments}
else
@@ -943,7 +1179,7 @@ runUpdateTool() {
cleanupAndExit 15 "Error running $updateTool"
fi
else
echo "couldn't find $updateTool, continuing normally"
printlog "couldn't find $updateTool, continuing normally"
return 1
fi
return 0
@@ -968,7 +1204,7 @@ if [ -z "$archiveName" ]; then
archiveName="${name}.zip"
;;
*)
echo "Cannot handle type $type"
printlog "Cannot handle type $type"
cleanupAndExit 99
;;
esac
@@ -988,14 +1224,14 @@ if [ -z "$targetDir" ]; then
targetDir="/"
;;
*)
echo "Cannot handle type $type"
printlog "Cannot handle type $type"
cleanupAndExit 99
;;
esac
fi
if [[ -z $blockingProcesses ]]; then
echo "no blocking processes defined, using $name as default"
printlog "no blocking processes defined, using $name as default"
blockingProcesses=( $name )
fi
@@ -1009,9 +1245,9 @@ else
fi
# change directory to temporary working directory
echo "Changing directory to $tmpDir"
printlog "Changing directory to $tmpDir"
if ! cd "$tmpDir"; then
echo "error changing directory $tmpDir"
printlog "error changing directory $tmpDir"
#rm -Rf "$tmpDir"
cleanupAndExit 1
fi
@@ -1024,14 +1260,14 @@ if [[ -n $appVersion ]]; then
cleanupAndExit 0
fi # otherwise continue
else
echo "DEBUG mode enabled, not running update tool"
printlog "DEBUG mode enabled, not running update tool"
fi
fi
fi
# when user is logged in, and app is running, prompt user to quit app
if [[ $BLOCKING_PROCESS_ACTION == "ignore" ]]; then
echo "ignoring blocking processes"
printlog "ignoring blocking processes"
else
if [[ $currentUser != "loginwindow" ]]; then
if [[ ${#blockingProcesses} -gt 0 ]]; then
@@ -1045,12 +1281,12 @@ fi
# download the archive
if [ -f "$archiveName" ] && [ "$DEBUG" -ne 0 ]; then
echo "$archiveName exists and DEBUG enabled, skipping download"
printlog "$archiveName exists and DEBUG enabled, skipping download"
else
# download the dmg
echo "Downloading $downloadURL to $archiveName"
printlog "Downloading $downloadURL to $archiveName"
if ! curl --location --fail --silent "$downloadURL" -o "$archiveName"; then
echo "error downloading $downloadURL"
printlog "error downloading $downloadURL"
cleanupAndExit 2
fi
fi
@@ -1072,18 +1308,26 @@ case $type in
installPkgInZip
;;
*)
echo "Cannot handle type $type"
printlog "Cannot handle type $type"
cleanupAndExit 99
;;
esac
# print installed application location and version
sleep 10 # wait a moment to let spotlight catch up
getAppVersion
# TODO: notify when done
if [[ $currentUser != "loginwindow" ]]; then
echo "notifying"
displaynotification "Installed $name, version $appversion" "Installation complete!"
if [[ -z $appversion ]]; then
message="Installed $name"
else
message="Installed $name, version $appversion"
fi
printlog "$message"
if [[ $currentUser != "loginwindow" && $NOTIFY == "success" ]]; then
printlog "notifying"
displaynotification "$message" "Installation complete!"
fi
# all done!

View File

@@ -186,7 +186,7 @@
same "printed page" as the copyright notice for easier
identification within third-party archives.
Copyright 2018 Armin Briegel, Scripting OS X
Copyright 2020 Armin Briegel, Scripting OS X
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.

View File

@@ -1,25 +1,39 @@
1password7
adobereaderdc
appcleaner
atom
autodmg
aviatrix
bbedit
citrixworkspace
boxdrive
brave
coderunner
cyberduck
depnotify
desktoppr
discord
docker
dropbox
eraseinstall
firefox
firefoxesrpkg
firefoxpkg
githubdesktop
googlechrome
googlechromepkg
googledrivefilestream
googlejapaneseinput
grandperspective
handbrake
icons
iterm2
jamfmigrator
jamfpppcutility
jamfreenroller
karabinerelements
malwarebytes
microsoftautoupdate
microsoftcompanyportal
microsoftdefenderatp
microsoftedgeconsumerstable
microsoftedgeenterprisestable
@@ -30,7 +44,10 @@ microsoftonedrive
microsoftonenote
microsoftoutlook
microsoftpowerpoint
microsoftremotedesktop
microsoftsharepointplugin
microsoftskypeforbusiness
microsoftteams
microsoftword
netnewswire
omnifocus3
@@ -38,21 +55,33 @@ omnigraffle7
openvpnconnect
pacifist
plisteditpro
postman
powershell
powershell-lts
privileges
resiliosynchome
ringcentralapp
ringcentralmeetings
royaltsx
santa
sfsymbols
signal
slack
sonos
sourcetree
spotify
sublimetext
suspiciouspackage
swiftruntimeforcommandlinetools
teamviewer
textmate
things
tunnelbear
umbrellaroamingclient
visualstudiocode
vlc
webexmeetings
webexteams
whatsapp
wwdcformac
zoom

View File

@@ -4,6 +4,13 @@ _The one installer script to rule them all._
![](https://img.shields.io/github/v/release/scriptingosx/Installomator)&nbsp;![](https://img.shields.io/github/downloads/scriptingosx/Installomator/latest/total)&nbsp;![](https://img.shields.io/badge/macOS-10.14%2B-success)&nbsp;![](https://img.shields.io/github/license/scriptingosx/Installomator)
This script is in the "we find it useful, it is working for us" stage.
Your production and deployment environment will be different, please test thoroughly before rolling it out to your production.
I have put a lot of work into making it stable and safe, but I cannot - of course - make _any_ promises that it won't break in some not yet encountered edge case.
## Background
As a system engineer at [an Apple Authorized Enterprise Reseller](https://prowarehouse.nl), we manage a lot of Jamf instances.
@@ -12,23 +19,30 @@ Some of these instances are tightly managed, i.e. the versions of the operating
Installomator was _not_ written for these kinds of deployment.
If you are running this kind of deployment, you want [AutoPkg](https://github.com/autopkg/autopkg).
If you are running this kind of deployment, you want to use [AutoPkg](https://github.com/autopkg/autopkg) and you can stop reading here.
There are other kinds of deployments, though. In these deployments the management system is merely used to "get the user ready" as quickly as possible when they set up a new machine, and to offer software from a self service portal. In these deployments, system and software installations are 'latest version available' and updates are user driven (though we do want to nag them).
These are mostly 'user controlled' Macs and we mostly just want to assist the user in doing the right thing. And the right thing is (often) to install the latest versions and updates when they are available.
These deployments are
The Mac App Store and software pushed through the Mac App Store follow this approach. When you manage software through the App Store — whether it is on iOS or macOS — neither the MacAdmin nor the user get a choice of the application version. They will get the latest version.
- user driven
- low control
- minimal maintenance effort
In such deployments, keeping the installers hosted in your management system up to date is an extra burden that often feels unnecessary. Instead of downloading, re-packaging, uploading application installers to the management system, it is often easier to run a script which downloads the latest version directly from the vendor's servers and installs it.
These are mostly 'user controlled' Macs and we (the admins) just want to assist the user in doing the right thing. And the right thing is (often) to install the latest versions and updates when they are available.
The Mac App Store and software pushed through the Mac App Store follow this approach. When you manage and deploy software through the App Store — whether it is on iOS or macOS — neither the MacAdmin nor the user get a choice of the application version. They will get the latest version.
In such deployments, keeping the installers hosted in your management system up to date is an extra burden. AutoPkg can, well, automate much of the download/re-package/upload/stage cycle, but it still requires oversight and maintenance. Instead of downloading, re-packaging, uploading application installers to the management system, it is often easier to run a script which downloads the latest version directly from the vendor's servers and installs it.
There are obviously a few downsides to this approach:
- when your fleet is mostly on site and many will install or update at the same time, they will reach out over the internet to the vendor's servers, possibly overwhelming your internet connection
- when you download software from the internet, it should be verified to avoid man-in-the-middle or other injection attacks
- there is no control over which version the clients get
- there is no control over which version the clients get, you cannot "hold back" new versions for testing and approval workflows
- some application downloads are gated behind logins or paywalls and cannot be automated this way
Some of these disadvantages can be seen as advantages in different setups. When your fleet is mostly mobile and offsite, then downloading from vendor servers will relieve the inbound connection to your management server, or the data usage on your cloud server. Software vendors are pushing for subscriptions with continuous updates and feature releases, so moving the entire team to the latest versions quickly can make those available quickly. Also being on the latest release includes all current security patches.
Some of these disadvantages can be seen as advantages in different setups. When your fleet is mostly mobile and offsite, then downloading from vendor servers will relieve the inbound connection to your management server, or the data usage on your management system's cloud server. Software vendors are pushing for subscriptions with continuous updates and feature releases, so moving the entire team to the latest versions quickly can make those available quickly. Also being on the latest release includes all current security patches.
Because this is an attractive solution for _certain kinds_ of deployment, there are already many scripts out there that will download and install the latest version of a given software. And we have built and used quite a few in-house, as well. Most importantly, [William Smith has this script](https://gist.github.com/talkingmoose/a16ca849416ce5ce89316bacd75fc91a) which can be used to install several different Microsoft applications and bundles, because Microsoft has a nice unified URL scheme.
@@ -58,7 +72,7 @@ Installomator can work with the following common archive and installer types:
When the download yields a pkg file, Installomator will run `installer` to install it on the current system.
Applications in dmgs or zips will be copied to `/Applications` and their owner will be set to the current user, so the install works like a standard drag'b drop installation.
Applications in dmgs or zips will be copied to `/Applications` and their owner will be set to the current user, so the install works like a standard drag'n drop installation.
(I consider it a disgrace, that Jamf, after nearly 20 years, _still_ cannot deal with 'drag'n drop installation dmgs' natively. It's not _that_ hard.)
@@ -128,7 +142,7 @@ Other than the version arguments, the argument can be any of the labels listed i
### Debug mode
There is a variable named `DEBUG` which is set in line 21 of the script. When `DEBUG` is set to `1` (default) no actions that wousld actually modify the current system are taken. This is useful for testing most of the actions in the script, but obviously not all of them.
There is a variable named `DEBUG` which is set in line 21 of the script. When `DEBUG` is set to `1` (default) no actions that would actually modify the current system are taken. This is useful for testing most of the actions in the script, but obviously not all of them.
Also when the `DEBUG` variable is `1`, downloaded archives and extracted files will be written to the script's directory, rather than a temporary directory, which can make debugging easier.
@@ -138,6 +152,8 @@ _Always remember_ to change the `DEBUG` variable to `0` when deploying.
In Jamf Pro, create a new 'Script' and paste the contents of `Installomator.sh` into the 'Script Contents' area. Under 'Options' you can change the parameter label for argument 4 to 'Application Label.'
Remember to set `DEBUG` to `0`.
Then you can use the Installomator script in a policy and choose the application to install by setting the label for argument 4.
## What it does
@@ -180,6 +196,14 @@ There are four options:
The default is `prompt_user`.
### Notification
The `NOTIFY` variable controls the notifications shown to the user. As of now, there are two options: `success` (default) and `silent`.
- `success`: notify the user after a successful install
- `silent`: no notifications
### Adding applications/label blocks
#### Required Variables
@@ -267,4 +291,51 @@ Depending on the application or pkg there are a few more variables you can or ne
- `updateToolRunAsCurrentUser`:
When this variable is set (any value), `$updateTool` will be run as the current user. Default is unset and
## Frequently Asked Questions
### What if the latest version of the app is already installed?
Short answer: Installomator will re-download and re-install the latest over the existing installation.
Longer answer:
Installomator will try to find a currently installed app to log the version. When Installomator finds an existing app (any version) and the `updateTool` variable is set, then Installomator will _not_ download and install, but run the `updateTool` instead.
However, there is no simple generic method to actually determine the latest version of an application or installer.
We deploy Installomator usually for user initiated installations from Self Service, so re-installs don't really 'hurt' and may be a useful troubleshooting step.
When you want to have automated installations, you can use smart groups based on the app version to limit excessive re-installations.
### Why don't you just use `autopkg install`?
Short answer: `autopkg` is not designed or suited for this kind of workflow
Long answer:
The motivation to not re-invent the wheel and use and existing tool is understandable. However, `autopkg` was not designed with this use case in mind and has a few significant downsides.
First, you would have to deploy and manage autopkg on all the clients. But to do its work, `autopkg` requires recipes. So, you have to install, and update the recipe repos on the client, as well. For security reasons, you _really_ should only run trusted recipes, so you need to install and update your personal recipe overrides as well.
The recipes you use are probably spread across multiple community provided recipe-repos, so we have `autopkg` itself, several recipe-repos, and your overrides that we need to manage, each of which may need to be updated at any time.
The community recipe-repos contain several recipes for different applications. When you add a recipe-repo for an app you want, you will also install all the other recipes from that repo.
The `autopkg install` does _not_ require root or even administrative privileges. _Any_ user (even standard users) on the system can now install any of the random recipes that came with the community repos.
To prevent users installing random apps from the community repos, you can curate your own recipe-repo from the community repos and push that to the clients. At this point, you are managing autopkg, your curated repo, your recipe overrides on the clients and handling the additional work of curating and updating your recipe-repo and the overrides.
In addition, a really savvy user (or a malicious attacker) could build their own recipe and run it using the pre-installed `autopkg` you installed.
And then consider what your CISO department (if you have one) would say about the `autopkgserver` and `autopkginstalld` daemons running on all the clients...
At this point it would be easier to use AutoPkg the way it was intended: on a single admin Mac, and let it upload the pkgs to your management system, which deploys them. Each tool is doing what it is designed for.
Please don't misunderstand this as me saying that AutoPkg is a bad or poorly designed tool. AutoPkg is amazing, powerful, and useful. The [Scripting OS X recipe-repo](https://github.com/autopkg/scriptingosx-recipes) is one of the older repos. AutoPkg is valuable tool to help admins with many apps that cannot be automated with tools like Installomator, and with deployment strategies that require more control.
But it is not suited as a client install automation tool.
### Why don't you just use brew?
Read the explanation for `autopkg`, pretty much the same applies for `brew`, i.e. While it is useful on a single Mac, it is a un-manageable mess when you think about deploying and managing on a fleet of computers.