Skip to content

Instantly share code, notes, and snippets.

Show Gist options
  • Save mapedd/4151222 to your computer and use it in GitHub Desktop.
Save mapedd/4151222 to your computer and use it in GitHub Desktop.
Automatic TestFlight/HockeyApp Upload XCode Script
#!/bin/bash
#
# (Above line comes out when placing in Xcode scheme)
#
# Inspired by original script by incanus:
# https://gist.github.com/1186990
#
# Rewritten by martijnthe:
# https://gist.github.com/1379127
#
# Rewritten by c0diq:
# https://gist.github.com/2213571
#
# - Using Xcode's environment variables instead of 'guessing' what archive we need to upload
# - AppleScript dialogs for basic user interaction (upload yes/no, select code signing identity, enter release notes, )
# - Supports both HockeyApp & TestFlight
#
#
# =====================================================================================================================
# *** BASIC CONFIGURATION:
#
# Find your API_TOKEN at: https://testflightapp.com/account/
TESTFLIGHT_API_TOKEN="xxxxx"
#
# Find your TEAM_TOKEN at: https://testflightapp.com/dashboard/team/edit/
TESTFLIGHT_TEAM_TOKEN="xxxx"
#
# Find your API_TOKEN at: https://rink.hockeyapp.net/manage/auth_tokens
HOCKEYAPP_API_TOKEN="xxxx"
#
# Distribution List names, comma separated (quoted) string, e.g. "DevTeam,Clients,BetaTesters":
DISTRIBUTION_LISTS="DevTeam,QA,BetaTesters,Investors,Press"
#
# Default selection of Distribution List(s), e.g. "DevTeam,Clients":
DISTRIBUTION_LISTS_DEFAULT_SELECTION="DevTeam,QA"
#
# Default selection for the Notify team members dialog ("True" -> Notify team members, "False" -> Don't notify):
DEFAULT_NOTIFY_VALUE="True"
#
# Upload service ("HockeyApp" or "TestFlight")
UPLOAD_SERVICE="HockeyApp"
#
# =====================================================================================================================
# *** OPTIONAL CONFIGURATION:
#
# Uncomment this line to skip the resigning / re-provisioning steps:
# The application is expected to be already provisioned and signed.
#
SKIP_RESIGNING_AND_REPROVISIONING="YES"
#
# Uncomment this line to skip the Release Notes input step and to set a default value:
#
DEFAULT_RELEASE_NOTES="Just another test version."
#
# Uncomment this line to skip the Distribution Lists input step and to use the default value:
#
SKIP_DISTRIBUTION_LISTS="YES"
#
# Uncomment this line to skip the Notify? input step and to use the default value:
#
SKIP_NOTIFY="YES"
#
# Uncomment this line to enable loading Console.app
#
#SHOW_DEBUG_CONSOLE="YES"
#
# Uncomment this line to disable opening the browser with the TestFlight dashboard at the end of the ride
#
#DISABLE_OPEN_DASHBOARD="YES"
#
# =====================================================================================================================
# Setup logging stuff...
LOG="/tmp/package.log"
/bin/rm -f $LOG
echo "Starting Upload Process" > $LOG
if [ "$SHOW_DEBUG_CONSOLE" = "YES" ]; then
/usr/bin/open -a /Applications/Utilities/Console.app $LOG
fi
echo >> $LOG
echo "CODE_SIGN_IDENTITY: $CODE_SIGN_IDENTITY" >> $LOG
echo "WRAPPER_NAME: $WRAPPER_NAME" >> $LOG
echo "ARCHIVE_DSYMS_PATH: $ARCHIVE_DSYMS_PATH" >> $LOG
echo "ARCHIVE_PRODUCTS_PATH: $ARCHIVE_PRODUCTS_PATH" >> $LOG
echo "DWARF_DSYM_FILE_NAME: $DWARF_DSYM_FILE_NAME" >> $LOG
echo "INSTALL_PATH: $INSTALL_PATH" >> $LOG
echo "PRODUCT_NAME: $PRODUCT_NAME" >> $LOG
# Do some existence checks for the build settings that this script depends on:
if [ "$CODE_SIGN_IDENTITY" = "" -o "$WRAPPER_NAME" = "" -o "$ARCHIVE_DSYMS_PATH" = "" -o "$ARCHIVE_PRODUCTS_PATH" = "" -o "$DWARF_DSYM_FILE_NAME" = "" -o "$INSTALL_PATH" = "" ]; then
osascript -e "tell application \"Xcode\"" -e "display dialog \"It looks like we're missing build settings.\n\nYou can fix this by editing your scheme's Run Script action and selecting the appropriate target from the 'Provide build settings from...' drop down menu.\" buttons {\"OK\"} default button \"OK\" with icon stop" -e "end tell"
exit 1
fi
# Build paths from build settings environment vars:
DSYM_PATH="$ARCHIVE_DSYMS_PATH"
APP="$ARCHIVE_PRODUCTS_PATH/$INSTALL_PATH/$WRAPPER_NAME"
# Ask if we need to proceed to upload to TestFlight using an AppleScript dialog in Xcode:
SHOULD_UPLOAD=`osascript -e "tell application \"Xcode\"" -e "set noButton to \"No, Thanks\"" -e "set yesButton to \"Yes!\"" -e "set upload_dialog to display dialog \"Do you want to upload this build to $UPLOAD_SERVICE?\" buttons {noButton, yesButton} default button yesButton with icon 1" -e "set button to button returned of upload_dialog" -e "if button is equal to yesButton then" -e "return 1" -e "else" -e "return 0" -e "end if" -e "end tell"`
# Exit this script if the user indicated we shouldn't upload:
if [ "$SHOULD_UPLOAD" = "0" ]; then
echo "User indicated not to upload this archive. Quitting." >> $LOG
exit 0
fi #SHOULD_UPLOAD
# Now onto selecting signing identity and provisioning profiles...
if [ "$SKIP_RESIGNING_AND_REPROVISIONING" != "YES" ]; then
echo >> $LOG
echo "Finding signing identities..." >> $LOG
# Get all the user's code signing identities. Filter the response to get a neat list of quoted strings:
SIGNING_IDENTITIES_LIST=`security find-identity -v -p codesigning | egrep -oE '"[^"]+"'`
echo >> $LOG
echo "Found identities:" >> $LOG
echo "$SIGNING_IDENTITIES_LIST" >> $LOG
# Replace the newline characters in the list with commas and remove the last comma:
SIGNING_IDENTITIES_COMMA_SEPARATED_LIST=`echo "$SIGNING_IDENTITIES_LIST" | tr '\n' ',' | sed 's/,$//'`
# Present dialog with list of code signing identites and let the user pick one. The identity that from the build settings is selected by default.
CODE_SIGN_IDENTITY=`osascript -e "tell application \"Xcode\"" -e "set selected_identity to {choose from list {$SIGNING_IDENTITIES_COMMA_SEPARATED_LIST} with prompt \"Choose code signing identity:\" default items {\"$CODE_SIGN_IDENTITY\"}}" -e "end tell" -e "return selected_identity"`
echo >> $LOG
if [ "$CODE_SIGN_IDENTITY" = "false" ]; then
echo "User cancelled." >> $LOG
exit 0
fi
echo "Selected code signing identity:" >> $LOG
echo "$CODE_SIGN_IDENTITY" >> $LOG
# Now onto the provisioning profiles...
TEMP_MOBILEPROVISION_PLIST_PATH=/tmp/mobileprovision.plist
TEMP_CERTIFICATE_PATH=/tmp/certificate.cer
MOBILEDEVICE_PROVISIONING_PROFILES_FOLDER="${HOME}/Library/MobileDevice/Provisioning Profiles"
MATCHING_PROFILES_LIST=""
MATCHING_NAMES_LIST=""
cd "$MOBILEDEVICE_PROVISIONING_PROFILES_FOLDER"
for MOBILEPROVISION_FILENAME in *.mobileprovision
do
# Use sed to rid the signature data that is padding the plist and store clean plist to temp file:
sed -n '/<!DOCTYPE plist/,/<\/plist>/ p' \
< "$MOBILEPROVISION_FILENAME" \
> "$TEMP_MOBILEPROVISION_PLIST_PATH"
# The plist root dict contains an array called 'DeveloperCertificates'. It seems to contain one element with the certificate data. Dump to temp file:
/usr/libexec/PlistBuddy -c 'Print DeveloperCertificates:0' $TEMP_MOBILEPROVISION_PLIST_PATH > $TEMP_CERTIFICATE_PATH
# Get the common name (CN) from the certificate (regex capture between 'CN=' and '/OU'):
MOBILEPROVISION_IDENTITY_NAME=`openssl x509 -inform DER -in $TEMP_CERTIFICATE_PATH -subject -noout | perl -n -e '/CN=(.+)\/OU/ && print "$1"'`
if [ "$CODE_SIGN_IDENTITY" = "$MOBILEPROVISION_IDENTITY_NAME" ]; then
# Yay, this mobile provisioning profile matches up with the selected signing identity, let's continue...
# Get the name of the provisioning profile:
MOBILEPROVISION_PROFILE_NAME=`/usr/libexec/PlistBuddy -c 'Print Name' $TEMP_MOBILEPROVISION_PLIST_PATH`
MATCHING_PROFILES_LIST=`echo "$MATCHING_PROFILES_LIST\"$MOBILEPROVISION_PROFILE_NAME\"|\"$MOBILEPROVISION_FILENAME\","`
MATCHING_NAMES_LIST=`echo "$MATCHING_NAMES_LIST\"$MOBILEPROVISION_PROFILE_NAME\","`
fi
done
# Remove last comma:
MATCHING_NAMES_LIST=`echo "$MATCHING_NAMES_LIST" | sed 's/,$//'`
# Remove last pipe:
MATCHING_PROFILES_LIST=`echo "$MATCHING_PROFILES_LIST" | sed 's/,$//'`
echo >> $LOG
echo "Matching provisioning profiles:" >> $LOG
echo "$MATCHING_PROFILES_LIST" >> $LOG
# Add the (default) value for using the existing embedded.mobileprovision:
USE_EXISTING_PROFILE="\"Don't overwrite the current provisioning profile\""
MATCHING_NAMES_LIST=`echo "$USE_EXISTING_PROFILE,$MATCHING_NAMES_LIST"`
# Present dialog with list of matching provisioning profiles and let the user pick one.
SELECTED_PROFILE_NAME=`osascript -e "tell application \"Xcode\"" -e "set selected_profile to {choose from list {$MATCHING_NAMES_LIST} with prompt \"Choose provisioning profile:\" default items {$USE_EXISTING_PROFILE}}" -e "end tell" -e "return selected_profile"`
if [ "$SELECTED_PROFILE_NAME" = "false" ]; then
echo "User cancelled." >> $LOG
exit 0
fi
SELECTED_PROFILE_FILE=`echo "$MATCHING_PROFILES_LIST" | tr "," "\n" | grep "$SELECTED_PROFILE_NAME" | tr "|" "\n" | sed -n 2p`
echo >> $LOG
echo "Selected provisioning profile:" >> $LOG
if [ "$SELECTED_PROFILE_FILE" != "" ]; then
# Remove quotes (needed before for AppleScript):
SELECTED_PROFILE_FILE=`echo "$SELECTED_PROFILE_FILE" | tr -d "\""`
EMBED_PROFILE="$MOBILEDEVICE_PROVISIONING_PROFILES_FOLDER/$SELECTED_PROFILE_FILE"
echo "$SELECTED_PROFILE_FILE : $SELECTED_PROFILE_NAME" >> $LOG
echo "$EMBED_PROFILE" >> $LOG
else
EMBED_PROFILE="$APP/embedded.mobileprovision"
echo "None selected. Keeping existing embedded.mobileprovision file:" >> $LOG
echo "$EMBED_PROFILE" >> $LOG
fi
fi #SKIP_RESIGNING_AND_REPROVISIONING
# Now onto the Release Notes...
if [ "$DEFAULT_RELEASE_NOTES" = "" ]; then
# Bring up an AppleScript dialog in Xcode to enter the Release Notes for this (beta) build:
NOTES=`osascript -e "tell application \"Xcode\"" -e "set notes_dialog to display dialog \"Please provide some release notes:\nHint: use Ctrl-J for New Line.\" default answer \"\" buttons {\"Next\"} default button \"Next\" with icon 1" -e "set notes to text returned of notes_dialog" -e "end tell" -e "return notes"`
else
NOTES="$DEFAULT_RELEASE_NOTES"
fi #DEFAULT_RELEASE_NOTES
# add default if user entered nothing otherwise it will fail
if [ "$NOTES" = "" ]; then
NOTES="Just another test"
fi
echo "Added release notes:" >> $LOG
echo "$NOTES" >> $LOG
# Now onto selecting the Distribution Lists...
if [ "$SKIP_DISTRIBUTION_LISTS" != "YES" ]; then
DISTRIBUTION_LISTS_QUOTED=`echo "$DISTRIBUTION_LISTS" | tr "," "\n" | sed 's/$/"/' | sed 's/^/"/' | tr "\n" "," | sed 's/,$//'`
DISTRIBUTION_LISTS_DEFAULT_SELECTION_QUOTED=`echo "$DISTRIBUTION_LISTS_DEFAULT_SELECTION" | tr "," "\n" | sed 's/$/"/' | sed 's/^/"/' | tr "\n" "," | sed 's/,$//'`
SELECTED_DISTRIBUTION_LISTS_QUOTED=`osascript -e "tell application \"Xcode\"" -e "set selected_profile to {choose from list {$DISTRIBUTION_LISTS_QUOTED} with prompt \"Choose Distribution List(s):\" default items {$DISTRIBUTION_LISTS_DEFAULT_SELECTION_QUOTED} with multiple selections allowed}" -e "end tell" -e "return selected_profile"`
if [ "$SELECTED_DISTRIBUTION_LISTS_QUOTED" = "false" ]; then
echo "User cancelled." >> $LOG
exit 0
fi
SELECTED_DISTRIBUTION_LISTS=`echo "$SELECTED_DISTRIBUTION_LISTS_QUOTED" | sed 's/, /,/'`
else
SELECTED_DISTRIBUTION_LISTS="$DISTRIBUTION_LISTS_DEFAULT_SELECTION"
fi #SKIP_DISTRIBUTION_LISTS
echo >> $LOG
echo "Selected Distribution Lists: '$SELECTED_DISTRIBUTION_LISTS'" >> $LOG
if [ "$SKIP_NOTIFY" != "YES" ]; then
# Ask if we need to notify the permitted team members of the new build:
if [ "$DEFAULT_NOTIFY_VALUE" = "True" ]; then
SELECTED_NOTIFY_BUTTON="Yes, Please!"
else
SELECTED_NOTIFY_BUTTON="No, Thanks"
fi
SHOULD_NOTIFY=`osascript -e "tell application \"Xcode\"" -e "set noButton to \"No, Thanks\"" -e "set yesButton to \"Yes, Please!\"" -e "set upload_dialog to display dialog \"Do you want to have your team members notified by TestFlight about this new version?\" buttons {noButton, yesButton} default button yesButton with icon 1" -e "set button to button returned of upload_dialog" -e "if button is equal to yesButton then" -e "return \"True\"" -e "else" -e "return \"False\"" -e "end if" -e "end tell"`
else
SHOULD_NOTIFY="$DEFAULT_NOTIFY_VALUE"
fi
echo >> $LOG
echo "Notify: $SHOULD_NOTIFY" >> $LOG
# Now onto creating the IPA...
echo >> $LOG
echo "Creating IPA at /tmp/$PRODUCT_NAME.ipa ..." >> $LOG
/bin/rm -f /tmp/app.ipa >> $LOG 2>&1
if [ "$SKIP_RESIGNING_AND_REPROVISIONING" != "YES" ]; then
/usr/bin/xcrun -sdk iphoneos PackageApplication "${APP}" -o "/tmp/$PRODUCT_NAME.ipa" --embed "$EMBED_PROFILE" --sign "${CODE_SIGN_IDENTITY}" >> $LOG 2>&1
else
/usr/bin/xcrun -sdk iphoneos PackageApplication "${APP}" -o "/tmp/$PRODUCT_NAME.ipa" >> $LOG 2>&1
fi #SKIP_RESIGNING_AND_REPROVISIONING
if [ "$?" -ne 0 ]; then
echo "There were errors creating IPA." >> $LOG
osascript -e "tell application \"Xcode\"" -e "display dialog \"There were errors creating IPA... Check $LOG\" buttons {\"OK\"} with icon stop" -e "end tell"
/usr/bin/open -a /Applications/Utilities/Console.app $LOG
exit 1
fi
echo "Done creating IPA ..." >> $LOG
# Now onto creating the zipped .dSYM debugging symbols
echo >> $LOG
echo "Zipping $DSYM_PATH ($DWARF_DSYM_FILE_NAME) to /tmp/$DWARF_DSYM_FILE_NAME.zip..." >> $LOG
/bin/rm -f "/tmp/$DWARF_DSYM_FILE_NAME.zip"
pushd "$DSYM_PATH"
/usr/bin/zip -r "/tmp/$DWARF_DSYM_FILE_NAME.zip" "$DWARF_DSYM_FILE_NAME"
popd
echo "Done zipping ..." >> $LOG
# Now onto the upload itself
echo >> $LOG
echo "Uploading ... " >> $LOG
pushd "/tmp"
if [ "$UPLOAD_SERVICE" = "HockeyApp" ]; then
/usr/bin/curl "https://rink.hockeyapp.net/api/2/apps" \
-F status="2" \
-F notify="$SHOULD_NOTIFY" \
-F notes="$NOTES" \
-F ipa=@"$PRODUCT_NAME.ipa" \
-F dsym=@"$DWARF_DSYM_FILE_NAME.zip" \
-H "X-HockeyAppToken: $HOCKEYAPP_API_TOKEN" >> $LOG 2>&1
else
/usr/bin/curl "https://testflightapp.com/api/builds.json" \
-F file=@"$PRODUCT_NAME.ipa" \
-F dsym=@"$DWARF_DSYM_FILE_NAME.zip" \
-F api_token="$TESTFLIGHT_API_TOKEN" \
-F team_token="$TESTFLIGHT_TEAM_TOKEN" \
-F replace=True \
-F notify="$SHOULD_NOTIFY" \
-F distribution_lists="$SELECTED_DISTRIBUTION_LISTS" \
-F notes="$NOTES" >> $LOG 2>&1
fi
popd
if [ "$?" -ne 0 ]; then
echo "There were errors uploading." >> $LOG
osascript -e "tell application \"Xcode\"" -e "display dialog \"There were errors uploading... Check $LOG\" buttons {\"OK\"} with icon stop" -e "end tell"
/usr/bin/open -a /Applications/Utilities/Console.app $LOG
exit 1
fi
echo >> $LOG
echo "Uploaded to TestFlight!" >> $LOG
osascript -e "tell application \"Xcode\"" -e "display dialog \"Upload to $UPLOAD_SERVICE done!\" buttons {\"OK\"} default button \"OK\"" -e "end tell"
if [ "$DISABLE_OPEN_DASHBOARD" != "YES" ]; then
echo >> $LOG
if [ "$UPLOAD_SERVICE" = "HockeyApp" ]; then
echo "Opening https://rink.hockeyapp.net/manage/dashboard now..." >> $LOG
/usr/bin/open "https://rink.hockeyapp.net/manage/dashboard"
else
echo "Opening https://testflightapp.com/dashboard/builds/ now..." >> $LOG
/usr/bin/open "https://testflightapp.com/dashboard/builds/"
fi
fi
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment