xamarin-macios/system-dependencies.sh

945 строки
30 KiB
Bash
Executable File

#!/bin/bash -e
set -o pipefail
cd $(dirname $0)
FAIL=
PROVISION_DOWNLOAD_DIR=/tmp/x-provisioning
SUDO=sudo
VERBOSE=
OPTIONAL_SHARPIE=1
OPTIONAL_SIMULATORS=1
# parse command-line arguments
while ! test -z $1; do
case $1 in
--no-sudo)
SUDO=
shift
;;
--provision-xcode)
PROVISION_XCODE=1
unset IGNORE_XCODE
shift
;;
--provision)
# historical reasons :(
PROVISION_XCODE=1
PROVISION_VS=1
unset IGNORE_XCODE
unset IGNORE_VISUAL_STUDIO
shift
;;
--provision-*-studio)
PROVISION_VS=1
unset IGNORE_VISUAL_STUDIO
shift
;;
--provision-mono)
PROVISION_MONO=1
unset IGNORE_MONO
shift
;;
--provision-cmake)
PROVISION_CMAKE=1
unset IGNORE_CMAKE
shift
;;
--provision-7z)
PROVISION_7Z=1
unset IGNORE_7Z
shift
;;
--provision-autotools)
PROVISION_AUTOTOOLS=1
unset IGNORE_AUTOTOOLS
shift
;;
--provision-python3)
# building mono from source requires having python3 installed
PROVISION_PYTHON3=1
unset IGNORE_PYTHON3
shift
;;
--provision-sharpie)
PROVISION_SHARPIE=1
unset OPTIONAL_SHARPIE
unset IGNORE_SHARPIE
shift
;;
--provision-simulators)
PROVISION_SIMULATORS=1
unset OPTIONAL_SIMULATORS
unset IGNORE_SIMULATORS
shift
;;
--provision-dotnet)
PROVISION_DOTNET=1
unset IGNORE_DOTNET
shift
;;
--provision-all)
PROVISION_MONO=1
unset IGNORE_MONO
PROVISION_VS=1
unset IGNORE_VISUAL_STUDIO
PROVISION_XCODE=1
unset IGNORE_XCODE
PROVISION_CMAKE=1
unset IGNORE_CMAKE
PROVISION_7Z=1
unset IGNORE_7Z
PROVISION_AUTOTOOLS=1
unset IGNORE_AUTOTOOLS
PROVISION_HOMEBREW=1
unset IGNORE_HOMEBREW
PROVISION_SHARPIE=1
unset IGNORE_SHARPIE
PROVISION_SIMULATORS=1
unset IGNORE_SIMULATORS
PROVISION_PYTHON3=1
unset IGNORE_PYTHON3
PROVISION_DOTNET=1
unset IGNORE_DOTNET
shift
;;
--ignore-all)
IGNORE_OSX=1
IGNORE_MONO=1
IGNORE_VISUAL_STUDIO=1
IGNORE_XCODE=1
IGNORE_CMAKE=1
IGNORE_7Z=1
IGNORE_AUTOTOOLS=1
IGNORE_HOMEBREW=1
IGNORE_SHARPIE=1
IGNORE_SIMULATORS=1
IGNORE_PYTHON3=1
IGNORE_DOTNET=1
shift
;;
--ignore-osx)
IGNORE_OSX=1
shift
;;
--ignore-xcode)
IGNORE_XCODE=1
shift
;;
--ignore-*-studio)
IGNORE_VISUAL_STUDIO=1
shift
;;
--ignore-mono)
IGNORE_MONO=1
shift
;;
--ignore-autotools)
IGNORE_AUTOTOOLS=1
shift
;;
--ignore-python3)
IGNORE_PYTHON3=1
shift
;;
--ignore-cmake)
IGNORE_CMAKE=1
shift
;;
--ignore-7z)
IGNORE_7Z=1
shift
;;
--ignore-sharpie)
IGNORE_SHARPIE=1
shift
;;
--enforce-sharpie)
unset IGNORE_SHARPIE
unset OPTIONAL_SHARPIE
shift
;;
--ignore-simulators)
IGNORE_SIMULATORS=1
shift
;;
--enforce-simulators)
unset IGNORE_SIMULATORS
unset OPTIONAL_SIMULATORS
shift
;;
--ignore-dotnet)
IGNORE_DOTNET=1
shift
;;
-v | --verbose)
set -x
shift
VERBOSE=1
;;
*)
echo "Unknown argument: $1"
exit 1
;;
esac
done
# reporting functions
COLOR_RED=$(tput setaf 1 2>/dev/null || true)
COLOR_ORANGE=$(tput setaf 3 2>/dev/null || true)
COLOR_MAGENTA=$(tput setaf 5 2>/dev/null || true)
COLOR_BLUE=$(tput setaf 6 2>/dev/null || true)
COLOR_CLEAR=$(tput sgr0 2>/dev/null || true)
COLOR_RESET=uniquesearchablestring
FAILURE_PREFIX=
if test -z "$COLOR_RED"; then FAILURE_PREFIX="** failure ** "; fi
function fail () {
echo " $FAILURE_PREFIX${COLOR_RED}${1//${COLOR_RESET}/${COLOR_RED}}${COLOR_CLEAR}"
FAIL=1
}
function warn () {
echo " ${COLOR_ORANGE}${1//${COLOR_RESET}/${COLOR_ORANGE}}${COLOR_CLEAR}"
}
function ok () {
echo " ${1//${COLOR_RESET}/${COLOR_CLEAR}}"
}
function log () {
echo " ${1//${COLOR_RESET}/${COLOR_CLEAR}}"
}
# $1: the version to check
# $2: the minimum version to check against
function is_at_least_version () {
ACT_V=$1
MIN_V=$2
if [[ "$ACT_V" == "$MIN_V" ]]; then
return 0
fi
IFS=. read -a V_ACT <<< "$ACT_V"
IFS=. read -a V_MIN <<< "$MIN_V"
# get the minimum # of elements
AC=${#V_ACT[@]}
MC=${#V_MIN[@]}
COUNT=$(($AC>$MC?$MC:$AC))
C=0
while (( $C < $COUNT )); do
ACT=${V_ACT[$C]}
MIN=${V_MIN[$C]}
if (( $ACT > $MIN )); then
return 0
elif (( "$MIN" > "$ACT" )); then
return 1
fi
let C++
done
if (( $AC == $MC )); then
# identical?
return 0
fi
if (( $AC > $MC )); then
# more version fields in actual than min: OK
return 0
elif (( $AC == $MC )); then
# entire strings aren't equal (first check in function), but each individual field is?
return 0
else
# more version fields in min than actual (1.0 vs 1.0.1 for instance): not OK
return 1
fi
}
function install_mono () {
local MONO_URL=`grep MIN_MONO_URL= Make.config | sed 's/.*=//'`
local MIN_MONO_VERSION=`grep MIN_MONO_VERSION= Make.config | sed 's/.*=//'`
if test -z $MONO_URL; then
fail "No MIN_MONO_URL set in Make.config, cannot provision"
return
fi
mkdir -p $PROVISION_DOWNLOAD_DIR
log "Downloading Mono $MIN_MONO_VERSION from $MONO_URL to $PROVISION_DOWNLOAD_DIR..."
local MONO_NAME=`basename $MONO_URL`
local MONO_PKG=$PROVISION_DOWNLOAD_DIR/$MONO_NAME
curl -L $MONO_URL > $MONO_PKG
log "Installing Mono $MIN_MONO_VERSION from $MONO_URL..."
$SUDO installer -pkg $MONO_PKG -target /
rm -f $MONO_PKG
}
function run_xcode_first_launch ()
{
local XCODE_VERSION="$1"
local XCODE_DEVELOPER_ROOT="$2"
# xcodebuild -runFirstLaunch seems to have been introduced in Xcode 9
if ! is_at_least_version "$XCODE_VERSION" 9.0; then
return
fi
if ! "$XCODE_DEVELOPER_ROOT/usr/bin/xcodebuild" -checkFirstLaunchStatus; then
if ! test -z "$PROVISION_XCODE"; then
# Run the first launch tasks
log "Executing '$SUDO $XCODE_DEVELOPER_ROOT/usr/bin/xcodebuild -runFirstLaunch'"
$SUDO "$XCODE_DEVELOPER_ROOT/usr/bin/xcodebuild" -runFirstLaunch
log "Executed '$SUDO $XCODE_DEVELOPER_ROOT/usr/bin/xcodebuild -runFirstLaunch'"
else
fail "Xcode has pending first launch tasks. Execute 'make fix-xcode-first-run' to execute those tasks."
return
fi
fi
}
function install_specific_xcode () {
local XCODE_URL=`grep XCODE$1_URL= Make.config | sed 's/.*=//'`
local XCODE_VERSION=`grep XCODE$1_VERSION= Make.config | sed 's/.*=//'`
local XCODE_DEVELOPER_ROOT="$2"
local XCODE_ROOT="$(dirname "$(dirname "$XCODE_DEVELOPER_ROOT")")"
if test -z $XCODE_URL; then
fail "No XCODE$1_URL set in Make.config, cannot provision"
return
fi
mkdir -p $PROVISION_DOWNLOAD_DIR
log "Downloading Xcode $XCODE_VERSION from $XCODE_URL to $PROVISION_DOWNLOAD_DIR..."
local XCODE_NAME=`basename $XCODE_URL`
local XCODE_DMG=$PROVISION_DOWNLOAD_DIR/$XCODE_NAME
# To test this script with new Xcode versions, copy the downloaded file to $XCODE_DMG,
# uncomment the following curl line, and run ./system-dependencies.sh --provision-xcode
if test -f "$HOME/Downloads/$XCODE_NAME"; then
log "Found $XCODE_NAME in your ~/Downloads folder, copying that version to $XCODE_DMG instead of re-downloading it."
cp "$HOME/Downloads/$XCODE_NAME" "$XCODE_DMG"
else
curl -L $XCODE_URL > $XCODE_DMG
fi
if [[ ${XCODE_DMG: -4} == ".dmg" ]]; then
local XCODE_MOUNTPOINT=$PROVISION_DOWNLOAD_DIR/$XCODE_NAME-mount
log "Mounting $XCODE_DMG into $XCODE_MOUNTPOINT..."
hdiutil attach $XCODE_DMG -mountpoint $XCODE_MOUNTPOINT -quiet -nobrowse
log "Removing previous Xcode from $XCODE_ROOT"
rm -Rf $XCODE_ROOT
log "Installing Xcode $XCODE_VERSION to $XCODE_ROOT..."
cp -R $XCODE_MOUNTPOINT/*.app $XCODE_ROOT
log "Unmounting $XCODE_DMG..."
hdiutil detach $XCODE_MOUNTPOINT -quiet
elif [[ ${XCODE_DMG: -4} == ".xip" ]]; then
log "Extracting $XCODE_DMG..."
pushd . > /dev/null
cd $PROVISION_DOWNLOAD_DIR
# make sure there's nothing interfering
rm -Rf *.app
rm -Rf $XCODE_ROOT
# extract
xip --expand "$XCODE_DMG"
log "Installing Xcode $XCODE_VERSION to $XCODE_ROOT..."
mv *.app $XCODE_ROOT
popd > /dev/null
else
fail "Don't know how to install $XCODE_DMG"
fi
rm -f $XCODE_DMG
log "Removing any com.apple.quarantine attributes from the installed Xcode"
$SUDO xattr -s -d -r com.apple.quarantine $XCODE_ROOT
if is_at_least_version $XCODE_VERSION 5.0; then
log "Accepting Xcode license"
$SUDO $XCODE_DEVELOPER_ROOT/usr/bin/xcodebuild -license accept
fi
if is_at_least_version "$XCODE_VERSION" 9.0; then
run_xcode_first_launch "$XCODE_VERSION" "$XCODE_DEVELOPER_ROOT"
elif is_at_least_version $XCODE_VERSION 8.0; then
PKGS="MobileDevice.pkg MobileDeviceDevelopment.pkg XcodeSystemResources.pkg"
for pkg in $PKGS; do
if test -f "$XCODE_DEVELOPER_ROOT/../Resources/Packages/$pkg"; then
log "Installing $pkg"
$SUDO /usr/sbin/installer -dumplog -verbose -pkg "$XCODE_DEVELOPER_ROOT/../Resources/Packages/$pkg" -target /
log "Installed $pkg"
else
log "Not installing $pkg because it doesn't exist."
fi
done
fi
log "Clearing xcrun cache..."
xcrun -k
ok "Xcode $XCODE_VERSION provisioned"
}
function install_coresimulator ()
{
local XCODE_DEVELOPER_ROOT
local CORESIMULATOR_PKG
local CORESIMULATOR_PKG_DIR
local XCODE_ROOT
local TARGET_CORESIMULATOR_VERSION
local CURRENT_CORESIMULATOR_VERSION
XCODE_DEVELOPER_ROOT=$(grep XCODE_DEVELOPER_ROOT= Make.config | sed 's/.*=//')
XCODE_ROOT=$(dirname "$(dirname "$XCODE_DEVELOPER_ROOT")")
CORESIMULATOR_PKG=$XCODE_ROOT/Contents/Resources/Packages/XcodeSystemResources.pkg
if ! test -f "$CORESIMULATOR_PKG"; then
warn "Could not find XcodeSystemResources.pkg (which contains CoreSimulator.framework) in $XCODE_DEVELOPER_ROOT ($CORESIMULATOR_PKG doesn't exist)."
return
fi
# Get the CoreSimulator.framework version from our Xcode
# Extract the .pkg to get the pkg's PackageInfo file, which contains the CoreSimulator.framework version.
CORESIMULATOR_PKG_DIR=$(mktemp -d)
pkgutil --expand "$CORESIMULATOR_PKG" "$CORESIMULATOR_PKG_DIR/extracted"
if ! TARGET_CORESIMULATOR_VERSION=$(xmllint --xpath 'string(/pkg-info/bundle-version/bundle[@id="com.apple.CoreSimulator"]/@CFBundleShortVersionString)' "$CORESIMULATOR_PKG_DIR/extracted/PackageInfo"); then
rm -rf "$CORESIMULATOR_PKG_DIR"
warn "Failed to look up the CoreSimulator version of $XCODE_DEVELOPER_ROOT"
return
fi
rm -rf "$CORESIMULATOR_PKG_DIR"
# Get the CoreSimulator.framework currently installed
local CURRENT_CORESIMULATOR_PATH=/Library/Developer/PrivateFrameworks/CoreSimulator.framework/Versions/A/CoreSimulator
local CURRENT_CORESIMULATOR_VERSION=0.0
if test -f "$CURRENT_CORESIMULATOR_PATH"; then
CURRENT_CORESIMULATOR_VERSION=$(otool -L $CURRENT_CORESIMULATOR_PATH | grep "$CURRENT_CORESIMULATOR_PATH.*current version" | sed -e 's/.*current version//' -e 's/)//' -e 's/[[:space:]]//g' | uniq)
fi
# Either version may be composed of either 2 or 3 numbers.
# We only care about the first two, so strip off the 3rd number if it exists.
# shellcheck disable=SC2001
CURRENT_CORESIMULATOR_VERSION=$(echo "$CURRENT_CORESIMULATOR_VERSION" | sed 's/\([0-9]*[.][0-9]*\).*/\1/')
# shellcheck disable=SC2001
TARGET_CORESIMULATOR_VERSION=$(echo "$TARGET_CORESIMULATOR_VERSION" | sed 's/\([0-9]*[.][0-9]*\).*/\1/')
# Compare versions to see if we got what we need
if [[ x"$TARGET_CORESIMULATOR_VERSION" == x"$CURRENT_CORESIMULATOR_VERSION" ]]; then
log "Found CoreSimulator.framework $CURRENT_CORESIMULATOR_VERSION (exactly $TARGET_CORESIMULATOR_VERSION is recommended)"
return
fi
if test -z $PROVISION_XCODE; then
# This is not a failure for now, until this logic has been tested thoroughly
warn "You should have exactly CoreSimulator.framework version $TARGET_CORESIMULATOR_VERSION (found $CURRENT_CORESIMULATOR_VERSION). Execute './system-dependencies.sh --provision-xcode' to install the expected version."
return
fi
# Just installing the package won't work, because there's a version check somewhere
# that prevents the macOS installer from downgrading, so remove the existing
# CoreSimulator.framework manually first.
log "Installing CoreSimulator.framework $CURRENT_CORESIMULATOR_VERSION..."
$SUDO rm -Rf /Library/Developer/PrivateFrameworks/CoreSimulator.framework
$SUDO installer -pkg "$CORESIMULATOR_PKG" -target /
CURRENT_CORESIMULATOR_VERSION=$(otool -L $CURRENT_CORESIMULATOR_PATH | grep "$CURRENT_CORESIMULATOR_PATH.*current version" | sed -e 's/.*current version//' -e 's/)//' -e 's/[[:space:]]//g')
log "Installed CoreSimulator.framework $CURRENT_CORESIMULATOR_VERSION successfully."
}
function check_specific_xcode () {
local XCODE_DEVELOPER_ROOT=`grep XCODE$1_DEVELOPER_ROOT= Make.config | sed 's/.*=//'`
local XCODE_VERSION=`grep XCODE$1_VERSION= Make.config | sed 's/.*=//'`
local XCODE_ROOT=$(dirname `dirname $XCODE_DEVELOPER_ROOT`)
local ENABLE_XAMARIN=$(grep -s ^ENABLE_XAMARIN= Make.config.local configure.inc | sed 's/.*=//')
if ! test -d $XCODE_DEVELOPER_ROOT; then
if ! test -z $PROVISION_XCODE; then
if ! test -z $ENABLE_XAMARIN; then
install_specific_xcode "$1" "$XCODE_DEVELOPER_ROOT"
else
fail "Automatic provisioning of Xcode is only supported for provisioning internal build bots."
fail "Please download and install Xcode $XCODE_VERSION here: https://developer.apple.com/downloads/index.action?name=Xcode"
fi
else
fail "You must install Xcode ($XCODE_VERSION) in $XCODE_ROOT. You can download Xcode $XCODE_VERSION here: https://developer.apple.com/downloads/index.action?name=Xcode"
fi
return
else
if is_at_least_version $XCODE_VERSION 5.0; then
if ! $XCODE_DEVELOPER_ROOT/usr/bin/xcodebuild -license check >/dev/null 2>&1; then
if ! test -z $PROVISION_XCODE; then
$SUDO $XCODE_DEVELOPER_ROOT/usr/bin/xcodebuild -license accept
else
fail "The license for Xcode $XCODE_VERSION has not been accepted. Execute '$SUDO $XCODE_DEVELOPER_ROOT/usr/bin/xcodebuild' to review the license and accept it."
return
fi
fi
fi
run_xcode_first_launch "$XCODE_VERSION" "$XCODE_DEVELOPER_ROOT"
fi
local XCODE_ACTUAL_VERSION=`/usr/libexec/PlistBuddy -c 'Print :CFBundleShortVersionString' "$XCODE_DEVELOPER_ROOT/../version.plist"`
# this is a hard match, having 4.5 when requesting 4.4 is not OK (but 4.4.1 is OK)
if [[ ! "x$XCODE_ACTUAL_VERSION" =~ "x$XCODE_VERSION" ]]; then
fail "You must install Xcode $XCODE_VERSION in $XCODE_ROOT (found $XCODE_ACTUAL_VERSION). You can download Xcode $XCODE_VERSION here: https://developer.apple.com/downloads/index.action?name=Xcode";
return
fi
ok "Found Xcode $XCODE_ACTUAL_VERSION in $XCODE_ROOT"
}
function check_xcode () {
if ! test -z $IGNORE_XCODE; then return; fi
# must have latest Xcode in /Applications/Xcode<version>.app
check_specific_xcode
install_coresimulator
local IOS_SDK_VERSION MACOS_SDK_VERSION WATCH_SDK_VERSION TVOS_SDK_VERSION
local XCODE_DEVELOPER_ROOT=`grep ^XCODE_DEVELOPER_ROOT= Make.config | sed 's/.*=//'`
IOS_SDK_VERSION=$(grep ^IOS_NUGET_OS_VERSION= Make.versions | sed -e 's/.*=//')
MACOS_SDK_VERSION=$(grep ^MACOS_NUGET_OS_VERSION= Make.versions | sed -e 's/.*=//')
WATCH_SDK_VERSION=$(grep ^WATCHOS_NUGET_OS_VERSION= Make.versions | sed -e 's/.*=//')
TVOS_SDK_VERSION=$(grep ^TVOS_NUGET_OS_VERSION= Make.versions | sed -e 's/.*=//')
local D=$XCODE_DEVELOPER_ROOT/Platforms/iPhoneSimulator.platform/Developer/SDKs/iPhoneSimulator${IOS_SDK_VERSION}.sdk
if test ! -d $D -a -z "$FAIL"; then
fail "The directory $D does not exist. If you've updated the Xcode location it means you also need to update IOS_SDK_VERSION in Make.config."
fi
local D=$XCODE_DEVELOPER_ROOT/Platforms/MacOSX.platform/Developer/SDKs/MacOSX${MACOS_SDK_VERSION}.sdk
if test ! -d $D -a -z "$FAIL"; then
fail "The directory $D does not exist. If you've updated the Xcode location it means you also need to update MACOS_SDK_VERSION in Make.config."
fi
local D=$XCODE_DEVELOPER_ROOT/Platforms/AppleTVOS.platform/Developer/SDKs/AppleTVOS${TVOS_SDK_VERSION}.sdk
if test ! -d $D -a -z "$FAIL"; then
fail "The directory $D does not exist. If you've updated the Xcode location it means you also need to update TVOS_SDK_VERSION in Make.config."
fi
local D=$XCODE_DEVELOPER_ROOT/Platforms/WatchOS.platform/Developer/SDKs/WatchOS${WATCH_SDK_VERSION}.sdk
if test ! -d $D -a -z "$FAIL"; then
fail "The directory $D does not exist. If you've updated the Xcode location it means you also need to update WATCH_SDK_VERSION in Make.config."
fi
}
function check_mono () {
if ! test -z $IGNORE_MONO; then return; fi
MONO_VERSION_FILE=/Library/Frameworks/Mono.framework/Versions/Current/VERSION
if ! /Library/Frameworks/Mono.framework/Commands/mono --version 2>/dev/null >/dev/null; then
if ! test -z $PROVISION_MONO; then
install_mono
else
fail "You must install the Mono MDK. Download URL: $MIN_MONO_URL"
return
fi
elif ! test -e $MONO_VERSION_FILE; then
if ! test -z $PROVISION_MONO; then
install_mono
else
fail "Could not find Mono's VERSION file, you must install the Mono MDK. Download URL: $MIN_MONO_URL"
return
fi
fi
MIN_MONO_VERSION=`grep MIN_MONO_VERSION= Make.config | sed 's/.*=//'`
MAX_MONO_VERSION=`grep MAX_MONO_VERSION= Make.config | sed 's/.*=//'`
ACTUAL_MONO_VERSION=`cat $MONO_VERSION_FILE`
if ! is_at_least_version $ACTUAL_MONO_VERSION $MIN_MONO_VERSION; then
if ! test -z $PROVISION_MONO; then
install_mono
ACTUAL_MONO_VERSION=`cat $MONO_VERSION_FILE`
else
MIN_MONO_URL=$(grep ^MIN_MONO_URL= Make.config | sed 's/.*=//')
fail "You must have at least Mono $MIN_MONO_VERSION, found $ACTUAL_MONO_VERSION. Download URL: $MIN_MONO_URL"
return
fi
elif [[ "$ACTUAL_MONO_VERSION" == "$MAX_MONO_VERSION" ]]; then
: # this is ok
elif is_at_least_version $ACTUAL_MONO_VERSION $MAX_MONO_VERSION; then
if ! test -z $PROVISION_MONO; then
install_mono
ACTUAL_MONO_VERSION=`cat $MONO_VERSION_FILE`
else
fail "Your mono version is too new, max version is $MAX_MONO_VERSION, found $ACTUAL_MONO_VERSION."
fail "You may edit Make.config and change MAX_MONO_VERSION to your actual version to continue the"
fail "build (unless you're on a release branch). Once the build completes successfully, please"
fail "commit the new MAX_MONO_VERSION value."
fail "Alternatively you can ${COLOR_MAGENTA}export IGNORE_MONO=1${COLOR_RED} to skip this check."
return
fi
fi
if ! which mono > /dev/null 2>&1; then
fail "Mono is not in PATH. You must add '/Library/Frameworks/Mono.framework/Versions/Current/Commands' to PATH. Current PATH is: $PATH".
return
fi
ok "Found Mono $ACTUAL_MONO_VERSION (at least $MIN_MONO_VERSION and not more than $MAX_MONO_VERSION is required)"
}
function install_autoconf () {
if ! brew --version >& /dev/null; then
fail "Asked to install autoconf, but brew is not installed."
return
fi
brew install autoconf
}
function install_python3 () {
if ! brew --version >& /dev/null; then
fail "Asked to install python3, but brew is not installed."
return
fi
ok "Installing ${COLOR_BLUE}python3${COLOR_RESET}..."
brew install python3
}
function install_libtool () {
if ! brew --version >& /dev/null; then
fail "Asked to install libtool, but brew is not installed."
return
fi
brew install libtool
}
function install_automake () {
if ! brew --version >& /dev/null; then
fail "Asked to install automake, but brew is not installed."
return
fi
brew install automake
}
function check_autotools () {
if ! test -z $IGNORE_AUTOTOOLS; then return; fi
IFStmp=$IFS
IFS='
'
if AUTOCONF_VERSION=($(autoconf --version 2>/dev/null)); then
ok "Found ${AUTOCONF_VERSION[0]} (no specific version is required)"
elif ! test -z $PROVISION_AUTOTOOLS; then
install_autoconf
else
fail "You must install autoconf, read the README.md for instructions"
fi
if ! LIBTOOL=$(which glibtool 2>/dev/null); then
LIBTOOL=$(which libtool)
fi
if LIBTOOL_VERSION=($($LIBTOOL --version 2>/dev/null )); then
ok "Found ${LIBTOOL_VERSION[0]} (no specific version is required)"
elif ! test -z $PROVISION_AUTOTOOLS; then
install_libtool
else
fail "You must install libtool, read the README.md for instructions"
fi
if AUTOMAKE_VERSION=($(automake --version 2>/dev/null)); then
ok "Found ${AUTOMAKE_VERSION[0]} (no specific version is required)"
elif ! test -z $PROVISION_AUTOTOOLS; then
install_automake
else
fail "You must install automake, read the README.md for instructions"
fi
IFS=$IFS_tmp
}
function check_python3 () {
if ! test -z $IGNORE_PYTHON3; then return; fi
IFStmp=$IFS
IFS='
'
if PYTHON3_VERSION=$(python3 --version 2>/dev/null); then
ok "Found $PYTHON3_VERSION (no specific version is required)"
elif ! test -z $PROVISION_PYTHON3; then
install_python3
else
fail "You must install python3. The easiest way is to use homebrew, and execute ${COLOR_MAGENTA}brew install python3${COLOR_RESET}."
fi
IFS=$IFS_tmp
}
function check_osx_version () {
if ! test -z $IGNORE_OSX; then return; fi
MIN_OSX_BUILD_VERSION=`grep MIN_OSX_BUILD_VERSION= Make.config | sed 's/.*=//'`
ACTUAL_OSX_VERSION=$(sw_vers -productVersion)
if ! is_at_least_version $ACTUAL_OSX_VERSION $MIN_OSX_BUILD_VERSION; then
fail "You must have at least OSX $MIN_OSX_BUILD_VERSION (found $ACTUAL_OSX_VERSION)"
return
fi
ok "Found OSX $ACTUAL_OSX_VERSION (at least $MIN_OSX_BUILD_VERSION is required)"
}
function check_checkout_dir () {
# use apple script to get the possibly translated special folders and check that we are not a subdir
for special in documents downloads desktop; do
path=$(osascript -e "set result to POSIX path of (path to $special folder as string)")
if [[ $PWD == $path* ]]; then
fail "Your checkout is under $path which is a special path. This can result in problems running the tests."
fi
done
ok "Checkout location will not result in test problems."
}
function install_cmake () {
if ! brew --version >& /dev/null; then
fail "Asked to install cmake, but brew is not installed."
return
fi
brew install cmake
}
function check_cmake () {
if ! test -z $IGNORE_CMAKE; then return; fi
local MIN_CMAKE_VERSION=`grep MIN_CMAKE_VERSION= Make.config | sed 's/.*=//'`
local CMAKE_URL=`grep CMAKE_URL= Make.config | sed 's/.*=//'`
if ! cmake --version &> /dev/null; then
if ! test -z $PROVISION_CMAKE; then
install_cmake
else
fail "You must install CMake ($CMAKE_URL)"
fi
return
fi
ACTUAL_CMAKE_VERSION=$(cmake --version | grep "cmake version" | sed 's/cmake version //')
if ! is_at_least_version $ACTUAL_CMAKE_VERSION $MIN_CMAKE_VERSION; then
fail "You must have at least CMake $MIN_CMAKE_VERSION (found $ACTUAL_CMAKE_VERSION)"
return
fi
ok "Found CMake $ACTUAL_CMAKE_VERSION (at least $MIN_CMAKE_VERSION is required)"
}
function install_7z () {
if ! brew --version >& /dev/null; then
fail "Asked to install 7z, but brew is not installed."
return
fi
brew install p7zip
}
function check_7z () {
if ! test -z $IGNORE_7Z; then return; fi
if ! 7z &> /dev/null; then
if ! test -z $PROVISION_7Z; then
install_7z
else
fail "You must install 7z (no specific version is required)"
fi
return
fi
ok "Found 7z (no specific version is required)"
}
function check_homebrew ()
{
if ! test -z $IGNORE_HOMEBREW; then return; fi
IFStmp=$IFS
IFS='
'
if HOMEBREW_VERSION=($(brew --version 2>/dev/null)); then
ok "Found Homebrew ($HOMEBREW_VERSION)"
elif ! test -z $PROVISION_HOMEBREW; then
log "Installing Homebrew..."
/usr/bin/ruby -e "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/master/install)"
HOMEBREW_VERSION=($(brew --version 2>/dev/null))
log "Installed Homebrew ($HOMEBREW_VERSION)"
else
warn "Could not find Homebrew. Homebrew is required to auto-provision some dependencies (autotools, cmake), but not required otherwise."
fi
IFS=$IFS_tmp
}
function install_objective_sharpie () {
local SHARPIE_URL=$(grep MIN_SHARPIE_URL= Make.config | sed 's/.*=//')
local MIN_SHARPIE_VERSION=$(grep MIN_SHARPIE_VERSION= Make.config | sed 's/.*=//')
if test -z "$SHARPIE_URL"; then
fail "No MIN_SHARPIE_URL set in Make.config, cannot provision Objective Sharpie"
return
fi
mkdir -p "$PROVISION_DOWNLOAD_DIR"
log "Downloading Objective Sharpie $MIN_SHARPIE_VERSION from $SHARPIE_URL to $PROVISION_DOWNLOAD_DIR..."
local SHARPIE_NAME=$(basename "$SHARPIE_URL")
local SHARPIE_PKG=$PROVISION_DOWNLOAD_DIR/$SHARPIE_NAME
curl -L "$SHARPIE_URL" > "$SHARPIE_PKG"
log "Installing Objective-Sharpie $MIN_SHARPIE_VERSION from $SHARPIE_URL..."
sudo installer -pkg "$SHARPIE_PKG" -target /
rm -f "$SHARPIE_PKG"
}
function check_objective_sharpie () {
if ! test -z $IGNORE_SHARPIE; then return; fi
SHARPIE_URL=$(grep MIN_SHARPIE_URL= Make.config | sed 's/.*=//')
MIN_SHARPIE_VERSION=$(grep MIN_SHARPIE_VERSION= Make.config | sed 's/.*=//')
MAX_SHARPIE_VERSION=$(grep MAX_SHARPIE_VERSION= Make.config | sed 's/.*=//')
if ! test -f /Library/Frameworks/ObjectiveSharpie.framework/Versions/Current/Version; then
if ! test -z "$PROVISION_SHARPIE"; then
install_objective_sharpie
ACTUAL_SHARPIE_VERSION=$(cat /Library/Frameworks/ObjectiveSharpie.framework/Versions/Current/Version)
else
if test -z $OPTIONAL_SHARPIE; then
fail "You must install Objective Sharpie, at least $MIN_SHARPIE_VERSION (no Objective Sharpie found). You can download it from $SHARPIE_URL"
fail "Alternatively you can ${COLOR_MAGENTA}export IGNORE_SHARPIE=1${COLOR_RED} to skip this check."
else
warn "You do not have Objective Sharpie installed (should be at least $MIN_SHARPIE_VERSION). You can download it from $SHARPIE_URL"
fi
return
fi
else
ACTUAL_SHARPIE_VERSION=$(cat /Library/Frameworks/ObjectiveSharpie.framework/Versions/Current/Version)
if ! is_at_least_version "$ACTUAL_SHARPIE_VERSION" "$MIN_SHARPIE_VERSION"; then
if ! test -z "$PROVISION_SHARPIE"; then
install_objective_sharpie
ACTUAL_SHARPIE_VERSION=$(cat /Library/Frameworks/ObjectiveSharpie.framework/Versions/Current/Version)
else
if test -z $OPTIONAL_SHARPIE; then
fail "You must have at least Objective Sharpie $MIN_SHARPIE_VERSION, found $ACTUAL_SHARPIE_VERSION. You can download it from $SHARPIE_URL"
fail "Alternatively you can ${COLOR_MAGENTA}export IGNORE_SHARPIE=1${COLOR_RED} to skip this check."
else
warn "You do not have have at least Objective Sharpie $MIN_SHARPIE_VERSION (found $ACTUAL_SHARPIE_VERSION). You can download it from $SHARPIE_URL"
fi
return
fi
elif [[ "$ACTUAL_SHARPIE_VERSION" == "$MAX_SHARPIE_VERSION" ]]; then
: # this is ok
elif is_at_least_version "$ACTUAL_SHARPIE_VERSION" "$MAX_SHARPIE_VERSION"; then
if ! test -z "$PROVISION_SHARPIE"; then
install_objective_sharpie
ACTUAL_SHARPIE_VERSION=$(cat /Library/Frameworks/ObjectiveSharpie.framework/Versions/Current/Version)
else
if test -z $OPTIONAL_SHARPIE; then
fail "Your Objective Sharpie version is too new, max version is $MAX_SHARPIE_VERSION, found $ACTUAL_SHARPIE_VERSION. We recommend you download $SHARPIE_URL"
fail "Alternatively you can ${COLOR_MAGENTA}export IGNORE_SHARPIE=1${COLOR_RED} to skip this check."
else
warn "You do not have have at most Objective Sharpie $MAX_SHARPIE_VERSION (found $ACTUAL_SHARPIE_VERSION). We recommend you download $SHARPIE_URL"
fi
return
fi
fi
fi
if test -z $OPTIONAL_SHARPIE; then
ok "Found Objective Sharpie $ACTUAL_SHARPIE_VERSION (at least $MIN_SHARPIE_VERSION and not more than $MAX_SHARPIE_VERSION is required)"
else
ok "Found Objective Sharpie $ACTUAL_SHARPIE_VERSION (at least $MIN_SHARPIE_VERSION and not more than $MAX_SHARPIE_VERSION is recommended)"
fi
}
function check_simulators ()
{
if test -n "$IGNORE_SIMULATORS"; then return; fi
local EXTRA_SIMULATORS
local XCODE
EXTRA_SIMULATORS=$(grep ^EXTRA_SIMULATORS= Make.config | sed 's/.*=//')
XCODE=$(dirname "$(dirname "$(grep ^XCODE_DEVELOPER_ROOT= Make.config | sed 's/.*=//')")")
if ! make -C tools/siminstaller >/dev/null; then
warn "Can't check if simulators are available, because siminstaller failed to build."
return
fi
if ! test -d "$XCODE"; then
# can't test unless Xcode is present
warn "Can't check if simulators are available unless Xcode is already installed."
return
fi
IFS=', ' read -r -a SIMS <<< "$EXTRA_SIMULATORS"
arraylength=${#SIMS[@]}
for (( i=1; i<arraylength+1; i++ )); do
SIMS[$i-1]="--install=${SIMS[$i-1]}"
done
if ! FAILED_SIMULATORS=$(mono --debug tools/siminstaller/bin/Debug/siminstaller.exe -q --xcode "$XCODE" --only-check "${SIMS[@]}"); then
local action=warn
if test -z $OPTIONAL_SIMULATORS; then
action=fail
fi
if [[ "$FAILED_SIMULATORS" =~ "Unknown simulators:" ]]; then
$action "${FAILED_SIMULATORS}"
$action " If you just updated the Xcode version, it's possible Apple stopped shipping these simulators with the new version of Xcode."
$action " If that's the case, you can list the available simulators with ${COLOR_MAGENTA}make -C tools/siminstaller print-simulators --xcode $XCODE${COLOR_RESET},"
$action " and then update the ${COLOR_MAGENTA}MIN_<OS>_SIMULATOR_VERSION${COLOR_RESET} and ${COLOR_MAGENTA}EXTRA_SIMULATORS${COLOR_RESET} variables in Make.config to the earliest available simulators."
$action " Another possibility is that Apple is not shipping any simulators (yet?) for the new version of Xcode (if the previous list shows no simulators)."
else
if ! test -z $PROVISION_SIMULATORS; then
if ! mono --debug tools/siminstaller/bin/Debug/siminstaller.exe -q --xcode "$XCODE" "${SIMS[@]}"; then
$action "Failed to install extra simulators."
else
ok "Extra simulators installed successfully: '${FAILED_SIMULATORS//$'\n'/', '}'"
fi
else
$action "The simulators '${FAILED_SIMULATORS//$'\n'/', '}' are not installed or need to be upgraded."
fi
fi
else
ok "Found all extra simulators: ${EXTRA_SIMULATORS// /, }"
fi
}
echo "Checking system..."
check_osx_version
check_checkout_dir
check_xcode
check_homebrew
check_autotools
check_python3
check_mono
check_cmake
check_7z
check_objective_sharpie
check_simulators
if test -z "$IGNORE_DOTNET"; then
ok "Installed .NET SDKs:"
(IFS=$'\n'; for i in $(/usr/local/share/dotnet/dotnet --list-sdks); do log "$i"; done)
fi
if test -z $FAIL; then
echo "System check succeeded"
else
echo "System check failed"
exit 1
fi