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