1 | #!/usr/bin/env bash
|
2 | #
|
3 | # Script for contributors to build dev dependencies -- packaged as cross-distro
|
4 | # "wedges". Tested in the Soil CI.
|
5 | #
|
6 | # Usage:
|
7 | # build/deps.sh <function name>
|
8 | #
|
9 | # Examples:
|
10 | # build/deps.sh fetch
|
11 | # build/deps.sh install-wedges # for both Python and C++
|
12 | # build/deps.sh rm-oils-crap # rm -r -f /wedge ~/wedge to start over
|
13 | #
|
14 | # TODO: Do we need something faster, just python2, re2c, and cmark?
|
15 | #
|
16 | # - build/deps.sh fetch-py
|
17 | # - build/deps.sh install-wedges-py
|
18 | #
|
19 | # TODO: Can we make most of them non-root deps? This requires rebuilding
|
20 | # containers, which requires podman.
|
21 | #
|
22 | # rm -r -f ~/wedge # would be better
|
23 |
|
24 | set -o nounset
|
25 | set -o pipefail
|
26 | set -o errexit
|
27 |
|
28 | REPO_ROOT=$(cd "$(dirname $0)/.."; pwd)
|
29 |
|
30 | source build/dev-shell.sh # python3 in PATH, PY3_LIBS_VERSION
|
31 | source deps/from-apt.sh # PY3_BUILD_DEPS
|
32 | #source deps/podman.sh
|
33 | source devtools/run-task.sh # run-task
|
34 | source test/tsv-lib.sh # tsv-concat
|
35 | source web/table/html.sh # table-sort-{begin,end}
|
36 |
|
37 | # Also in build/dev-shell.sh
|
38 | USER_WEDGE_DIR=~/wedge/oils-for-unix.org
|
39 | ROOT_WEDGE_DIR=/wedge/oils-for-unix.org
|
40 |
|
41 | readonly DEPS_SOURCE_DIR=_build/deps-source
|
42 |
|
43 | readonly RE2C_VERSION=3.0
|
44 | readonly RE2C_URL="https://github.com/skvadrik/re2c/releases/download/$RE2C_VERSION/re2c-$RE2C_VERSION.tar.xz"
|
45 |
|
46 | readonly CMARK_VERSION=0.29.0
|
47 | readonly CMARK_URL="https://github.com/commonmark/cmark/archive/$CMARK_VERSION.tar.gz"
|
48 |
|
49 | readonly PY_FTP_MIRROR="${PY_FTP_MIRROR:-https://www.python.org/ftp}"
|
50 |
|
51 | readonly PY2_VERSION=2.7.18
|
52 | readonly PY2_URL="$PY_FTP_MIRROR/python/$PY2_VERSION/Python-$PY2_VERSION.tar.xz"
|
53 |
|
54 | readonly PY3_VERSION=3.10.4
|
55 | readonly PY3_URL="$PY_FTP_MIRROR/python/$PY3_VERSION/Python-$PY3_VERSION.tar.xz"
|
56 |
|
57 | readonly BASH_VER=4.4 # don't clobber BASH_VERSION
|
58 | readonly BASH_URL="https://www.oilshell.org/blob/spec-bin/bash-$BASH_VER.tar.gz"
|
59 |
|
60 | # Another version of bash to test
|
61 | readonly BASH5_VER=5.2
|
62 | readonly BASH5_URL="https://www.oilshell.org/blob/spec-bin/bash-$BASH5_VER.tar.gz"
|
63 |
|
64 | readonly DASH_VERSION=0.5.10.2
|
65 | readonly DASH_URL="https://www.oilshell.org/blob/spec-bin/dash-$DASH_VERSION.tar.gz"
|
66 |
|
67 | readonly ZSH_VERSION=5.1.1
|
68 | readonly ZSH_URL="https://www.oilshell.org/blob/spec-bin/zsh-$ZSH_VERSION.tar.xz"
|
69 |
|
70 | readonly MKSH_VERSION=R52c
|
71 | readonly MKSH_URL="https://www.oilshell.org/blob/spec-bin/mksh-$MKSH_VERSION.tgz"
|
72 |
|
73 | readonly BUSYBOX_VERSION='1.35.0'
|
74 | readonly BUSYBOX_URL="https://www.oilshell.org/blob/spec-bin/busybox-$BUSYBOX_VERSION.tar.bz2"
|
75 |
|
76 | readonly YASH_VERSION=2.49
|
77 | readonly YASH_URL="https://www.oilshell.org/blob/spec-bin/yash-$YASH_VERSION.tar.xz"
|
78 |
|
79 | readonly MYPY_GIT_URL=https://github.com/python/mypy
|
80 | readonly MYPY_VERSION=0.780
|
81 |
|
82 | readonly PY3_LIBS=~/wedge/oils-for-unix.org/pkg/py3-libs/$MYPY_VERSION
|
83 |
|
84 | # Version 2.4.0 from 2021-10-06 was the last version that supported Python 2
|
85 | # https://github.com/PyCQA/pyflakes/blob/main/NEWS.rst
|
86 | readonly PYFLAKES_VERSION=2.4.0
|
87 | #readonly PYFLAKES_URL='https://files.pythonhosted.org/packages/15/60/c577e54518086e98470e9088278247f4af1d39cb43bcbd731e2c307acd6a/pyflakes-2.4.0.tar.gz'
|
88 | # 2023-07: Mirrored to avoid network problem on broome during release
|
89 | readonly PYFLAKES_URL='https://www.oilshell.org/blob/pyflakes-2.4.0.tar.gz'
|
90 |
|
91 | readonly BLOATY_VERSION=1.1
|
92 | readonly BLOATY_URL='https://github.com/google/bloaty/releases/download/v1.1/bloaty-1.1.tar.bz2'
|
93 |
|
94 | readonly UFTRACE_VERSION=0.13
|
95 | readonly UFTRACE_URL='https://github.com/namhyung/uftrace/archive/refs/tags/v0.13.tar.gz'
|
96 |
|
97 | readonly SOUFFLE_VERSION=2.4.1
|
98 | readonly SOUFFLE_URL=https://github.com/souffle-lang/souffle/archive/refs/tags/2.4.1.tar.gz
|
99 |
|
100 | log() {
|
101 | echo "$0: $@" >& 2
|
102 | }
|
103 |
|
104 | die() {
|
105 | log "$@"
|
106 | exit 1
|
107 | }
|
108 |
|
109 | rm-oils-crap() {
|
110 | ### When you want to start over
|
111 |
|
112 | rm -r -f -v ~/wedge
|
113 | sudo rm -r -f -v /wedge
|
114 | }
|
115 |
|
116 | # Note: git is an implicit dependency -- that's how we got the repo in the
|
117 | # first place!
|
118 |
|
119 | # python2-dev is no longer available on Debian 12
|
120 | # python-dev also seems gone
|
121 | #
|
122 | # wget: for fetching wedges (not on Debian by default!)
|
123 | # tree: tiny package that's useful for showing what we installed
|
124 | # g++: essential
|
125 | # libreadline-dev: needed for the build/prepare.sh Python build.
|
126 | # gawk: used by spec-runner.sh for the special match() function.
|
127 | # cmake: for cmark
|
128 | # PY3_BUILD_DEPS - I think these will be used for building the Python 2 wedge
|
129 | # as well
|
130 | readonly -a WEDGE_DEPS_DEBIAN=(
|
131 | bzip2
|
132 | wget
|
133 | tree
|
134 | gawk
|
135 | g++
|
136 | ninja-build
|
137 | cmake
|
138 | libreadline-dev
|
139 | systemtap-sdt-dev
|
140 |
|
141 | # for Souffle, flex and bison
|
142 | #flex bison
|
143 |
|
144 | "${PY3_BUILD_DEPS[@]}"
|
145 | )
|
146 |
|
147 | readonly -a WEDGE_DEPS_ALPINE=(
|
148 | bzip2
|
149 | xz
|
150 |
|
151 | wget tree gawk
|
152 |
|
153 | gcc g++
|
154 | ninja-build
|
155 | # https://pkgs.alpinelinux.org/packages?name=ninja-is-really-ninja&branch=v3.19&repo=&arch=&maintainer=
|
156 | ninja-is-really-ninja
|
157 | cmake
|
158 |
|
159 | readline-dev
|
160 | zlib-dev
|
161 | libffi-dev
|
162 | openssl-dev
|
163 |
|
164 | ncurses-dev
|
165 |
|
166 | # for Souffle, flex and bison
|
167 | #flex bison
|
168 | )
|
169 |
|
170 | readonly -a WEDGE_DEPS_FEDORA=(
|
171 |
|
172 | # Weird, Fedora doesn't have these by default!
|
173 | hostname
|
174 | tar
|
175 | bzip2
|
176 |
|
177 | # https://packages.fedoraproject.org/pkgs/wget/wget/
|
178 | wget
|
179 | # https://packages.fedoraproject.org/pkgs/tree-pkg/tree/
|
180 | tree
|
181 | gawk
|
182 |
|
183 | # https://packages.fedoraproject.org/pkgs/gcc/gcc/
|
184 | gcc gcc-c++
|
185 |
|
186 | ninja-build
|
187 | cmake
|
188 |
|
189 | readline-devel
|
190 |
|
191 | # Like PY3_BUILD_DEPS
|
192 | # https://packages.fedoraproject.org/pkgs/zlib/zlib-devel/
|
193 | zlib-devel
|
194 | # https://packages.fedoraproject.org/pkgs/libffi/libffi-devel/
|
195 | libffi-devel
|
196 | # https://packages.fedoraproject.org/pkgs/openssl/openssl-devel/
|
197 | openssl-devel
|
198 |
|
199 | # For building zsh from source?
|
200 | # https://koji.fedoraproject.org/koji/rpminfo?rpmID=36987813
|
201 | ncurses-devel
|
202 | #libcap-devel
|
203 |
|
204 | # still have a job control error compiling bash
|
205 | # https://packages.fedoraproject.org/pkgs/glibc/glibc-devel/
|
206 | # glibc-devel
|
207 | )
|
208 |
|
209 | install-debian-packages() {
|
210 | ### Packages for build/py.sh all, building wedges, etc.
|
211 |
|
212 | set -x # show what needs sudo
|
213 |
|
214 | # pass -y for say gitpod
|
215 | sudo apt "$@" install "${WEDGE_DEPS_DEBIAN[@]}"
|
216 | set +x
|
217 |
|
218 | # maybe pass -y through
|
219 | test/spec-bin.sh install-shells-with-apt "$@"
|
220 | }
|
221 |
|
222 | install-ubuntu-packages() {
|
223 | ### Debian and Ubuntu packages are the same; this function is suggested on the wiki
|
224 | install-debian-packages "$@"
|
225 | }
|
226 |
|
227 | wedge-deps-debian() {
|
228 | # Install packages without prompt
|
229 |
|
230 | # 2024-02 - there was an Ubuntu update, and we started needing this
|
231 | sudo apt-get -y update
|
232 |
|
233 | install-debian-packages -y
|
234 | }
|
235 |
|
236 | wedge-deps-fedora() {
|
237 | # https://linuxconfig.org/install-development-tools-on-redhat-8
|
238 | # Trying to get past compile errors
|
239 | # sudo dnf group install --assumeyes 'Development Tools'
|
240 |
|
241 | sudo dnf install --assumeyes "${WEDGE_DEPS_FEDORA[@]}"
|
242 | }
|
243 |
|
244 | wedge-deps-alpine() {
|
245 | # https://linuxconfig.org/install-development-tools-on-redhat-8
|
246 | # Trying to get past compile errors
|
247 | # sudo dnf group install --assumeyes 'Development Tools'
|
248 |
|
249 | sudo apk add "${WEDGE_DEPS_ALPINE[@]}"
|
250 | }
|
251 |
|
252 | #
|
253 | # Unused patch, was experiment for Fedora
|
254 | #
|
255 |
|
256 | get-typed-ast-patch() {
|
257 | curl -o deps/typed_ast.patch https://github.com/python/typed_ast/commit/123286721923ae8f3885dbfbad94d6ca940d5c96.patch
|
258 | }
|
259 |
|
260 | # Work around typed_ast bug:
|
261 | # https://github.com/python/typed_ast/issues/169
|
262 | #
|
263 | # Apply this patch
|
264 | # https://github.com/python/typed_ast/commit/123286721923ae8f3885dbfbad94d6ca940d5c96
|
265 | #
|
266 | # typed_ast is tarred up though
|
267 | patch-typed-ast() {
|
268 | local package_dir=_cache/py3-libs
|
269 | local patch=$PWD/deps/typed_ast.patch
|
270 |
|
271 | pushd $package_dir
|
272 | cat $patch
|
273 | echo
|
274 |
|
275 | local dir=typed_ast-1.4.3
|
276 | local tar=typed_ast-1.4.3.tar.gz
|
277 |
|
278 | echo OLD
|
279 | ls -l $tar
|
280 | echo
|
281 |
|
282 | rm -r -f -v $dir
|
283 | tar -x -z < $tar
|
284 |
|
285 | pushd $dir
|
286 | patch -p1 < $patch
|
287 | popd
|
288 | #find $dir
|
289 |
|
290 | # Create a new one
|
291 | tar --create --gzip --file $tar typed_ast-1.4.3
|
292 |
|
293 | echo NEW
|
294 | ls -l $tar
|
295 | echo
|
296 |
|
297 | popd
|
298 | }
|
299 |
|
300 | #
|
301 | # Fetch
|
302 | #
|
303 |
|
304 | download-to() {
|
305 | local dir=$1
|
306 | local url=$2
|
307 | wget --no-clobber --directory-prefix "$dir" "$url"
|
308 | }
|
309 |
|
310 | maybe-extract() {
|
311 | local wedge_dir=$1
|
312 | local tar_name=$2
|
313 | local out_dir=$3
|
314 |
|
315 | if test -d "$wedge_dir/$out_dir"; then
|
316 | log "Not extracting because $wedge_dir/$out_dir exists"
|
317 | return
|
318 | fi
|
319 |
|
320 | local tar=$wedge_dir/$tar_name
|
321 | case $tar_name in
|
322 | *.gz|*.tgz) # mksh ends with .tgz
|
323 | flag='--gzip'
|
324 | ;;
|
325 | *.bz2)
|
326 | flag='--bzip2'
|
327 | ;;
|
328 | *.xz)
|
329 | flag='--xz'
|
330 | ;;
|
331 | *)
|
332 | die "tar with unknown extension: $tar_name"
|
333 | ;;
|
334 | esac
|
335 |
|
336 | tar --extract $flag --file $tar --directory $wedge_dir
|
337 | }
|
338 |
|
339 | clone-mypy() {
|
340 | ### replaces deps/from-git
|
341 | local dest_dir=$1
|
342 | local version=${2:-$MYPY_VERSION}
|
343 |
|
344 | local dest=$dest_dir/mypy-$version
|
345 | if test -d $dest; then
|
346 | log "Not cloning because $dest exists"
|
347 | return
|
348 | fi
|
349 |
|
350 | # v$VERSION is a tag, not a branch
|
351 |
|
352 | # size optimization: --depth=1 --shallow-submodules
|
353 | # https://git-scm.com/docs/git-clone
|
354 |
|
355 | git clone --recursive --branch v$version \
|
356 | --depth=1 --shallow-submodules \
|
357 | $MYPY_GIT_URL $dest
|
358 |
|
359 | # TODO: verify commit checksum
|
360 | }
|
361 |
|
362 | copy-source-medo() {
|
363 | mkdir -p $DEPS_SOURCE_DIR
|
364 |
|
365 | # Copy the whole tree, including the .treeptr files
|
366 | cp --verbose --recursive --no-target-directory \
|
367 | deps/source.medo/ $DEPS_SOURCE_DIR/
|
368 | }
|
369 |
|
370 | fetch() {
|
371 | local py_only=${1:-}
|
372 |
|
373 | # For now, simulate what 'medo expand deps/source.medo _build/deps-source'
|
374 | # would do: fetch compressed tarballs designated by .treeptr files, and
|
375 | # expand them.
|
376 |
|
377 | # _build/deps-source/
|
378 | # re2c/
|
379 | # WEDGE
|
380 | # re2c-3.0/ # expanded .tar.xz file
|
381 |
|
382 | copy-source-medo
|
383 |
|
384 | download-to $DEPS_SOURCE_DIR/re2c "$RE2C_URL"
|
385 | download-to $DEPS_SOURCE_DIR/cmark "$CMARK_URL"
|
386 | maybe-extract $DEPS_SOURCE_DIR/re2c "$(basename $RE2C_URL)" re2c-$RE2C_VERSION
|
387 | maybe-extract $DEPS_SOURCE_DIR/cmark "$(basename $CMARK_URL)" cmark-$CMARK_VERSION
|
388 |
|
389 | if test -n "$py_only"; then
|
390 | log "Fetched dependencies for 'build/py.sh'"
|
391 | return
|
392 | fi
|
393 |
|
394 | download-to $DEPS_SOURCE_DIR/pyflakes "$PYFLAKES_URL"
|
395 | maybe-extract $DEPS_SOURCE_DIR/pyflakes "$(basename $PYFLAKES_URL)" \
|
396 | pyflakes-$PYFLAKES_VERSION
|
397 |
|
398 | download-to $DEPS_SOURCE_DIR/python2 "$PY2_URL"
|
399 | download-to $DEPS_SOURCE_DIR/python3 "$PY3_URL"
|
400 | maybe-extract $DEPS_SOURCE_DIR/python2 "$(basename $PY2_URL)" Python-$PY2_VERSION
|
401 | maybe-extract $DEPS_SOURCE_DIR/python3 "$(basename $PY3_URL)" Python-$PY3_VERSION
|
402 |
|
403 | download-to $DEPS_SOURCE_DIR/bash "$BASH_URL"
|
404 | maybe-extract $DEPS_SOURCE_DIR/bash "$(basename $BASH_URL)" bash-$BASH_VER
|
405 |
|
406 | download-to $DEPS_SOURCE_DIR/bash "$BASH5_URL"
|
407 | maybe-extract $DEPS_SOURCE_DIR/bash "$(basename $BASH5_URL)" bash-$BASH5_VER
|
408 |
|
409 | download-to $DEPS_SOURCE_DIR/dash "$DASH_URL"
|
410 | maybe-extract $DEPS_SOURCE_DIR/dash "$(basename $DASH_URL)" dash-$DASH_VERSION
|
411 |
|
412 | download-to $DEPS_SOURCE_DIR/zsh "$ZSH_URL"
|
413 | maybe-extract $DEPS_SOURCE_DIR/zsh "$(basename $ZSH_URL)" zsh-$ZSH_VERSION
|
414 |
|
415 | download-to $DEPS_SOURCE_DIR/mksh "$MKSH_URL"
|
416 | maybe-extract $DEPS_SOURCE_DIR/mksh "$(basename $MKSH_URL)" mksh-$MKSH_VERSION
|
417 |
|
418 | download-to $DEPS_SOURCE_DIR/busybox "$BUSYBOX_URL"
|
419 | maybe-extract $DEPS_SOURCE_DIR/busybox "$(basename $BUSYBOX_URL)" busybox-$BUSYBOX_VERSION
|
420 |
|
421 | download-to $DEPS_SOURCE_DIR/yash "$YASH_URL"
|
422 | maybe-extract $DEPS_SOURCE_DIR/yash "$(basename $YASH_URL)" yash-$DASH_VERSION
|
423 |
|
424 | # Patch: this tarball doesn't follow the convention $name-$version
|
425 | if test -d $DEPS_SOURCE_DIR/mksh/mksh; then
|
426 | pushd $DEPS_SOURCE_DIR/mksh
|
427 | mv -v mksh mksh-$MKSH_VERSION
|
428 | popd
|
429 | fi
|
430 |
|
431 | # bloaty and uftrace are for benchmarks, in containers
|
432 | download-to $DEPS_SOURCE_DIR/bloaty "$BLOATY_URL"
|
433 | download-to $DEPS_SOURCE_DIR/uftrace "$UFTRACE_URL"
|
434 | maybe-extract $DEPS_SOURCE_DIR/bloaty "$(basename $BLOATY_URL)" uftrace-$BLOATY_VERSION
|
435 | maybe-extract $DEPS_SOURCE_DIR/uftrace "$(basename $UFTRACE_URL)" bloaty-$UFTRACE_VERSION
|
436 |
|
437 | # This is in $DEPS_SOURCE_DIR to COPY into containers, which mycpp will directly import.
|
438 |
|
439 | # It's also copied into a wedge in install-wedges.
|
440 | clone-mypy $DEPS_SOURCE_DIR/mypy
|
441 |
|
442 | if false; then
|
443 | download-to $DEPS_SOURCE_DIR/souffle "$SOUFFLE_URL"
|
444 | maybe-extract $DEPS_SOURCE_DIR/souffle "$(basename $SOUFFLE_URL)" souffle-$SOUFFLE_VERSION
|
445 | fi
|
446 |
|
447 | if command -v tree > /dev/null; then
|
448 | tree -L 2 $DEPS_SOURCE_DIR
|
449 | fi
|
450 | }
|
451 |
|
452 | fetch-py() {
|
453 | fetch py_only
|
454 | }
|
455 |
|
456 | mirror-pyflakes() {
|
457 | ### Workaround for network error during release
|
458 | scp \
|
459 | $DEPS_SOURCE_DIR/pyflakes/"$(basename $PYFLAKES_URL)" \
|
460 | oilshell.org:oilshell.org/blob/
|
461 | }
|
462 |
|
463 | wedge-exists() {
|
464 | ### Does an installed wedge already exist?
|
465 |
|
466 | local name=$1
|
467 | local version=$2
|
468 | local wedge_dir=${3:-/wedge/oils-for-unix.org}
|
469 |
|
470 | local installed=$wedge_dir/pkg/$name/$version
|
471 |
|
472 | if test -d $installed; then
|
473 | log "$installed already exists"
|
474 | return 0
|
475 | else
|
476 | return 1
|
477 | fi
|
478 | }
|
479 |
|
480 | #
|
481 | # Install
|
482 | #
|
483 |
|
484 | # TODO: py3-libs needs to be a WEDGE, so that that you can run
|
485 | # 'wedge build deps/source.medo/py3-libs/' and then get it in
|
486 | #
|
487 | # _build/wedge/{absolute,relative} # which one?
|
488 | #
|
489 | # It needs a BUILD DEPENDENCY on:
|
490 | # - the python3 wedge, so you can do python3 -m pip install.
|
491 | # - the mypy repo, which has test-requirements.txt
|
492 |
|
493 | download-py3-libs() {
|
494 | ### Download source/binary packages, AFTER python3 is installed
|
495 |
|
496 | # Note that this is NOT source code; there is binary code, e.g. in
|
497 | # lxml-*.whl
|
498 |
|
499 | local mypy_dir=${1:-$DEPS_SOURCE_DIR/mypy/mypy-$MYPY_VERSION}
|
500 | local py_package_dir=_cache/py3-libs
|
501 | mkdir -p $py_package_dir
|
502 |
|
503 | # Avoids a warning, but doesn't fix typed_ast
|
504 | #python3 -m pip download -d $py_package_dir wheel
|
505 |
|
506 | python3 -m pip download -d $py_package_dir -r $mypy_dir/test-requirements.txt
|
507 | python3 -m pip download -d $py_package_dir pexpect
|
508 | }
|
509 |
|
510 | install-py3-libs-in-venv() {
|
511 | local venv_dir=$1
|
512 | local mypy_dir=$2 # This is a param for host build vs. container build
|
513 | local package_dir=_cache/py3-libs
|
514 |
|
515 | source $venv_dir/bin/activate # enter virtualenv
|
516 |
|
517 | # 2023-07 note: we're installing yapf in a DIFFERENT venv, because it
|
518 | # conflicts with MyPy deps!
|
519 | # "ERROR: pip's dependency resolver does not currently take into account all
|
520 | # the packages that are installed."
|
521 |
|
522 | # --find-links uses a "cache dir" for packages (weird flag name!)
|
523 |
|
524 | # Avoids a warning, but doesn't fix typed_ast
|
525 | #time python3 -m pip install --find-links $package_dir wheel
|
526 |
|
527 | # for mycpp/
|
528 | time python3 -m pip install --find-links $package_dir -r $mypy_dir/test-requirements.txt
|
529 |
|
530 | # pexpect: for spec/stateful/*.py
|
531 | time python3 -m pip install --find-links $package_dir pexpect
|
532 | }
|
533 |
|
534 | install-py3-libs-from-cache() {
|
535 |
|
536 | # As well as end users
|
537 |
|
538 | local mypy_dir=${1:-$DEPS_SOURCE_DIR/mypy/mypy-$MYPY_VERSION}
|
539 |
|
540 | local py3
|
541 | py3=$(command -v python3)
|
542 | case $py3 in
|
543 | *wedge/oils-for-unix.org/*)
|
544 | ;;
|
545 | *)
|
546 | die "python3 is '$py3', but expected it to be in a wedge"
|
547 | ;;
|
548 | esac
|
549 |
|
550 | log "Ensuring pip is installed (interpreter $(command -v python3)"
|
551 | python3 -m ensurepip
|
552 |
|
553 | local venv_dir=$USER_WEDGE_DIR/pkg/py3-libs/$PY3_LIBS_VERSION
|
554 | log "Creating venv in $venv_dir"
|
555 |
|
556 | # Note: the bin/python3 in this venv is a symlink to python3 in $PATH, i.e.
|
557 | # the /wedge we just built
|
558 | python3 -m venv $venv_dir
|
559 |
|
560 | log "Installing MyPy deps in venv"
|
561 |
|
562 | # Run in a subshell because it mutates shell state
|
563 | $0 install-py3-libs-in-venv $venv_dir $mypy_dir
|
564 | }
|
565 |
|
566 | install-py3-libs() {
|
567 | ### Invoked by Dockerfile.cpp-small, etc.
|
568 |
|
569 | download-py3-libs
|
570 | install-py3-libs-from-cache
|
571 | }
|
572 |
|
573 | # OBSOLETE in favor of install-spec-bin-fast
|
574 | install-spec-bin() {
|
575 | if ! wedge-exists dash $DASH_VERSION $USER_WEDGE_DIR; then
|
576 | deps/wedge.sh unboxed _build/deps-source/dash
|
577 | fi
|
578 |
|
579 | if ! wedge-exists mksh $MKSH_VERSION $USER_WEDGE_DIR; then
|
580 | deps/wedge.sh unboxed _build/deps-source/mksh
|
581 | fi
|
582 |
|
583 | if ! wedge-exists busybox $BUSYBOX_VERSION $USER_WEDGE_DIR; then
|
584 | deps/wedge.sh unboxed _build/deps-source/busybox
|
585 | fi
|
586 |
|
587 | # Fedora compile error - count_all_jobs
|
588 | if ! wedge-exists bash $BASH_VER $USER_WEDGE_DIR; then
|
589 | deps/wedge.sh unboxed _build/deps-source/bash
|
590 | fi
|
591 |
|
592 | # Fedora compiler error
|
593 | # zsh ./configure is NOT detecting 'boolcodes', and then it has a broken
|
594 | # fallback in Src/Modules/termcap.c that causes a compile error! It seems
|
595 | # like ncurses-devel should fix this, but it doesn't
|
596 | #
|
597 | # https://koji.fedoraproject.org/koji/rpminfo?rpmID=36987813
|
598 | #
|
599 | # from /home/build/oil/_build/deps-source/zsh/zsh-5.1.1/Src/Modules/termcap.c:38:
|
600 | # /usr/include/term.h:783:56: note: previous declaration of ‘boolcodes’ with type ‘const char * const[]’
|
601 | # 783 | extern NCURSES_EXPORT_VAR(NCURSES_CONST char * const ) boolcodes[];
|
602 | #
|
603 | # I think the ./configure is out of sync with the actual build?
|
604 |
|
605 | if ! wedge-exists zsh $ZSH_VERSION ''; then
|
606 | deps/wedge.sh unboxed _build/deps-source/zsh
|
607 | fi
|
608 |
|
609 | return
|
610 |
|
611 | # Hm this has problem with out-of-tree build? I think Oils does too actually
|
612 | if ! wedge-exists yash $YASH_VERSION $USER_WEDGE_DIR; then
|
613 | deps/wedge.sh unboxed _build/deps-source/yash
|
614 | fi
|
615 | }
|
616 |
|
617 | # TODO:
|
618 | # - $ROOT_WEDGE_DIR vs. $USER_WEDGE_DIR is duplicating information that's
|
619 | # already in each WEDGE file
|
620 |
|
621 | py-wedges() {
|
622 | ### for build/py.sh all
|
623 |
|
624 | echo cmark $CMARK_VERSION $ROOT_WEDGE_DIR
|
625 | echo re2c $RE2C_VERSION $ROOT_WEDGE_DIR
|
626 | echo python2 $PY2_VERSION $ROOT_WEDGE_DIR
|
627 | echo pyflakes $PYFLAKES_VERSION $USER_WEDGE_DIR
|
628 | }
|
629 |
|
630 | cpp-wedges() {
|
631 | ### for ninja / mycpp translation
|
632 |
|
633 | echo python3 $PY3_VERSION $ROOT_WEDGE_DIR
|
634 | echo mypy $MYPY_VERSION $USER_WEDGE_DIR
|
635 | #echo souffle $SOUFFLE_VERSION $USER_WEDGE_DIR
|
636 |
|
637 | # py3-libs has a built time dep on both python3 and MyPy, so we're doing it
|
638 | # separately for now
|
639 | #echo py3-libs $PY3_LIBS_VERSION $USER_WEDGE_DIR
|
640 | }
|
641 |
|
642 | spec-bin-wedges() {
|
643 | ### for test/spec-py.sh osh-all
|
644 |
|
645 | echo dash $DASH_VERSION $USER_WEDGE_DIR
|
646 | echo bash $BASH_VER $USER_WEDGE_DIR
|
647 | echo bash $BASH5_VER $USER_WEDGE_DIR
|
648 | echo mksh $MKSH_VERSION $USER_WEDGE_DIR
|
649 | echo zsh $ZSH_VERSION $USER_WEDGE_DIR
|
650 | echo busybox $BUSYBOX_VERSION $USER_WEDGE_DIR
|
651 | }
|
652 |
|
653 | timestamp() {
|
654 | date '+%H:%M:%S'
|
655 | }
|
656 |
|
657 | my-time-tsv() {
|
658 | python3 benchmarks/time_.py \
|
659 | --tsv \
|
660 | --time-span --rusage \
|
661 | "$@"
|
662 | }
|
663 |
|
664 | maybe-install-wedge() {
|
665 | local name=$1
|
666 | local version=$2
|
667 | local wedge_dir=$3 # e.g. $USER_WEDGE_DIR or empty
|
668 |
|
669 | local task_file=$WEDGE_LOG_DIR/$name-$version.task.tsv
|
670 | local log_file=$WEDGE_LOG_DIR/$name-$version.log.txt
|
671 |
|
672 | echo " TASK $(timestamp) $name $version > $log_file"
|
673 |
|
674 | # python3 because it's OUTSIDE the container
|
675 | # Separate columns that could be joined: number of files, total size
|
676 | my-time-tsv --print-header \
|
677 | --field xargs_slot \
|
678 | --field wedge \
|
679 | --field wedge_HREF \
|
680 | --field version \
|
681 | --output $task_file
|
682 |
|
683 | if wedge-exists "$name" "$version" "$wedge_dir"; then
|
684 | echo "CACHED $(timestamp) $name $version"
|
685 | return
|
686 | fi
|
687 |
|
688 | local -a cmd=( deps/wedge.sh unboxed _build/deps-source/$name/ $version)
|
689 |
|
690 | set +o errexit
|
691 | my-time-tsv \
|
692 | --field "$XARGS_SLOT" \
|
693 | --field "$name" \
|
694 | --field "$name-$version.log.txt" \
|
695 | --field "$version" \
|
696 | --append \
|
697 | --output $task_file \
|
698 | "${cmd[@]}" "$@" >$log_file 2>&1
|
699 | local status=$?
|
700 | set -o errexit
|
701 |
|
702 | if test "$status" -eq 0; then
|
703 | echo " OK $(timestamp) $name $version"
|
704 | else
|
705 | echo " FAIL $(timestamp) $name $version"
|
706 | fi
|
707 | }
|
708 |
|
709 | dummy-task() {
|
710 | ### For testing log capture
|
711 | local name=$1
|
712 | local version=$2
|
713 |
|
714 | echo "Building $name $version"
|
715 |
|
716 | # random float between 0 and 3
|
717 | # weirdly we need a seed from bash
|
718 | # https://stackoverflow.com/questions/4048378/random-numbers-generation-with-awk-in-bash-shell
|
719 | local secs
|
720 | secs=$(awk -v seed=$RANDOM 'END { srand(seed); print rand() * 3 }' < /dev/null)
|
721 |
|
722 | echo "sleep $secs"
|
723 | sleep $secs
|
724 |
|
725 | echo 'stdout'
|
726 | log 'stderr'
|
727 |
|
728 | if test $name = 'mksh'; then
|
729 | echo "simulate failure for $name"
|
730 | exit 2
|
731 | fi
|
732 | }
|
733 |
|
734 | readonly WEDGE_LOG_DIR=_build/wedge/logs
|
735 |
|
736 | dummy-task-wrapper() {
|
737 | # Similar to test/common.sh run-task-with-status, used by
|
738 | # test/{spec,wild}-runner.sh
|
739 | local name=$1
|
740 | local version=$2
|
741 |
|
742 | local task_file=$WEDGE_LOG_DIR/$name.task.tsv
|
743 | local log_file=$WEDGE_LOG_DIR/$name.log.txt
|
744 |
|
745 | echo " TASK $(timestamp) $name $version > $log_file"
|
746 |
|
747 | # python3 because it's OUTSIDE the container
|
748 | # Separate columns that could be joined: number of files, total size
|
749 | my-time-tsv --print-header \
|
750 | --field xargs_slot \
|
751 | --field wedge \
|
752 | --field wedge_HREF \
|
753 | --field version \
|
754 | --output $task_file
|
755 |
|
756 | my-time-tsv \
|
757 | --field "$XARGS_SLOT" \
|
758 | --field "$name" \
|
759 | --field "$name.log.txt" \
|
760 | --field "$version" \
|
761 | --append \
|
762 | --output $task_file \
|
763 | $0 dummy-task "$@" >$log_file 2>&1 || true
|
764 |
|
765 | echo " DONE $(timestamp) $name $version"
|
766 | }
|
767 |
|
768 | html-head() {
|
769 | # python3 because we're outside containers
|
770 | PYTHONPATH=. python3 doctools/html_head.py "$@"
|
771 | }
|
772 |
|
773 | index-html() {
|
774 | local tasks_tsv=$1
|
775 |
|
776 | local base_url='../../../web'
|
777 | html-head --title 'Wedge Builds' \
|
778 | "$base_url/ajax.js" \
|
779 | "$base_url/table/table-sort.js" \
|
780 | "$base_url/table/table-sort.css" \
|
781 | "$base_url/base.css"\
|
782 |
|
783 | table-sort-begin 'width60'
|
784 |
|
785 | cat <<EOF
|
786 | <p id="home-link">
|
787 | <a href="/">oilshell.org</a>
|
788 | </p>
|
789 |
|
790 | <h1>Wedge Builds</h1>
|
791 | EOF
|
792 |
|
793 | tsv2html3 $tasks_tsv
|
794 |
|
795 | cat <<EOF
|
796 | <p>
|
797 | <a href="tasks.tsv">tasks.tsv</a>
|
798 | </p>
|
799 | EOF
|
800 |
|
801 | table-sort-end 'tasks' # ID for sorting
|
802 | }
|
803 |
|
804 | NPROC=$(nproc)
|
805 | #NPROC=1
|
806 |
|
807 | install-wedge-list() {
|
808 | ### Reads task rows from stdin
|
809 | local parallel=${1:-}
|
810 |
|
811 | mkdir -p _build/wedge/logs
|
812 |
|
813 | local -a flags
|
814 | if test -n "$parallel"; then
|
815 | log ""
|
816 | log "=== Installing wedges with $NPROC jobs in parallel"
|
817 | log ""
|
818 | flags=( -P $NPROC )
|
819 | else
|
820 | log ""
|
821 | log "=== Installing wedges serially"
|
822 | log ""
|
823 | fi
|
824 |
|
825 | # Reads from stdin
|
826 | # Note: --process-slot-var requires GNU xargs! busybox args doesn't have it.
|
827 | #
|
828 | # $name $version $wedge_dir
|
829 | xargs "${flags[@]}" -n 3 --process-slot-var=XARGS_SLOT -- $0 maybe-install-wedge
|
830 |
|
831 | #xargs "${flags[@]}" -n 3 --process-slot-var=XARGS_SLOT -- $0 dummy-task-wrapper
|
832 | }
|
833 |
|
834 | write-task-report() {
|
835 | local tasks_tsv=_build/wedge/logs/tasks.tsv
|
836 |
|
837 | python3 devtools/tsv_concat.py $WEDGE_LOG_DIR/*.task.tsv > $tasks_tsv
|
838 | log "Wrote $tasks_tsv"
|
839 |
|
840 | # TODO: version can be right-justified?
|
841 | here-schema-tsv-4col >_build/wedge/logs/tasks.schema.tsv <<EOF
|
842 | column_name type precision strftime
|
843 | status integer 0 -
|
844 | elapsed_secs float 1 -
|
845 | user_secs float 1 -
|
846 | start_time float 1 %H:%M:%S
|
847 | end_time float 1 %H:%M:%S
|
848 | sys_secs float 1 -
|
849 | max_rss_KiB integer 0 -
|
850 | xargs_slot integer 0 -
|
851 | wedge string 0 -
|
852 | wedge_HREF string 0 -
|
853 | version string 0 -
|
854 | EOF
|
855 |
|
856 | index-html $tasks_tsv > $WEDGE_LOG_DIR/index.html
|
857 | log "Wrote $WEDGE_LOG_DIR/index.html"
|
858 | }
|
859 |
|
860 | install-spec-bin-fast() {
|
861 | spec-bin-wedges | install-wedge-list T
|
862 | write-task-report
|
863 | }
|
864 |
|
865 | fake-py3-libs-wedge() {
|
866 | local name=py3-libs
|
867 | local version=$PY3_LIBS_VERSION
|
868 |
|
869 | local task_file=$WEDGE_LOG_DIR/$name.task.tsv
|
870 | local log_file=$WEDGE_LOG_DIR/$name.log.txt
|
871 |
|
872 | my-time-tsv --print-header \
|
873 | --field xargs_slot \
|
874 | --field wedge \
|
875 | --field wedge_HREF \
|
876 | --field version \
|
877 | --output $task_file
|
878 |
|
879 | # There is no xargs slot!
|
880 | my-time-tsv \
|
881 | --field "-1" \
|
882 | --field "$name" \
|
883 | --field "$name.log.txt" \
|
884 | --field "$version" \
|
885 | --append \
|
886 | --output $task_file \
|
887 | $0 install-py3-libs >$log_file 2>&1 || true
|
888 |
|
889 | echo " FAKE $(timestamp) $name $version"
|
890 | }
|
891 |
|
892 | install-wedges-fast() {
|
893 | echo " START $(timestamp)"
|
894 |
|
895 | # Do all of them in parallel
|
896 | { py-wedges; cpp-wedges; spec-bin-wedges; } | install-wedge-list T
|
897 |
|
898 | fake-py3-libs-wedge
|
899 | echo " END $(timestamp)"
|
900 |
|
901 | write-task-report
|
902 | }
|
903 |
|
904 | # OBSOLETE in favor of install-wedges-fast
|
905 | install-wedges() {
|
906 | local py_only=${1:-}
|
907 |
|
908 | # TODO:
|
909 | # - Make all of these RELATIVE wedges
|
910 | # - Add
|
911 | # - unboxed-rel-smoke-test -- move it inside container
|
912 | # - rel-smoke-test -- mount it in a different location
|
913 |
|
914 | if ! wedge-exists cmark $CMARK_VERSION; then
|
915 | deps/wedge.sh unboxed _build/deps-source/cmark/
|
916 | fi
|
917 |
|
918 | if ! wedge-exists re2c $RE2C_VERSION; then
|
919 | deps/wedge.sh unboxed _build/deps-source/re2c/
|
920 | fi
|
921 |
|
922 | if ! wedge-exists python2 $PY2_VERSION; then
|
923 | deps/wedge.sh unboxed _build/deps-source/python2/
|
924 | fi
|
925 |
|
926 | if test -n "$py_only"; then
|
927 | log "Installed dependencies for 'build/py.sh'"
|
928 | return
|
929 | fi
|
930 |
|
931 | # Just copy this source tarball
|
932 | if ! wedge-exists pyflakes $PYFLAKES_VERSION $USER_WEDGE_DIR; then
|
933 | local dest_dir=$USER_WEDGE_DIR/pkg/pyflakes/$PYFLAKES_VERSION
|
934 | mkdir -p $dest_dir
|
935 |
|
936 | cp --verbose --recursive --no-target-directory \
|
937 | $DEPS_SOURCE_DIR/pyflakes/pyflakes-$PYFLAKES_VERSION $dest_dir
|
938 | fi
|
939 |
|
940 | if ! wedge-exists python3 $PY3_VERSION; then
|
941 | deps/wedge.sh unboxed _build/deps-source/python3/
|
942 | fi
|
943 |
|
944 | # Copy all the contents, except for .git folder.
|
945 | if ! wedge-exists mypy $MYPY_VERSION $USER_WEDGE_DIR; then
|
946 |
|
947 | # NOTE: We have to also copy the .git dir, because it has
|
948 | # .git/modules/typeshed
|
949 | local dest_dir=$USER_WEDGE_DIR/pkg/mypy/$MYPY_VERSION
|
950 | mkdir -p $dest_dir
|
951 |
|
952 | # Note: pack files in .git/modules/typeshed/objects/pack are read-only
|
953 | # this can fail
|
954 | cp --verbose --recursive --no-target-directory \
|
955 | $DEPS_SOURCE_DIR/mypy/mypy-$MYPY_VERSION $dest_dir
|
956 | fi
|
957 |
|
958 | if ! wedge-exists py3-libs $PY3_LIBS_VERSION $USER_WEDGE_DIR; then
|
959 | download-py3-libs
|
960 | # This patch doesn't work?
|
961 | # patch-typed-ast
|
962 | install-py3-libs
|
963 | fi
|
964 |
|
965 | if ! wedge-exists souffle $SOUFFLE_VERSION $USER_WEDGE_DIR; then
|
966 | deps/wedge.sh unboxed _build/deps-source/souffle/
|
967 | fi
|
968 |
|
969 | if command -v tree > /dev/null; then
|
970 | tree -L 3 $USER_WEDGE_DIR
|
971 | echo
|
972 | tree -L 3 /wedge/oils-for-unix.org
|
973 | fi
|
974 | }
|
975 |
|
976 | install-wedges-py() {
|
977 | install-wedges py_only
|
978 | }
|
979 |
|
980 | #
|
981 | # Unboxed wedge builds
|
982 | #
|
983 |
|
984 | uftrace-host() {
|
985 | ### built on demand; run $0 first
|
986 |
|
987 | # BUG: doesn't detect python3
|
988 | # WEDGE tells me that it depends on pkg-config
|
989 | # 'apt-get install pkgconf' gets it
|
990 | # TODO: Should use python3 WEDGE instead of SYSTEM python3?
|
991 |
|
992 | deps/wedge.sh unboxed _build/deps-source/uftrace
|
993 | }
|
994 |
|
995 | bloaty-host() {
|
996 | deps/wedge.sh unboxed _build/deps-source/bloaty
|
997 | }
|
998 |
|
999 | R-libs-host() {
|
1000 | deps/wedge.sh unboxed _build/deps-source/R-libs
|
1001 | }
|
1002 |
|
1003 | #
|
1004 | # Wedges built inside a container, for copying into a container
|
1005 | #
|
1006 |
|
1007 | container-wedges() {
|
1008 | #### host _build/wedge/binary -> guest container /wedge or ~/wedge
|
1009 |
|
1010 | #export-podman
|
1011 |
|
1012 | # TODO:
|
1013 | #
|
1014 | # - Add equivalents of spec-bin
|
1015 | # - Use the same manifest as install-wedges-fast
|
1016 | # - so then you can delete the _build/wedge dir to re-run it
|
1017 | # - use xargs -n 1 so it's done serially
|
1018 |
|
1019 | # - Do these lazily like we do in install-wedges-fast?
|
1020 |
|
1021 | # We can test if the dir _build/wedge/binary/oils-for-unix.org/pkg/FOO exists
|
1022 | # if wedge-exists "$name" "$version" "$wedge_dir"; then
|
1023 | # echo "CACHED $(timestamp) $name $version"
|
1024 | # return
|
1025 | # fi
|
1026 |
|
1027 | if false; then
|
1028 | deps/wedge.sh boxed deps/source.medo/time-helper
|
1029 | deps/wedge.sh boxed deps/source.medo/cmark/
|
1030 | deps/wedge.sh boxed deps/source.medo/re2c/
|
1031 | deps/wedge.sh boxed deps/source.medo/python3/
|
1032 | fi
|
1033 |
|
1034 | if false; then
|
1035 | deps/wedge.sh boxed deps/source.medo/bloaty/
|
1036 | fi
|
1037 |
|
1038 | if true; then
|
1039 | # build with debian-12, because soil-benchmarks2 is, because it has R
|
1040 | #deps/wedge.sh boxed deps/source.medo/uftrace/ '' debian-12
|
1041 | # python2 needed everywhere
|
1042 | #deps/wedge.sh boxed deps/source.medo/python2/ '' debian-12
|
1043 |
|
1044 | # TODO: build with debian-12
|
1045 | # Used in {benchmarks,benchmarks2,other-tests}
|
1046 | deps/wedge.sh boxed deps/source.medo/R-libs/ '' debian-12
|
1047 | fi
|
1048 | }
|
1049 |
|
1050 | #
|
1051 | # Report
|
1052 | #
|
1053 |
|
1054 | commas() {
|
1055 | # Wow I didn't know this :a trick
|
1056 | #
|
1057 | # OK this is a label and a loop, which makes sense. You can't do it with
|
1058 | # pure regex.
|
1059 | #
|
1060 | # https://shallowsky.com/blog/linux/cmdline/sed-improve-comma-insertion.html
|
1061 | # https://shallowsky.com/blog/linux/cmdline/sed-improve-comma-insertion.html
|
1062 | sed ':a;s/\b\([0-9]\+\)\([0-9]\{3\}\)\b/\1,\2/;ta'
|
1063 | }
|
1064 |
|
1065 | wedge-sizes() {
|
1066 | local tmp=_tmp/wedge-sizes.txt
|
1067 |
|
1068 | # -b is --bytes, but use short flag for busybox compat
|
1069 | du -s -b /wedge/*/*/* ~/wedge/*/*/* | awk '
|
1070 | { print $0 # print the line
|
1071 | total_bytes += $1 # accumulate
|
1072 | }
|
1073 | END { print total_bytes " TOTAL" }
|
1074 | ' > $tmp
|
1075 |
|
1076 | # printf justifies du output
|
1077 | cat $tmp | commas | xargs -n 2 printf '%15s %s\n'
|
1078 | echo
|
1079 |
|
1080 | #du -s --si /wedge/*/*/* ~/wedge/*/*/*
|
1081 | #echo
|
1082 | }
|
1083 |
|
1084 | wedge-report() {
|
1085 | # 4 levels deep shows the package
|
1086 | if command -v tree > /dev/null; then
|
1087 | tree -L 4 /wedge ~/wedge
|
1088 | echo
|
1089 | fi
|
1090 |
|
1091 | wedge-sizes
|
1092 |
|
1093 | local tmp=_tmp/wedge-manifest.txt
|
1094 |
|
1095 | echo 'Biggest files'
|
1096 | if ! find /wedge ~/wedge -type f -a -printf '%10s %P\n' > $tmp; then
|
1097 | # busybox find doesn't have -printf
|
1098 | echo 'find -printf failed'
|
1099 | return
|
1100 | fi
|
1101 |
|
1102 | set +o errexit # ignore SIGPIPE
|
1103 | sort -n --reverse $tmp | head -n 20 | commas
|
1104 | set -o errexit
|
1105 |
|
1106 | echo
|
1107 |
|
1108 | # Show the most common file extensions
|
1109 | #
|
1110 | # I feel like we should be able to get rid of .a files? That's 92 MB, second
|
1111 | # most common
|
1112 | #
|
1113 | # There are also duplicate .a files for Python -- should look at how distros
|
1114 | # get rid of those
|
1115 |
|
1116 | cat $tmp | python3 -c '
|
1117 | import os, sys, collections
|
1118 |
|
1119 | bytes = collections.Counter()
|
1120 | files = collections.Counter()
|
1121 |
|
1122 | for line in sys.stdin:
|
1123 | size, path = line.split(None, 1)
|
1124 | path = path.strip() # remove newline
|
1125 | _, ext = os.path.splitext(path)
|
1126 | size = int(size)
|
1127 |
|
1128 | bytes[ext] += size
|
1129 | files[ext] += 1
|
1130 |
|
1131 | #print(bytes)
|
1132 | #print(files)
|
1133 |
|
1134 | n = 20
|
1135 |
|
1136 | print("Most common file types")
|
1137 | for ext, count in files.most_common()[:n]:
|
1138 | print("%10d %s" % (count, ext))
|
1139 |
|
1140 | print()
|
1141 |
|
1142 | print("Total bytes by file type")
|
1143 | for ext, total_bytes in bytes.most_common()[:n]:
|
1144 | print("%10d %s" % (total_bytes, ext))
|
1145 | ' | commas
|
1146 | }
|
1147 |
|
1148 | run-task "$@"
|