Current development phase: ALPHA. DO NOT USE IN PRODUCTION!!!
InstallApplications Swiftly (IAS) is a Swift reimplementation of the popular Python based tool InstallApplications (IA) created by @erikng.
IAS has several advantages over the original IA:
- No Python dependency. Implementation done purely with Swift and Apple frameworks.
- Smaller package footprint and faster installation. Current package size is less than 1 MB. There are <10 files to install. IA package size is around 40 MB and it contains more than 6000 files because of the bundled Python 3 framework.
- Faster workflow execution:
- Items are downloaded in parallel using separate threads.
- Ability to run designated tasks in parallel.
IAS is a bootstrap tool to be used primarily during the macOS MDM Enrolment. Main goal is to easily deploy other bootstrap scripts and packages right after the MDM enrolment. List of packages and script is maintained server side so you don't have to modify your MDM bootstrap package every time you want to make a change to your bootstrap process.
IAS is ment to be deployed by the MDM using the InstallEnterpriseApplication
command.
Your MDM has to support bootstrap package functionality. You can find more on this topic
in the original IA README -> MDMs that support Custom DEP.
IAS requires macOS 11.1 or newer.
General outline of the entire IAS lifecycle.
Steps you need to do to prepare IAS for deployment:
- Prepare signed distribution package containing IAS and its configuration. See section Preparing IAS for deployment.
- Import IAS package into your MDM and assign it to desired set of Macs.
- Create JSON control file and serve it with a webserver. See section JSON control file.
- Make sure all package abd script URLs contained in the JSON control file are reachable.
- Mac enrolls into MDM.
- MDM sends a
InstallEnterpriseApplication
command to inform the device is should download and install IAS package. - Package script loads the IAS LaunchDaemon
iasd
and LaunchAgentiasagent
. isasd
downloads the JSON control file from URL specified by its configuration.isasd
starts executing the Phases. See section Phases.- If there is a userland phase defined in the JSON control file,
IAS will wait until there is active user GUI session before installing.
iasagent
is used to execute scripts on behalf of the logged in user. - After the work is done
isasd
deletes alls its files, unloads itself and theiasagent
from the launchd and exists.
There are three phases of the bootstrap process.
You can specify only rootscript items for this phase.
preflight
phase is useful for running IAS on previously deployed machines or if you simply want to re-run it.
If all preflight
scripts exit with zero exit code, isad
finishes bootstrap process, bypassing the setupassistant
and userland
phases.
If one of the preflight
scripts exists with non-zero exit code, isad
continues to the setupassistant phase.
At the beginning of setupassistant
phase isad
starts downloding items for the setupassistant
and userland
phases.
setupassistant
phase should only contain rootscript a package items which can be installed without need for active user session.
userland
phase starts after the user logs in and all items in the setupassistant
phase has been executed.
It can contain package, rootscript or userscript items.
Items of type userscript are run by the iasagent
with the privileges of the logged in user.
Running items in the userland
phase is extremely useful for situations where bootstrap process needs to communicate with the user via GUI elements.
When IAS is configured only using the configuration profile from MDM you can deploy release package without any modifications. You need to make sure MDM issues command to install the configuration profile before the command to install the IAS package.
You can use macOS UserDefaults
to provide the settings for IAS. To do that just set the settings keys for domain: cz.macadmin.iasd
.
It makes sense to deploy the settings via the configuration profile pushed by the MDM.
Example: SampleProfile.mobileconfig.
IAS releases include a zip archive containing the ias-munkipkg
directory to used with the munkipkg to create the IAS package.
Alternatively you can use munkipkg
directory from the git repository itself. However you need to include binaries manually.
You can provide the configuration using the plist file bundled within the package.
- Put the file named
iasd.plist
into the directory containingiasd
binary. - Provide path to config plist as a first argument of
iasd
. This means editing the LaunchDaemon plist:
<key>Program</key>
<string>/Library/installapplications/iasd</string>
<key>ProgramArguments</key>
<array>
<string>/path/to/your/iasd/config.plist</string>
</array>
If for some reason you manage do do both, file provided as an argument takes precedence.
Example: SampleOptions.plist.
Distribution package must be signed with a valid Apple installer signing certificate. Without the signature, package can not be deployed by the MDM. Learn more about Apple Developer programs and certificates at Apple Developer portal.
You can use the munkipkg
to sign the package by adding signing_info
dictionary to build-info
file:
<key>signing_info</key>
<dict>
<key>identity</key>
<string>Mac Installer: Your package signing certificate name (XXXXXXXXXXX)</string>
<key>timestamp</key>
<true/>
</dict>
You build the package using the munkipkg
:
munkipkg /path/to/munkipkg/directory
By default all package files are installed with the ownership root:wheel
and the unix permissions set to 0755
(directories) or 0644
(files).
With these default every macOS user can read them. This might not be desirable in all situations.
For example if you put HTTP authentication credentials into the configuration file you would like to prevent the user from reading it easily.
If you want to change these permission to more restrictive values (f.e. 0700
|0600
) you can do that with the pkgbuild
--ownership
flag.
Inspect man pkgbuild
and corresponding munkipkg
documentation for more information.
Settings are provided by configuration plist file (section Adding the configuration) or via the configuration profile (section Using the configuration profile).
Setting key | Default value | Type | Description |
---|---|---|---|
General settings | |||
HTTPAuthPassword | String | Password for HTTP Basic or HTTP Digest authentication. Applied to all HTTP authentication challenges. | |
HTTPAuthUser | String | Username for HTTP Basic or HTTP Digest authentication. Applied to all HTTP authentication challenges. | |
JSONURL | String | REQUIRED URL of the JSON control file. HTTPS strongly recommended. If you want to use SkipJSONValidation setting and include the JSON control file inside the IAS package you can provide any url but the last component of this URL must match the file name of the JSON control file. |
|
Reboot | false | Bool | When true iasd will initiate macOS reboot before it exits. |
Customization | |||
HashCheckPolicy | Strict | String | Can be either Strict , Warning or Ignore . See section Hack check policy. |
InstallPath | /Library/installapplications |
String | IAS main directory. iasd ensures this directory is created (including the userscripts subdirectory) before downloading the JSON control file. iasd also deletes this directory during the cleanup. |
Identifier | cz.macadmin.ias |
String | IAS identifier. See section Identifiers for more information. |
LaunchDaemonIdentifier | cz.macadmin.iasd |
String | IAS LaunchDaemon identifier. See section Identifiers for more information. |
LaunchAgentIdentifier | cz.macadmin.iasagent |
String | IAS LaunchAgent identifier. See section Identifiers for more information. |
MinDownloadConcurrency | 1 | Integer | See section Parallel downloads for more information. |
MaxDownloadConcurrency | 4 | Integer | See section Parallel downloads for more information. |
MaximumRedownloads | 3 | Integer | Maximum number of redownload attempts. |
SkipJSONValidation | false | Bool | When true the JSON control file is not deleted and redowloaded if found inside the InstallPath directory at begging of the iasd run. |
WaitForAgentTimeout | 86400 | Integer | Amount of time in seconds indicating how long iasd is willing to wait for iasagent to connect when beginning the userland phase. When reached, iasd exits without doing the clean up. |
Reporting | |||
DEPNotifyEnable | false | Bool | Enable reporting to DEPNotify control file |
DEPNotifyControlFile | /var/tmp/depnotify.log |
String | Path to DEPNotify control file. IAS will attempt to create the file if it does not exists but not the parent directory. |
DEPNotifyDeterminate | true | Bool | Set the DEPNotify DeterminateManualStep to the number of items in the JSON control file. |
Troubleshooting | |||
DryRun | false | Bool | When true iasd downloads all the items but nothing is installed/executed. If the userland phase is present iasd waits for agent to connect (Issue #11). |
Settings can be loaded from multiple sources. Priority
- Configuration file
- UserDefaults (configuration profile)
- IAS default value
IAS can compare known SHA256 digest value to the digest computed from the actual downloaded file.
Hash checking only applies to items with the url
key set. IAS never checks hash digest of items which are ment not to be downloaded.
HashCheckPolicy
has three modes how it verifies dowloaded item integrity:
- Strict (Default): After the item download, SHA256 digest of the file is computed and compared to value of item
hash
key in the JSON control file. If the hash digests does not match, file is downloaded again until the digest value matches orMaximumRedownloads
limit is reached. If the hash digest still does not match after the last redownload attempt, download is considered failed. Also if thehash
is missing from the JSON control file, download is considered failed. This is the same behavior as in the original IA tool. - Warning: SHA256 digest is compared to the
hash
key in the JSON control file. If the hash digests does not match, warning is issued but download is considered succesfull. If thehash
is missing from the JSON control, warning is issued but download is considered succesfull. - Ignore: SHA256 is never computed or compared. Finished downloads are always considered succesfull.
IAS uses four identifiers:
Identifier
: Tool identifier. Used mainly for system log subsystem identifier. See section Logging.LaunchDaemonIdentifier
: Identifier of the IAS LaunchDaemon.LaunchAgentIdentifier
: Identifier of the IAS LaunchAgent.XPCServiceIdentifier
: Currently not configurable. Alwayscz.macadmin.ias.xpc
.
You can change the identifiers without modifying the source code and compiling binaries via respective Settings. See following sections how to do that.
There are places where configurable identifiers do not apply:
- Log subsystem id before the settings are loaded.
iasagent
log subsystem during the XPC initialization.XPCServiceIdentifier
.
Some of theses cases might be mitigated in the future by implementing command line arguments.
For now, if you want to completely change the values of the identifiers, edit their constants in Shared/Constants.swift
and compile the binaries.
To change the id of the log subsystem, set the Identifier
to desired value key using the Settings.
To change the package id of the IAS package modify the value of the identifier
key in munkipkg build-info file and build the IAS package.
You need to change the identifier in all of the following places:
- Set the
LaunchDaemonIdentifier
key toyour.new.daemon.id
using the Settings. - Rename the LaunchDaemon file
cz.macadmin.iasd.plist
toyour.new.daemon.id.plist
. - Change the value of the LaunchDaemon
Label
key toyour.new.daemon.id
. - Change the value of the
launch_daemon_id
variable in the IAS package preinstall script toyour.new.daemon.id
. - Change the value of the
launch_daemon_id
variable in the IAS package postinstall script toyour.new.daemon.id
.
You need to change the identifier in all of the following places:
- Set the
LaunchAgentIdentifier
key toyour.new.agent.id
using the Settings. - Rename the LaunchAgent file
cz.macadmin.iasgent.plist
toyour.new.agent.id.plist
. - Change the value of the LaunchAgent
Label
key toyour.new.agent.id
. - Change the value of the
launch_agent_id
variable in the IAS package preinstall script toyour.new.agent.id
. - Change the value of the
launch_agent_id
variable in the IAS package postinstall script toyour.new.agent.id
.
IAS downloads the files in parallel. There are two options to control this bahavior:
- At the start of preflight and setupassistant phases number of allowed concurrent downloads starts at
MinDownloadConcurrency
. - With each completed download level of concurrency is increased by one until the
MaxDownloadConcurrency
is reached.
When enabled (DEPNotifyEnable
) IAS can write status messages to the DEPNotify control file (DEPNotifyControlFile
).
IAS also manipulates the DEPNotify progress bar using determinate feature (DEPNotifyDeterminate
).
Please note IAS does not launch or quit the DEPNotify application. Setting that up is completely up to the administrator.
IAS is 100% compatible with the IA JSON control file format. However there are few new options.
Item keys | Default value | Type | Description |
---|---|---|---|
Standard general keys | |||
file | String | REQUIRED System path where the item file is present or is going to be downloaded to. | |
name | String | REQUIRED Name of the item. Used for logging purposes. | |
type | String | REQUIRED Item type. One of the following: package , rootscript or userscript . See sections Packages and Scripts. |
|
donotwait | false | Bool | When true iasd starts the item execution but does not wait for it to finish and proceeds to the next item. |
hash | String | SHA256 digest of the item file. Required unless HashCheckPolicy is set to Ignore . |
|
pkg_required | false | Bool | When true package is always installed by the iasd with no regard to existing package receipt |
url | String | URL of the item file where it can be downloaded from. | |
New general keys | |||
fail_policy | failable_execution | String | One of the following: failable , failable_execution , failure_is_not_an_option . See section Fail policy. |
parallel_group | String | Parallel group id. See section Parallel groups. | |
Package specific keys | |||
packageid | String | Package identifier. See section Package. | |
version | String | Package version. See section Package. |
Example:
{
"preflight": [
{
"file": "/Library/installapplications/preflight1.sh",
"hash": "sha256 hash",
"name": "First Preflight Script",
"type": "rootscript",
"url": "https://domain.tld/preflight1.sh"
},
{
"file": "/Library/installapplications/preflight2.sh",
"hash": "sha256 hash",
"name": "Second Preflight Script",
"type": "rootscript",
"url": "https://domain.tld/preflight2.sh"
}
],
"setupassistant": [
{
"file": "/Library/installapplications/package1.pkg",
"hash": "sha256 hash",
"name": "Package 1 installed in parallel with the Rootscript 1 run",
"packageid": "com.package.package1",
"parallel_group": "alpha",
"type": "package",
"url": "https://domain.tld/package1.pkg",
"version": "1.0"
},
{
"file": "/Library/installapplications/rootscript1.py",
"hash": "sha256 hash",
"name": "Rootscript 1 run in parallel with the Package 1 installation",
"parallel_group": "alpha",
"type": "rootscript",
"url": "https://domain.tld/userland_examplerootscript.py"
}
],
"userland": [
{
"fail_policy": "failure_is_not_an_option",
"file": "/Library/installapplications/package2.pkg",
"hash": "sha256 hash",
"name": "Package 2 installed in the userland phase which must not fail for IAS to proceed",
"packageid": "com.package.package2",
"type": "package",
"url": "https://domain.tld/package2.pkg",
"version": "1.0"
},
{
"donotwait": true,
"file": "/Library/installapplications/rootscript2.sh",
"hash": "sha256 hash",
"name": "Rootscript 2 run asynchronously",
"type": "rootscript",
"url": "https://domain.tld/rootscript2.sh"
},
{
"file": "/Library/installapplications/userscripts/userscript.sh",
"hash": "sha256 hash",
"name": "Userscript to be run by iasagent",
"type": "userscript",
"url": "https://domain.tld/userscript.py"
}
]
}
JSON control file can be created manually or automatically generated by some other tool. Check out generatejson.py from the original IA.
You can compute file SHA256 digest by using the command: shasum -a 256 /path/to/file
.
Read more about hash digest checking in the Hack check policy section.
IAS uses packageid
and version
to check whether package receipt exits or not.
If receipt is found and already installed version is the same or newer than version of the item, package is considered installed.
In the default state IAS does not attempt install packages it considers already installed.
You can override this behavior by providing the pkg_required
key.
Package items with pkg_required
set to true are alway installed by IAS.
Omitting packageid
or/and version
keys leads to the same result but generates warning message in the log.
Since packages can only be installed by root user = iasd
, their file permissions are set to 0o600
.
Origial IA tool required to put items of type userscript
into separate userscripts
directory.
With IAS you can put userscript
item anywhere you like because it is downloaded by the iasd
daemon running as root.
However IAS creates userscripts
directory inside InstallPath
for compatibility with existing JSON control files.
Item file unix permissions:
rootscript
:0700
userscript
:0755
Every item can have its own fail_policy
. Three are three options:
failable
: Item download or execution can fail. IAS logs an error but proceeds to the next item.failable_execution
(DEFAULT): Item execution can produce non-zero exit code but download must succeed. If item download fails IAS aborts the entire run.failure_is_not_an_option
: If download fails or execution produces non-zero exit code IAS aborts the entire run.
All subsequent items with the same parallel_group
identifier can be executed in parallel.
This does not guarantee execution of these items to start at the same time since some of them might not yet be downloaded.
IAS does wait for all items in the same parallel_group
to finish their execution before proceeding to the following item (or parallel group).
You can combine parallel_group
with donotwait
.
Tasks marked with donotwait
start package install or script execution but do not wait for them finish.
IAS waits only for the task responsible of starting the work but not the work itself since that was launched fire-and-forget style.
Example
"name": "item1",
"parallel_group": "alpha",
...
"name": "item2",
"parallel_group": "alpha",
...
"name": "item3",
"parallel_group": "beta",
...
"name": "item4",
"parallel_group": "beta",
...
"donotwait": true,
"name": "item5",
"parallel_group": "beta",
...
"name": "item6",
"parallel_group": "alpha",
- Start execution of
item1
anditem2
(alpha
parallel group) as soon as their individual downloads are complete. - Wait for the
item1
anditem2
to finish. - Start execution of
item3
,item4
anditem5
(beta
parallel group) as soon as their individual downloads are complete. - Wait for the
item3
anditem4
to finish. No need to wait foritem5
since it is marked withdonotwait
. - Start execution of
item6
as soon as its download is complete. Note theparallel_group
set toalpha
. This is probably by mistake,parallel_group
identifier should be gamma. IAS treats this as third parallel group regardless of the name because there are other non-alpha items in between.
There is not much gain from trying to install multiple packages in parallel.
macOS installer
installs only one package at the time.
You can configure IAS to authenticate HTTP requests by providing HTTPAuthPassword
and HTTPAuthUser
settings.
HTTP authentication happens on demand (only when server requires it).
Supported methods are HTTP Basic and HTTP Digest.
HTTP redirects are followed automatically.
iasd
and iasagent
do not print anything to stdout or stderr.
All log messages are sent to the macOS system log instead.
You can use Console.app or log
command line tool to inspect them.
You can filter them out by specifying the subsystem: cz.macadmin.ias
(default).
Please note even when you change the Identifier
some of the log messages are still marked
with subsystem cz.macadmin.ias
. Read more about this in Identifiers section.
View all past IAS messages:
log show --predicate 'subsystem == "cz.macadmin.ias"'
View IAS messages within the last hour including those with info and debug severity levels:
log show --last 1h --info --debug --predicate 'subsystem == "cz.macadmin.ias"'
Stream new IAS messages:
log stream --predicate 'subsystem == "cz.macadmin.ias"'