17755: Merge branch 'main' into 17755-add-singularity-to-compute-image
[arvados.git] / build / run-build-packages-one-target.sh
1 #!/bin/bash
2 # Copyright (C) The Arvados Authors. All rights reserved.
3 #
4 # SPDX-License-Identifier: AGPL-3.0
5
6 read -rd "\000" helpmessage <<EOF
7 $(basename $0): Orchestrate run-build-packages.sh for one target
8
9 Syntax:
10         WORKSPACE=/path/to/arvados $(basename $0) [options]
11
12 --target <target>
13     Distribution to build packages for (default: debian10)
14 --command
15     Build command to execute (default: use built-in Docker image command)
16 --test-packages
17     Run package install test script "test-packages-[target].sh"
18 --debug
19     Output debug information (default: false)
20 --only-build <package>
21     Build only a specific package
22 --only-test <package>
23     Test only a specific package
24 --force-build
25     Build even if the package exists upstream or if it has already been
26     built locally
27 --force-test
28     Test even if there is no new untested package
29 --build-version <string>
30     Version to build (default:
31     \$ARVADOS_BUILDING_VERSION-\$ARVADOS_BUILDING_ITERATION or
32     0.1.timestamp.commithash)
33
34 WORKSPACE=path         Path to the Arvados source tree to build packages from
35
36 EOF
37
38 set -e
39
40 if ! [[ -n "$WORKSPACE" ]]; then
41   echo >&2 "$helpmessage"
42   echo >&2
43   echo >&2 "Error: WORKSPACE environment variable not set"
44   echo >&2
45   exit 1
46 fi
47
48 if ! [[ -d "$WORKSPACE" ]]; then
49   echo >&2 "$helpmessage"
50   echo >&2
51   echo >&2 "Error: $WORKSPACE is not a directory"
52   echo >&2
53   exit 1
54 fi
55
56 PARSEDOPTS=$(getopt --name "$0" --longoptions \
57     help,debug,test-packages,target:,command:,only-test:,force-test,only-build:,force-build,build-version: \
58     -- "" "$@")
59 if [ $? -ne 0 ]; then
60     exit 1
61 fi
62
63 TARGET=debian10
64 FORCE_BUILD=0
65 COMMAND=
66 DEBUG=
67
68 eval set -- "$PARSEDOPTS"
69 while [ $# -gt 0 ]; do
70     case "$1" in
71         --help)
72             echo >&2 "$helpmessage"
73             echo >&2
74             exit 1
75             ;;
76         --target)
77             TARGET="$2"; shift
78             ;;
79         --only-test)
80             test_packages=1
81             testing_one_package=1
82             packages="$2"; shift
83             ;;
84         --force-test)
85             FORCE_TEST=true
86             ;;
87         --force-build)
88             FORCE_BUILD=1
89             ;;
90         --only-build)
91             ONLY_BUILD="$2"; shift
92             ;;
93         --debug)
94             DEBUG=" --debug"
95             ARVADOS_DEBUG="1"
96             ;;
97         --command)
98             COMMAND="$2"; shift
99             ;;
100         --test-packages)
101             test_packages=1
102             ;;
103         --build-version)
104             if [[ -z "$2" ]]; then
105                 :
106             elif ! [[ "$2" =~ (.*)-(.*) ]]; then
107                 echo >&2 "FATAL: --build-version '$2' does not include an iteration. Try '${2}-1'?"
108                 exit 1
109             elif ! [[ "$2" =~ ^[0-9]+\.[0-9]+\.[0-9]+(\.[0-9]+|)(~rc[0-9]+|~dev[0-9]+|)-[0-9]+$ ]]; then
110                 echo >&2 "FATAL: --build-version '$2' is invalid, must match pattern ^[0-9]+\.[0-9]+\.[0-9]+(\.[0-9]+|)(~rc[0-9]+|~dev[0-9]+|)-[0-9]+$"
111                 exit 1
112             else
113                 [[ "$2" =~ (.*)-(.*) ]]
114                 ARVADOS_BUILDING_VERSION="${BASH_REMATCH[1]}"
115                 ARVADOS_BUILDING_ITERATION="${BASH_REMATCH[2]}"
116             fi
117             shift
118             ;;
119         --)
120             if [ $# -gt 1 ]; then
121                 echo >&2 "$0: unrecognized argument '$2'. Try: $0 --help"
122                 exit 1
123             fi
124             ;;
125     esac
126     shift
127 done
128
129 set -e
130
131 if [[ -n "$ARVADOS_BUILDING_VERSION" ]]; then
132     echo "build version='$ARVADOS_BUILDING_VERSION', package iteration='$ARVADOS_BUILDING_ITERATION'"
133 fi
134
135 if [[ -n "$test_packages" ]]; then
136   if [[ -n "$(find $WORKSPACE/packages/$TARGET -name '*.rpm')" ]] ; then
137     set +e
138     /usr/bin/which createrepo >/dev/null
139     if [[ "$?" != "0" ]]; then
140       echo >&2
141       echo >&2 "Error: please install createrepo. E.g. sudo apt-get install createrepo"
142       echo >&2
143       exit 1
144     fi
145     set -e
146     createrepo $WORKSPACE/packages/$TARGET
147   fi
148
149   if [[ -n "$(find $WORKSPACE/packages/$TARGET -name '*.deb')" ]] ; then
150     set +e
151     /usr/bin/which dpkg-scanpackages >/dev/null
152     if [[ "$?" != "0" ]]; then
153       echo >&2
154       echo >&2 "Error: please install dpkg-dev. E.g. sudo apt-get install dpkg-dev"
155       echo >&2
156       exit 1
157     fi
158     /usr/bin/which apt-ftparchive >/dev/null
159     if [[ "$?" != "0" ]]; then
160       echo >&2
161       echo >&2 "Error: please install apt-utils. E.g. sudo apt-get install apt-utils"
162       echo >&2
163       exit 1
164     fi
165     set -e
166     (cd $WORKSPACE/packages/$TARGET
167       dpkg-scanpackages .  2> >(grep -v 'warning' 1>&2) | tee Packages | gzip -c > Packages.gz
168       apt-ftparchive -o APT::FTPArchive::Release::Origin=Arvados release . > Release
169     )
170   fi
171
172   COMMAND="/jenkins/package-testing/test-packages-$TARGET.sh"
173   IMAGE="arvados/package-test:$TARGET"
174 else
175   IMAGE="arvados/build:$TARGET"
176   if [[ "$COMMAND" != "" ]]; then
177     COMMAND="/usr/local/rvm/bin/rvm-exec default bash /jenkins/$COMMAND --target $TARGET$DEBUG"
178   fi
179 fi
180
181 JENKINS_DIR=$(dirname "$(readlink -e "$0")")
182
183 if [[ -n "$test_packages" ]]; then
184     pushd "$JENKINS_DIR/package-test-dockerfiles"
185     make "$TARGET/generated"
186 else
187     pushd "$JENKINS_DIR/package-build-dockerfiles"
188     make "$TARGET/generated"
189 fi
190
191 echo $TARGET
192 cd $TARGET
193 time docker build --tag=$IMAGE .
194 popd
195
196 if test -z "$packages" ; then
197     packages="arvados-api-server
198         arvados-client
199         arvados-controller
200         arvados-dispatch-cloud
201         arvados-dispatch-lsf
202         arvados-docker-cleaner
203         arvados-git-httpd
204         arvados-health
205         arvados-server
206         arvados-src
207         arvados-sync-groups
208         arvados-workbench
209         arvados-workbench2
210         arvados-ws
211         crunch-dispatch-local
212         crunch-dispatch-slurm
213         crunch-run
214         crunchstat
215         keepproxy
216         keepstore
217         keep-balance
218         keep-block-check
219         keep-rsync
220         keep-exercise
221         keep-rsync
222         keep-block-check
223         keep-web
224         libarvados-perl
225         libpam-arvados-go
226         python3-cwltest
227         python3-arvados-fuse
228         python3-arvados-python-client
229         python3-arvados-cwl-runner
230         python3-crunchstat-summary
231         python3-arvados-user-activity"
232 fi
233
234 FINAL_EXITCODE=0
235
236 package_fails=""
237
238 mkdir -p "$WORKSPACE/apps/workbench/vendor/cache-$TARGET"
239 mkdir -p "$WORKSPACE/services/api/vendor/cache-$TARGET"
240
241 docker_volume_args=(
242     -v "$JENKINS_DIR:/jenkins"
243     -v "$WORKSPACE:/arvados"
244     -v /arvados/services/api/vendor/bundle
245     -v /arvados/apps/workbench/vendor/bundle
246     -v "$WORKSPACE/services/api/vendor/cache-$TARGET:/arvados/services/api/vendor/cache"
247     -v "$WORKSPACE/apps/workbench/vendor/cache-$TARGET:/arvados/apps/workbench/vendor/cache"
248 )
249
250 if [[ -n "$test_packages" ]]; then
251     for p in $packages ; do
252         if [[ -n "$ONLY_BUILD" ]] && [[ "$p" != "$ONLY_BUILD" ]]; then
253             continue
254         fi
255         if [[ -e "${WORKSPACE}/packages/.last_test_${TARGET}" ]] && [[ -z "$FORCE_TEST" ]]; then
256           MATCH=`find ${WORKSPACE}/packages/ -newer ${WORKSPACE}/packages/.last_test_${TARGET} -regex .*${TARGET}/$p.*`
257           if [[ "$MATCH" == "" ]]; then
258             # No new package has been built that needs testing
259             echo "Skipping $p test because no new package was built since the last test."
260             continue
261           fi
262         fi
263         # If we're testing all packages, we should not error out on packages that don't exist.
264         # If we are testing one specific package only (i.e. --only-test was given), we should
265         # error out if that package does not exist.
266         if [[ -z "$testing_one_package" ]]; then
267           MATCH=`find ${WORKSPACE}/packages/ -regextype posix-extended -regex .*${TARGET}/$p.*\\(deb\\|rpm\\)`
268           if [[ "$MATCH" == "" ]]; then
269             # No new package has been built that needs testing
270             echo "Skipping $p test because no package file is available to test."
271             continue
272           fi
273         fi
274         echo
275         echo "START: $p test on $IMAGE" >&2
276         if docker run \
277             --rm \
278             "${docker_volume_args[@]}" \
279             --env ARVADOS_DEBUG=$ARVADOS_DEBUG \
280             --env "TARGET=$TARGET" \
281             --env "WORKSPACE=/arvados" \
282             "$IMAGE" $COMMAND $p
283         then
284             echo "OK: $p test on $IMAGE succeeded" >&2
285         else
286             FINAL_EXITCODE=$?
287             package_fails="$package_fails $p"
288             echo "ERROR: $p test on $IMAGE failed with exit status $FINAL_EXITCODE" >&2
289         fi
290     done
291
292     if [[ "$FINAL_EXITCODE" == "0" ]]; then
293       touch ${WORKSPACE}/packages/.last_test_${TARGET}
294     fi
295 else
296     echo
297     echo "START: build packages on $IMAGE" >&2
298     # Move existing packages and other files into the processed/ subdirectory
299     if [[ ! -e "${WORKSPACE}/packages/${TARGET}/processed" ]]; then
300       mkdir -p "${WORKSPACE}/packages/${TARGET}/processed"
301     fi
302     set +e
303     mv -f ${WORKSPACE}/packages/${TARGET}/* ${WORKSPACE}/packages/${TARGET}/processed/ 2>/dev/null
304     set -e
305     # Build packages.
306     if docker run \
307         --rm \
308         "${docker_volume_args[@]}" \
309         --env ARVADOS_BUILDING_VERSION="$ARVADOS_BUILDING_VERSION" \
310         --env ARVADOS_BUILDING_ITERATION="$ARVADOS_BUILDING_ITERATION" \
311         --env ARVADOS_DEBUG=$ARVADOS_DEBUG \
312         --env "ONLY_BUILD=$ONLY_BUILD" \
313         --env "FORCE_BUILD=$FORCE_BUILD" \
314         "$IMAGE" $COMMAND
315     then
316         echo
317         echo "OK: build packages on $IMAGE succeeded" >&2
318     else
319         FINAL_EXITCODE=$?
320         echo "ERROR: build packages on $IMAGE failed with exit status $FINAL_EXITCODE" >&2
321     fi
322 fi
323
324 if test -n "$package_fails" ; then
325     echo "Failed package tests:$package_fails" >&2
326 fi
327
328 exit $FINAL_EXITCODE