blob: 826c282d18c00fdec6ca7a2f5232106d9a285a59 [file] [log] [blame]
#!/bin/bash
# Build iOS codelabs.
#
# If --verbose (-v) is specified, print the progress of each build.
#
# If xcpretty is installed (https://github.com/supermarin/xcpretty) then it will
# be used in verbose mode.
#
# Other options can be found by using -h.
readonly SCRIPTS_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
readonly ROOT_DIR="$SCRIPTS_DIR/.."
readonly CURRENT_DIR="$PWD"
# Given a path to an Xcode log file in $1, exit with status 0 if it looks like
# the failure is expected and can be ignored, or exit with non-zero status
# otherwise.
function is_expected_failure() {
# A test target was specified with the 'build' command.
grep --quiet "is not configured for Running" "$1"
}
# Test if the xcpretty command is available.
#
# Returns exit status zero if available and non-zero if not.
function is_xcpretty_available() {
xcpretty > /dev/null 2>&1
# Exit code 127 is the standard "command not found" exit code.
if [ $? -eq 127 ]; then
return 1
else
return 0
fi
}
# Check that flags have arguments.
#
# Exits if the flag does not have an argument.
function check_arguments() {
if [ -z $1 ]; then
echo "Argument must be provided to flag."
echo "$(basename $0) -h for help"
exit 1
fi
}
# Checks that the argument is a valid reference.
#
# Exits if the reference is not valid.
function check_reference() {
reference=$(git ls-remote https://github.com/material-components/material-components-ios.git $1 | wc -l)
if [ $reference -eq 0 ]; then
echo "Unable to find $1 on GitHub."
echo "$(basename $0) -h for help"
exit 1
fi
}
# Checks that the argument is a valid commit.
#
# Exits if the commit is not valid.
function check_commit() {
if [ ! $(git cat-file commit $1) ]; then
echo "Invalid commit SHA."
echo "$(basename $0) -h for help"
exit 1
fi
}
# Parse command-line arguments.
#
# Note that we're following the command-line exit status convention of zero
# to mean "success."
verbose=1
all_codelabs=0
has_source=0
# Set default branch to this branch
export DEFAULT_BRANCH=$(git symbolic-ref --short HEAD)
while [ $# -gt 0 ]; do
case $1 in
-v|--verbose)
verbose=0
shift
;;
-rb|--remote-branch)
shift
check_arguments $1
check_reference $1
if [ $has_source -eq 1 ]; then
echo "Only one of --remote-branch/--commit/--tag allowed."
echo "$(basename $0) -h for help"
exit -1
fi
export TEST_BRANCH=$1
has_source=1
shift
;;
-c|--commit)
shift
check_arguments $1
check_commit $1
if [ $has_source -eq 1 ];then
echo "Only one of --remote-branch/--commit/--tag allowed."
echo "$(basename $0) -h for help"
exit -1
fi
export TEST_COMMIT=$1
has_source=1
shift
;;
-t|--tag)
shift
check_arguments $1
check_reference $1
if [ $has_source -eq 1 ];then
echo "Only one of --remote-branch/--commit/--tag allowed."
echo "$(basename $0) -h for help"
exit -1
fi
export TEST_TAG=$1
has_source=1
shift
;;
-a|--all)
all_codelabs=1
shift
;;
-h|--help)
echo
echo "Builds MDC iOS Codelabs."
echo
echo "By default the codelabs are tested against the remote branch tracking your current branch."
echo "Your current branch is: $DEFAULT_BRANCH"
echo
echo "By default the Swift and Objective-C Google I/O 2018 Codelab 104 and Codelab 111 targets will be built."
echo "You can build all codelabs by specifying --all."
echo
echo "Options:"
echo "-v, --verbose Verbose logging."
echo "-rb, --remote-branch Remote branch of MDC-iOS with which to test codelabs."
echo "-c, --commit Commit SHA of MDC-iOS with which to test codelabs."
echo "-t, --tag Tag of MDC-iOS with which to test codelabs."
echo "-a, --all Build all the codelabs."
echo "-h, --help Print this usage information."
echo
echo "Note: Only one of --remote-branch/--commit/--tag is allowed."
echo
exit 0
;;
*)
echo "Unknown option $1, aborting."
exit -1
;;
esac
done
# Ensure the current branch is a valid remote branch
if [ $has_source -eq 0 ]; then
check_reference $DEFAULT_BRANCH
fi
readonly SIGNING_OPTIONS="CODE_SIGNING_REQUIRED=NO CODE_SIGNING_ALLOWED=NO"
# Check for xcpretty once and cache the result.
is_xcpretty_available
readonly IS_XCPRETTY_AVAILABLE=$?
all_builds_ok=1
cd $SCRIPTS_DIR/external
# Remove the codelabs folder if it exists already so it can be re-cloned.
if [ -d "material-components-ios-codelabs" ]; then
rm -rf material-components-ios-codelabs
fi
git clone https://github.com/material-components/material-components-ios-codelabs.git
cd material-components-ios-codelabs
# Build only 104 and 111 complete by default. With -a flag, build all.
if [ "$all_codelabs" -eq 1 ]; then
readonly WORKSPACE_PATHS=$(find $SCRIPTS_DIR/external/material-components-ios-codelabs -name "*.xcworkspace" ! -name "project.xcworkspace")
else
readonly WORKSPACE_PATHS=$(find $SCRIPTS_DIR/external/material-components-ios-codelabs/MDC-104/*/Complete $SCRIPTS_DIR/external/material-components-ios-codelabs/MDC-111/*/Complete -name "*.xcworkspace" ! -name "project.xcworkspace")
fi
for workspace_path in $WORKSPACE_PATHS; do
# Give the output some meaning, since all the filenames (mostly) the same.
workspace=$(basename "$workspace_path")
scheme=$(echo "$workspace" | cut -d '.' -f1)
codelab=$(echo "$workspace_path" | cut -d/ -f10)
language=$(echo "$workspace_path" | cut -d/ -f11)
codelab_state=$(echo "$workspace_path" | cut -d/ -f12)
log_file=$(dirname "$workspace_path")/"build_log_for_scheme_${scheme}.txt"
options="-workspace $workspace_path -scheme $scheme $SIGNING_OPTIONS"
build_command="xcodebuild $options build"
pod_install="pod install --project-directory=$(dirname "$workspace_path")"
# pod install so the specified branch/commit/tag can be used
if [ -f $(dirname "$workspace_path")/Podfile ]; then
if [ "$verbose" -eq 0 ]; then
$pod_install
else
$pod_install --silent
fi
else
echo "No Podfile available."
exit 1
fi
echo "xcodebuild $scheme in $workspace for $codelab $codelab_state in $language."
if [ "$verbose" -eq 0 ]; then
if [ "$IS_XCPRETTY_AVAILABLE" -eq 0 ]; then
$build_command 2>&1 | tee "$log_file" | xcpretty
else
$build_command 2>&1 | tee "$log_file"
fi
else
$build_command >"$log_file" 2>&1
fi
# We need to have the output in a log file in all cases so we can check for
# expected failures.
if [ ${PIPESTATUS[0]} -eq 0 ] || is_expected_failure "$log_file"; then
rm "$log_file"
else
all_builds_ok=0
echo
echo "Failed to build $scheme in $workspace:"
echo "Log left in $log_file."
echo "Continuing with next build..."
echo
fi
done
cd $CURRENT_DIR
# If any build failed, exit with a failure exit status so continuous integration
# tools can react appropriately.
if [ "$all_builds_ok" -eq 1 ]; then
exit 0
else
exit 1
fi