1*cc02d7e2SAndroid Build Coastguard Worker#!/bin/bash 2*cc02d7e2SAndroid Build Coastguard Worker# Copyright 2016 gRPC authors. 3*cc02d7e2SAndroid Build Coastguard Worker# 4*cc02d7e2SAndroid Build Coastguard Worker# Licensed under the Apache License, Version 2.0 (the "License"); 5*cc02d7e2SAndroid Build Coastguard Worker# you may not use this file except in compliance with the License. 6*cc02d7e2SAndroid Build Coastguard Worker# You may obtain a copy of the License at 7*cc02d7e2SAndroid Build Coastguard Worker# 8*cc02d7e2SAndroid Build Coastguard Worker# http://www.apache.org/licenses/LICENSE-2.0 9*cc02d7e2SAndroid Build Coastguard Worker# 10*cc02d7e2SAndroid Build Coastguard Worker# Unless required by applicable law or agreed to in writing, software 11*cc02d7e2SAndroid Build Coastguard Worker# distributed under the License is distributed on an "AS IS" BASIS, 12*cc02d7e2SAndroid Build Coastguard Worker# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13*cc02d7e2SAndroid Build Coastguard Worker# See the License for the specific language governing permissions and 14*cc02d7e2SAndroid Build Coastguard Worker# limitations under the License. 15*cc02d7e2SAndroid Build Coastguard Worker# 16*cc02d7e2SAndroid Build Coastguard Worker# Builds selected testing docker images and pushes them to artifact registry. 17*cc02d7e2SAndroid Build Coastguard Worker# NOTE: These images are not intended to be used by gRPC end users, 18*cc02d7e2SAndroid Build Coastguard Worker# they simply provide an easily reproducible environment for running gRPC 19*cc02d7e2SAndroid Build Coastguard Worker# tests. 20*cc02d7e2SAndroid Build Coastguard Worker 21*cc02d7e2SAndroid Build Coastguard Workerset -ex 22*cc02d7e2SAndroid Build Coastguard Worker 23*cc02d7e2SAndroid Build Coastguard Workercd $(dirname $0)/../.. 24*cc02d7e2SAndroid Build Coastguard Workergit_root=$(pwd) 25*cc02d7e2SAndroid Build Coastguard Workercd - 26*cc02d7e2SAndroid Build Coastguard Worker 27*cc02d7e2SAndroid Build Coastguard Worker# Recognized env variables that can be used as params. 28*cc02d7e2SAndroid Build Coastguard Worker# LOCAL_ONLY_MODE: if set (e.g. LOCAL_ONLY_MODE=true), script will only operate locally and it won't query artifact registry and won't upload to it. 29*cc02d7e2SAndroid Build Coastguard Worker# CHECK_MODE: if set, the script will check that all the .current_version files are up-to-date (used by sanity tests). 30*cc02d7e2SAndroid Build Coastguard Worker# SKIP_UPLOAD: if set, script won't push docker images it built to artifact registry. 31*cc02d7e2SAndroid Build Coastguard Worker# TRANSFER_FROM_DOCKERHUB: if set, will attempt to grab docker images missing in artifact registry from dockerhub instead of building them from scratch locally. 32*cc02d7e2SAndroid Build Coastguard Worker 33*cc02d7e2SAndroid Build Coastguard Worker# How to configure docker before running this script for the first time: 34*cc02d7e2SAndroid Build Coastguard Worker# Configure docker: 35*cc02d7e2SAndroid Build Coastguard Worker# $ gcloud auth configure-docker us-docker.pkg.dev 36*cc02d7e2SAndroid Build Coastguard Worker# Login with gcloud: 37*cc02d7e2SAndroid Build Coastguard Worker# $ gcloud auth login 38*cc02d7e2SAndroid Build Coastguard Worker 39*cc02d7e2SAndroid Build Coastguard Worker# Various check that the environment is setup correctly. 40*cc02d7e2SAndroid Build Coastguard Worker# The enviroment checks are skipped when running as a sanity check on CI. 41*cc02d7e2SAndroid Build Coastguard Workerif [ "${CHECK_MODE}" == "" ] 42*cc02d7e2SAndroid Build Coastguard Workerthen 43*cc02d7e2SAndroid Build Coastguard Worker # Check that docker is installed and sudoless docker works. 44*cc02d7e2SAndroid Build Coastguard Worker docker run --rm -it debian:11 bash -c 'echo "sudoless docker run works!"' || \ 45*cc02d7e2SAndroid Build Coastguard Worker (echo "Error: docker not installed or sudoless docker doesn't work?" && exit 1) 46*cc02d7e2SAndroid Build Coastguard Worker 47*cc02d7e2SAndroid Build Coastguard Worker # Some of the images we build are for arm64 architecture and the easiest 48*cc02d7e2SAndroid Build Coastguard Worker # way of allowing them to build locally on x64 machine is to use 49*cc02d7e2SAndroid Build Coastguard Worker # qemu binfmt-misc hook that automatically runs arm64 binaries under 50*cc02d7e2SAndroid Build Coastguard Worker # an emulator. 51*cc02d7e2SAndroid Build Coastguard Worker # Perform a check that "qemu-user-static" with binfmt-misc hook 52*cc02d7e2SAndroid Build Coastguard Worker # is installed, to give an early warning (otherwise building arm64 images won't work) 53*cc02d7e2SAndroid Build Coastguard Worker docker run --rm -it arm64v8/debian:11 bash -c 'echo "able to run arm64 docker images with an emulator!"' || \ 54*cc02d7e2SAndroid Build Coastguard Worker (echo "Error: can't run arm64 images under an emulator. Have you run 'sudo apt-get install qemu-user-static'?" && exit 1) 55*cc02d7e2SAndroid Build Coastguard Workerfi 56*cc02d7e2SAndroid Build Coastguard Worker 57*cc02d7e2SAndroid Build Coastguard WorkerARTIFACT_REGISTRY_PREFIX=us-docker.pkg.dev/grpc-testing/testing-images-public 58*cc02d7e2SAndroid Build Coastguard Worker 59*cc02d7e2SAndroid Build Coastguard Worker# all dockerfile definitions we use for testing and for which we push an image to the registry 60*cc02d7e2SAndroid Build Coastguard WorkerALL_DOCKERFILE_DIRS=( 61*cc02d7e2SAndroid Build Coastguard Worker tools/dockerfile/test/* 62*cc02d7e2SAndroid Build Coastguard Worker tools/dockerfile/grpc_artifact_* 63*cc02d7e2SAndroid Build Coastguard Worker tools/dockerfile/interoptest/* 64*cc02d7e2SAndroid Build Coastguard Worker tools/dockerfile/distribtest/* 65*cc02d7e2SAndroid Build Coastguard Worker third_party/rake-compiler-dock/* 66*cc02d7e2SAndroid Build Coastguard Worker) 67*cc02d7e2SAndroid Build Coastguard Worker 68*cc02d7e2SAndroid Build Coastguard WorkerCHECK_FAILED="" 69*cc02d7e2SAndroid Build Coastguard Worker 70*cc02d7e2SAndroid Build Coastguard Workerif [ "${CHECK_MODE}" != "" ] 71*cc02d7e2SAndroid Build Coastguard Workerthen 72*cc02d7e2SAndroid Build Coastguard Worker # Check that there are no stale .current_version files (for which the corresponding 73*cc02d7e2SAndroid Build Coastguard Worker # dockerfile_dir doesn't exist anymore). 74*cc02d7e2SAndroid Build Coastguard Worker for CURRENTVERSION_FILE in $(find tools/ third_party/rake-compiler-dock -name '*.current_version') 75*cc02d7e2SAndroid Build Coastguard Worker do 76*cc02d7e2SAndroid Build Coastguard Worker DOCKERFILE_DIR="$(echo ${CURRENTVERSION_FILE} | sed 's/.current_version$//')" 77*cc02d7e2SAndroid Build Coastguard Worker if [ ! -e "${DOCKERFILE_DIR}/Dockerfile" ] 78*cc02d7e2SAndroid Build Coastguard Worker then 79*cc02d7e2SAndroid Build Coastguard Worker echo "Found that ${DOCKERFILE_DIR} has '.current_version' file but there is no corresponding Dockerfile." 80*cc02d7e2SAndroid Build Coastguard Worker echo "Should the ${CURRENTVERSION_FILE} file be deleted?" 81*cc02d7e2SAndroid Build Coastguard Worker CHECK_FAILED=true 82*cc02d7e2SAndroid Build Coastguard Worker fi 83*cc02d7e2SAndroid Build Coastguard Worker done 84*cc02d7e2SAndroid Build Coastguard Workerfi 85*cc02d7e2SAndroid Build Coastguard Worker 86*cc02d7e2SAndroid Build Coastguard Workerfor DOCKERFILE_DIR in "${ALL_DOCKERFILE_DIRS[@]}" 87*cc02d7e2SAndroid Build Coastguard Workerdo 88*cc02d7e2SAndroid Build Coastguard Worker # Generate image name based on Dockerfile checksum. That works well as long 89*cc02d7e2SAndroid Build Coastguard Worker # as can count on dockerfiles being written in a way that changing the logical 90*cc02d7e2SAndroid Build Coastguard Worker # contents of the docker image always changes the SHA (e.g. using "ADD file" 91*cc02d7e2SAndroid Build Coastguard Worker # cmd in the dockerfile in not ok as contents of the added file will not be 92*cc02d7e2SAndroid Build Coastguard Worker # reflected in the SHA). 93*cc02d7e2SAndroid Build Coastguard Worker DOCKER_IMAGE_NAME=$(basename $DOCKERFILE_DIR) 94*cc02d7e2SAndroid Build Coastguard Worker 95*cc02d7e2SAndroid Build Coastguard Worker if [ ! -e "$DOCKERFILE_DIR/Dockerfile" ]; then 96*cc02d7e2SAndroid Build Coastguard Worker continue 97*cc02d7e2SAndroid Build Coastguard Worker else 98*cc02d7e2SAndroid Build Coastguard Worker DOCKER_IMAGE_TAG=$(sha1sum $DOCKERFILE_DIR/Dockerfile | cut -f1 -d\ ) 99*cc02d7e2SAndroid Build Coastguard Worker fi 100*cc02d7e2SAndroid Build Coastguard Worker 101*cc02d7e2SAndroid Build Coastguard Worker echo "Visiting ${DOCKERFILE_DIR}" 102*cc02d7e2SAndroid Build Coastguard Worker 103*cc02d7e2SAndroid Build Coastguard Worker if [ "${LOCAL_ONLY_MODE}" == "" ] 104*cc02d7e2SAndroid Build Coastguard Worker then 105*cc02d7e2SAndroid Build Coastguard Worker # value obtained here corresponds to the "RepoDigests" from "docker image inspect", but without the need to actually pull the image 106*cc02d7e2SAndroid Build Coastguard Worker DOCKER_IMAGE_DIGEST_REMOTE=$(gcloud artifacts docker images describe "${ARTIFACT_REGISTRY_PREFIX}/${DOCKER_IMAGE_NAME}:${DOCKER_IMAGE_TAG}" --format=json | jq -r '.image_summary.digest') 107*cc02d7e2SAndroid Build Coastguard Worker 108*cc02d7e2SAndroid Build Coastguard Worker if [ "${DOCKER_IMAGE_DIGEST_REMOTE}" != "" ] 109*cc02d7e2SAndroid Build Coastguard Worker then 110*cc02d7e2SAndroid Build Coastguard Worker # skip building the image if it already exists in the destination registry 111*cc02d7e2SAndroid Build Coastguard Worker echo "Docker image ${DOCKER_IMAGE_NAME} already exists in artifact registry at the right version (tag ${DOCKER_IMAGE_TAG})." 112*cc02d7e2SAndroid Build Coastguard Worker 113*cc02d7e2SAndroid Build Coastguard Worker VERSION_FILE_OUT_OF_DATE="" 114*cc02d7e2SAndroid Build Coastguard Worker grep "^${ARTIFACT_REGISTRY_PREFIX}/${DOCKER_IMAGE_NAME}:${DOCKER_IMAGE_TAG}@${DOCKER_IMAGE_DIGEST_REMOTE}$" ${DOCKERFILE_DIR}.current_version >/dev/null || VERSION_FILE_OUT_OF_DATE="true" 115*cc02d7e2SAndroid Build Coastguard Worker 116*cc02d7e2SAndroid Build Coastguard Worker if [ "${VERSION_FILE_OUT_OF_DATE}" == "" ] 117*cc02d7e2SAndroid Build Coastguard Worker then 118*cc02d7e2SAndroid Build Coastguard Worker echo "Version file for ${DOCKER_IMAGE_NAME} is in sync with info from artifact registry." 119*cc02d7e2SAndroid Build Coastguard Worker continue 120*cc02d7e2SAndroid Build Coastguard Worker fi 121*cc02d7e2SAndroid Build Coastguard Worker 122*cc02d7e2SAndroid Build Coastguard Worker if [ "${CHECK_MODE}" != "" ] 123*cc02d7e2SAndroid Build Coastguard Worker then 124*cc02d7e2SAndroid Build Coastguard Worker echo "CHECK FAILED: Version file ${DOCKERFILE_DIR}.current_version is not in sync with info from artifact registry." 125*cc02d7e2SAndroid Build Coastguard Worker CHECK_FAILED=true 126*cc02d7e2SAndroid Build Coastguard Worker continue 127*cc02d7e2SAndroid Build Coastguard Worker fi 128*cc02d7e2SAndroid Build Coastguard Worker 129*cc02d7e2SAndroid Build Coastguard Worker # update info on what we consider to be the current version of the docker image (which will be used to run tests) 130*cc02d7e2SAndroid Build Coastguard Worker # we consider the sha256 image digest info from the artifact registry to be the canonical one 131*cc02d7e2SAndroid Build Coastguard Worker echo -n "${ARTIFACT_REGISTRY_PREFIX}/${DOCKER_IMAGE_NAME}:${DOCKER_IMAGE_TAG}@${DOCKER_IMAGE_DIGEST_REMOTE}" >${DOCKERFILE_DIR}.current_version 132*cc02d7e2SAndroid Build Coastguard Worker 133*cc02d7e2SAndroid Build Coastguard Worker continue 134*cc02d7e2SAndroid Build Coastguard Worker fi 135*cc02d7e2SAndroid Build Coastguard Worker 136*cc02d7e2SAndroid Build Coastguard Worker if [ "${CHECK_MODE}" != "" ] 137*cc02d7e2SAndroid Build Coastguard Worker then 138*cc02d7e2SAndroid Build Coastguard Worker echo "CHECK FAILED: Docker image ${DOCKER_IMAGE_NAME} not found in artifact registry." 139*cc02d7e2SAndroid Build Coastguard Worker CHECK_FAILED=true 140*cc02d7e2SAndroid Build Coastguard Worker continue 141*cc02d7e2SAndroid Build Coastguard Worker fi 142*cc02d7e2SAndroid Build Coastguard Worker 143*cc02d7e2SAndroid Build Coastguard Worker else 144*cc02d7e2SAndroid Build Coastguard Worker echo "Skipped querying artifact registry (running in local-only mode)." 145*cc02d7e2SAndroid Build Coastguard Worker fi 146*cc02d7e2SAndroid Build Coastguard Worker 147*cc02d7e2SAndroid Build Coastguard Worker # if the .current_version file doesn't exist or it doesn't contain the right SHA checksum, 148*cc02d7e2SAndroid Build Coastguard Worker # it is out of date and we will need to rebuild the docker image locally. 149*cc02d7e2SAndroid Build Coastguard Worker LOCAL_BUILD_REQUIRED="" 150*cc02d7e2SAndroid Build Coastguard Worker grep "^${ARTIFACT_REGISTRY_PREFIX}/${DOCKER_IMAGE_NAME}:${DOCKER_IMAGE_TAG}@sha256:.*$" ${DOCKERFILE_DIR}.current_version >/dev/null || LOCAL_BUILD_REQUIRED=true 151*cc02d7e2SAndroid Build Coastguard Worker 152*cc02d7e2SAndroid Build Coastguard Worker # If the current version file has contains SHA checksum, but not the remote image digest, 153*cc02d7e2SAndroid Build Coastguard Worker # it means the locally-built image hasn't been pushed to artifact registry yet. 154*cc02d7e2SAndroid Build Coastguard Worker DIGEST_MISSING_IN_CURRENT_VERSION_FILE="" 155*cc02d7e2SAndroid Build Coastguard Worker grep "^${ARTIFACT_REGISTRY_PREFIX}/${DOCKER_IMAGE_NAME}:${DOCKER_IMAGE_TAG}$" ${DOCKERFILE_DIR}.current_version >/dev/null && DIGEST_MISSING_IN_CURRENT_VERSION_FILE=true 156*cc02d7e2SAndroid Build Coastguard Worker 157*cc02d7e2SAndroid Build Coastguard Worker if [ "${LOCAL_BUILD_REQUIRED}" == "" ] 158*cc02d7e2SAndroid Build Coastguard Worker then 159*cc02d7e2SAndroid Build Coastguard Worker echo "Dockerfile for ${DOCKER_IMAGE_NAME} hasn't changed. Will skip 'docker build'." 160*cc02d7e2SAndroid Build Coastguard Worker continue 161*cc02d7e2SAndroid Build Coastguard Worker fi 162*cc02d7e2SAndroid Build Coastguard Worker 163*cc02d7e2SAndroid Build Coastguard Worker if [ "${CHECK_MODE}" != "" ] && [ "${DIGEST_MISSING_IN_CURRENT_VERSION_FILE}" != "" ] 164*cc02d7e2SAndroid Build Coastguard Worker then 165*cc02d7e2SAndroid Build Coastguard Worker echo "CHECK FAILED: Dockerfile for ${DOCKER_IMAGE_NAME} has changed and was built locally, but looks like it hasn't been pushed." 166*cc02d7e2SAndroid Build Coastguard Worker CHECK_FAILED=true 167*cc02d7e2SAndroid Build Coastguard Worker continue 168*cc02d7e2SAndroid Build Coastguard Worker fi 169*cc02d7e2SAndroid Build Coastguard Worker 170*cc02d7e2SAndroid Build Coastguard Worker if [ "${CHECK_MODE}" != "" ] 171*cc02d7e2SAndroid Build Coastguard Worker then 172*cc02d7e2SAndroid Build Coastguard Worker echo "CHECK FAILED: Dockerfile for ${DOCKER_IMAGE_NAME} has changed, but the ${DOCKERFILE_DIR}.current_version is not up to date." 173*cc02d7e2SAndroid Build Coastguard Worker CHECK_FAILED=true 174*cc02d7e2SAndroid Build Coastguard Worker continue 175*cc02d7e2SAndroid Build Coastguard Worker fi 176*cc02d7e2SAndroid Build Coastguard Worker 177*cc02d7e2SAndroid Build Coastguard Worker if [ "${TRANSFER_FROM_DOCKERHUB}" == "" ] 178*cc02d7e2SAndroid Build Coastguard Worker then 179*cc02d7e2SAndroid Build Coastguard Worker echo "Running 'docker build' for ${DOCKER_IMAGE_NAME}" 180*cc02d7e2SAndroid Build Coastguard Worker echo "==========" 181*cc02d7e2SAndroid Build Coastguard Worker docker build -t ${ARTIFACT_REGISTRY_PREFIX}/${DOCKER_IMAGE_NAME}:${DOCKER_IMAGE_TAG} ${DOCKERFILE_DIR} 182*cc02d7e2SAndroid Build Coastguard Worker echo "==========" 183*cc02d7e2SAndroid Build Coastguard Worker else 184*cc02d7e2SAndroid Build Coastguard Worker # TRANSFER_FROM_DOCKERHUB is a temporary feature that pulls the corresponding image from dockerhub instead 185*cc02d7e2SAndroid Build Coastguard Worker # of building it from scratch locally. This should simplify the dockerhub -> artifact registry migration. 186*cc02d7e2SAndroid Build Coastguard Worker # TODO(jtattermusch): remove this feature in Q1 2023. 187*cc02d7e2SAndroid Build Coastguard Worker DOCKERHUB_ORGANIZATION=grpctesting 188*cc02d7e2SAndroid Build Coastguard Worker # pull image from dockerhub 189*cc02d7e2SAndroid Build Coastguard Worker docker pull ${DOCKERHUB_ORGANIZATION}/${DOCKER_IMAGE_NAME}:${DOCKER_IMAGE_TAG} 190*cc02d7e2SAndroid Build Coastguard Worker # add the artifact registry tag 191*cc02d7e2SAndroid Build Coastguard Worker docker tag ${DOCKERHUB_ORGANIZATION}/${DOCKER_IMAGE_NAME}:${DOCKER_IMAGE_TAG} ${ARTIFACT_REGISTRY_PREFIX}/${DOCKER_IMAGE_NAME}:${DOCKER_IMAGE_TAG} 192*cc02d7e2SAndroid Build Coastguard Worker fi 193*cc02d7e2SAndroid Build Coastguard Worker 194*cc02d7e2SAndroid Build Coastguard Worker # After building the docker image locally, we don't know the image's RepoDigest (which is distinct from image's "Id" digest) yet 195*cc02d7e2SAndroid Build Coastguard Worker # so we can only update the .current_version file with the image tag (which will be enough for running tests under docker locally). 196*cc02d7e2SAndroid Build Coastguard Worker # The .current_version file will be updated with both tag and SHA256 repo digest later, once we actually push it. 197*cc02d7e2SAndroid Build Coastguard Worker # See b/278226801 for context. 198*cc02d7e2SAndroid Build Coastguard Worker echo -n "${ARTIFACT_REGISTRY_PREFIX}/${DOCKER_IMAGE_NAME}:${DOCKER_IMAGE_TAG}" >${DOCKERFILE_DIR}.current_version 199*cc02d7e2SAndroid Build Coastguard Worker 200*cc02d7e2SAndroid Build Coastguard Worker if [ "${SKIP_UPLOAD}" == "" ] && [ "${LOCAL_ONLY_MODE}" == "" ] 201*cc02d7e2SAndroid Build Coastguard Worker then 202*cc02d7e2SAndroid Build Coastguard Worker docker push ${ARTIFACT_REGISTRY_PREFIX}/${DOCKER_IMAGE_NAME}:${DOCKER_IMAGE_TAG} 203*cc02d7e2SAndroid Build Coastguard Worker 204*cc02d7e2SAndroid Build Coastguard Worker # After successful push, the image's RepoDigest info will become available in "docker image inspect", 205*cc02d7e2SAndroid Build Coastguard Worker # so we update the .current_version file with the repo digest. 206*cc02d7e2SAndroid Build Coastguard Worker DOCKER_IMAGE_DIGEST_REMOTE=$(docker image inspect "${ARTIFACT_REGISTRY_PREFIX}/${DOCKER_IMAGE_NAME}:${DOCKER_IMAGE_TAG}" | jq -e -r ".[0].RepoDigests[] | select(contains(\"${ARTIFACT_REGISTRY_PREFIX}/${DOCKER_IMAGE_NAME}@\"))" | sed 's/^.*@sha256:/sha256:/') 207*cc02d7e2SAndroid Build Coastguard Worker echo -n "${ARTIFACT_REGISTRY_PREFIX}/${DOCKER_IMAGE_NAME}:${DOCKER_IMAGE_TAG}@${DOCKER_IMAGE_DIGEST_REMOTE}" >${DOCKERFILE_DIR}.current_version 208*cc02d7e2SAndroid Build Coastguard Worker fi 209*cc02d7e2SAndroid Build Coastguard Workerdone 210*cc02d7e2SAndroid Build Coastguard Worker 211*cc02d7e2SAndroid Build Coastguard Workerif [ "${CHECK_MODE}" != "" ] 212*cc02d7e2SAndroid Build Coastguard Workerthen 213*cc02d7e2SAndroid Build Coastguard Worker # Check that dockerimage_current_versions.bzl is up to date. 214*cc02d7e2SAndroid Build Coastguard Worker CHECK_MODE="${CHECK_MODE}" tools/bazelify_tests/generate_dockerimage_current_versions_bzl.sh || CHECK_FAILED=true 215*cc02d7e2SAndroid Build Coastguard Workerelse 216*cc02d7e2SAndroid Build Coastguard Worker # Regenerate dockerimage_current_versions.bzl 217*cc02d7e2SAndroid Build Coastguard Worker tools/bazelify_tests/generate_dockerimage_current_versions_bzl.sh 218*cc02d7e2SAndroid Build Coastguard Workerfi 219*cc02d7e2SAndroid Build Coastguard Worker 220*cc02d7e2SAndroid Build Coastguard Workerif [ "${CHECK_FAILED}" != "" ] 221*cc02d7e2SAndroid Build Coastguard Workerthen 222*cc02d7e2SAndroid Build Coastguard Worker echo "ERROR: Some checks have failed." 223*cc02d7e2SAndroid Build Coastguard Worker exit 1 224*cc02d7e2SAndroid Build Coastguard Workerfi 225*cc02d7e2SAndroid Build Coastguard Worker 226*cc02d7e2SAndroid Build Coastguard Workerecho "All done." 227