forked from sagemath/sage
-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathsage-dist-helpers
484 lines (436 loc) · 15.6 KB
/
sage-dist-helpers
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
# -*- shell-script -*- functions for making spkg-install scripts a little easier to write,
# eliminating duplication. All Sage helper functions begin with sdh_ (for
# Sage-distribution helper). Consult the below documentation for the list of
# available helper functions.
#
# This documentation is also repeated in the Sage docs in
# src/doc/en/developer/packaging.rst, so if anything here changes, or
# if you add anything, please modify that file accordingly.
#
# - sdh_die MESSAGE
#
# Exit the build script with the error code of the last command if it was
# non-zero, or with 1 otherwise, and print an error message.
# Typically used like:
#
# command || sdh_die "Command failed"
#
# This function can also (if not given any arguments) read the error message
# from stdin. In particular this is useful in conjunction with a heredoc to
# write multi-line error messages:
#
# command || sdh_die << _EOF_
# Command failed.
# Reason given.
# _EOF_
#
# - sdh_check_vars [VARIABLE ...]
#
# Check that one or more variables are defined and non-empty, and exit with
# an error if any are undefined or empty. Variable names should be given
# without the '$' to prevent unwanted expansion.
#
# - sdh_guard
#
# Wrapper for `sdh_check_vars` that checks some common variables without
# which many/most packages won't build correctly (SAGE_ROOT, SAGE_LOCAL,
# SAGE_SHARE). This is important to prevent installation to unintended
# locations.
#
# - sdh_configure [...]
#
# Runs `./configure --prefix="$SAGE_LOCAL" --libdir="$SAGE_LOCAL/lib"`
# --disable-static, (for autoconf'd projects with extra
# --disable-maintainer-mode --disable-dependency-tracking) Additional
# arguments to `./configure` may be given as arguments.
#
# - sdh_make [...]
#
# Runs `$MAKE` with the default target. Additional arguments to `make` may
# be given as arguments.
#
# - sdh_make_install [...]
#
# Runs `$MAKE install` with DESTDIR correctly set to a temporary install
# directory, for staged installations. Additional arguments to `make` may
# be given as arguments. If $SAGE_DESTDIR is not set then the command is
# run with $SAGE_SUDO, if set.
#
# - sdh_pip_install [--no-deps] [--build-isolation] [--no-build-isolation] [...]
#
# Builds a wheel using `pip wheel` with the given options [...], then installs
# the wheel. Unless the special option --no-build-isolation is given,
# the wheel is built using build isolation.
# If the special option --no-deps is given, it is passed to pip install.
# If $SAGE_DESTDIR is not set then the command is run with $SAGE_SUDO, if
# set.
#
# - sdh_pip_uninstall [...]
#
# Runs `pip uninstall` with the given arguments. If unsuccessful, it displays a warning.
#
# - eval sdh_prefix_args PREFIX [...]
#
# Helper function for transforming build options so that they can be passed
# through "pip".
#
# - sdh_cmake [...]
#
# Runs `cmake` in the current directory with the given arguments, as well as
# additional arguments passed to cmake (assuming packages are using the
# GNUInstallDirs module) so that `CMAKE_INSTALL_PREFIX` and
# `CMAKE_INSTALL_LIBDIR` are set correctly.
#
# - sdh_install [-T] SRC [SRC...] DEST
#
# Copies one or more files or directories given as SRC (recursively in the
# case of directories) into the destination directory DEST, while ensuring
# that DEST and all its parent directories exist. DEST should be a path
# under $SAGE_LOCAL, generally. For DESTDIR installs the $SAGE_DESTDIR path
# is automatically prepended to the destination.
#
# The -T option treats DEST as a normal file instead (e.g. for copying a
# file to a different filename). All directory components are still created
# in this case.
#
# - sdh_preload_lib EXECUTABLE SONAME
#
# (Linux only--no-op on other platforms.) Check shared libraries loaded by
# EXECUTABLE (may be a program or another library) for a library starting
# with SONAME, and if found appends SONAME to the LD_PRELOAD environment
# variable. See https://github.com/sagemath/sage/issues/24885.
set -o allexport
# Utility function to get the terminal width in columns
# Returns 80 by default if nothing else works
_sdh_cols() {
local cols="${COLUMNS:-$(tput cols 2>/dev/null)}"
if [ "$?" -ne 0 -o -z "$cols" ]; then
# If we can't get the terminal width any other way just default to 80
cols=80
fi
echo $cols
}
# Utility function to print a terminal-width horizontal rule using the given
# character (or '-' by default)
_sdh_hr() {
local char="${1:--}"
printf '%*s\n' $(_sdh_cols) '' | tr ' ' "${char}"
}
sdh_die() {
local ret=$?
local msg
if [ $ret -eq 0 ]; then
# Always return non-zero, but if the last command run returned non-zero
# then return its exact error code
ret=1
fi
if [ $# -gt 0 ]; then
msg="$*"
else
msg="$(cat -)"
fi
_sdh_hr >&2 '*'
echo "$msg" | fmt -s -w $(_sdh_cols) >&2
_sdh_hr >&2 '*'
exit $ret
}
sdh_check_vars() {
while [ -n "$1" ]; do
[ -n "$(eval "echo "\${${1}+isset}"")" ] || sdh_die << _EOF_
${1} undefined ... exiting
Maybe run 'sage --buildsh'?
_EOF_
shift
done
}
sdh_guard() {
sdh_check_vars SAGE_ROOT SAGE_LOCAL SAGE_INST_LOCAL SAGE_SHARE
}
sdh_configure() {
echo "Configuring $PKG_NAME"
# Run all configure scripts with bash to work around bugs with
# non-portable scripts.
# See https://github.com/sagemath/sage/issues/24491
if [ -z "$CONFIG_SHELL" ]; then
export CONFIG_SHELL=`command -v bash`
fi
if [ "$UNAME" = "CYGWIN" ]; then
# TODO: To use --disable-static for all packages on Cygwin, need
# #30814: Cygwin: Fix remaining packages to build shared libraries, using AM_LDFLAGS=-no-undefined
DISABLE_STATIC=
else
DISABLE_STATIC=--disable-static
fi
./configure --prefix="$SAGE_INST_LOCAL" --libdir="$SAGE_INST_LOCAL/lib" $DISABLE_STATIC --disable-maintainer-mode --disable-dependency-tracking "$@"
if [ $? -ne 0 ]; then # perhaps it is a non-autoconf'd project
./configure --prefix="$SAGE_INST_LOCAL" --libdir="$SAGE_INST_LOCAL/lib" $DISABLE_STATIC "$@"
if [ $? -ne 0 ]; then
if [ -f "$(pwd)/config.log" ]; then
sdh_die <<_EOF_
Error configuring $PKG_NAME
See the file
$(pwd)/config.log
for details.
_EOF_
fi
sdh_die "Error configuring $PKG_NAME"
fi
fi
}
sdh_make() {
echo "Building $PKG_NAME"
${MAKE:-make} "$@" || sdh_die "Error building $PKG_NAME"
}
sdh_make_check() {
echo "Checking $PKG_NAME"
${MAKE:-make} check "$@" || sdh_die "Failures checking $PKG_NAME"
}
sdh_make_install() {
echo "Installing $PKG_NAME"
if [ -n "$SAGE_DESTDIR" ]; then
local sudo=""
else
local sudo="$SAGE_SUDO"
fi
$sudo ${MAKE:-make} install DESTDIR="$SAGE_DESTDIR" "$@" || \
sdh_die "Error installing $PKG_NAME"
}
sdh_setup_bdist_wheel() {
# Trac #32046: Most uses of this function can be replaced by sdh_pip_install
mkdir -p dist
rm -f dist/*.whl
BDIST_DIR="$(mktemp -d)"
python3 setup.py --no-user-cfg \
bdist_wheel --bdist-dir "$BDIST_DIR" \
"$@" || sdh_die "Error building a wheel for $PKG_NAME"
}
sdh_prefix_args () {
prefix="$1"
shift
while [ $# -gt 0 ]; do
# Quoted quotes because the result is to be run through eval
echo "$prefix" \"$1\"
shift
done
}
sdh_pip_install() {
echo "Installing $PKG_NAME"
mkdir -p dist
rm -f dist/*.whl
install_options=""
# pip has --no-build-isolation but no flag that turns the default back on...
build_isolation_option="--find-links=$SAGE_SPKG_WHEELS"
while [ $# -gt 0 ]; do
case "$1" in
--build-isolation)
# Our default after #33789 (Sage 9.7): We allow the package to provision
# its build environment using the stored wheels.
# We pass --find-links.
# The SPKG needs to declare "setuptools_wheel" as a dependency.
build_isolation_option="--find-links=$SAGE_SPKG_WHEELS"
;;
--no-build-isolation)
# Use --no-binary, so that no wheels from caches are used.
build_isolation_option="--no-build-isolation --no-binary :all:"
;;
--no-deps)
install_options="$install_options $1"
;;
*)
break
;;
esac
shift
done
if python3 -m pip wheel --wheel-dir=dist --verbose --no-deps --no-index --isolated --ignore-requires-python $build_isolation_option "$@"; then
: # successful
else
case $build_isolation_option in
*--no-build-isolation*)
sdh_die "Error building a wheel for $PKG_NAME"
;;
*)
echo >&2 "Warning: building with \"python3 -m pip wheel --wheel-dir=dist --verbose --no-deps --no-index --isolated --ignore-requires-python $build_isolation_option\" failed."
build_isolation_option="--no-build-isolation --no-binary :all:"
echo >&2 "Retrying with \"python3 -m pip wheel --wheel-dir=dist --verbose --no-deps --no-index --isolated --ignore-requires-python $build_isolation_option\"."
if python3 -m pip wheel --wheel-dir=dist --verbose --no-deps --no-index --isolated --ignore-requires-python $build_isolation_option "$@"; then
echo >&2 "Warning: Wheel building needed to use \"$build_isolation_option\" to succeed. This means that a dependencies file in build/pkgs/ needs to be updated. Please report this to [email protected], including the build log of this package."
else
sdh_die "Error building a wheel for $PKG_NAME"
fi
;;
esac
fi
sdh_store_and_pip_install_wheel $install_options .
}
sdh_pip_editable_install() {
echo "Installing $PKG_NAME (editable mode)"
# Until https://github.com/sagemath/sage/issues/34209 switches us to PEP 660 editable wheels
export SETUPTOOLS_ENABLE_FEATURES=legacy-editable
python3 -m pip install --verbose --no-deps --no-index --no-build-isolation --isolated --editable "$@" || \
sdh_die "Error installing $PKG_NAME"
}
sdh_store_wheel() {
if [ -n "$SAGE_DESTDIR" ]; then
local sudo=""
else
local sudo="$SAGE_SUDO"
fi
if [ "$*" != "." ]; then
sdh_die "Error: sdh_store_wheel requires . as only argument"
fi
wheel=""
for w in dist/*.whl; do
if [ -n "$wheel" ]; then
sdh_die "Error: more than one wheel found after building"
fi
if [ -f "$w" ]; then
wheel="$w"
fi
done
if [ -z "$wheel" ]; then
sdh_die "Error: no wheel found after building"
fi
mkdir -p "${SAGE_DESTDIR}${SAGE_SPKG_WHEELS}" && \
$sudo mv "$wheel" "${SAGE_DESTDIR}${SAGE_SPKG_WHEELS}/" || \
sdh_die "Error storing $wheel"
wheel="${SAGE_SPKG_WHEELS}/${wheel##*/}"
if [ -n "${SAGE_SPKG_SCRIPTS}" -a -n "${PKG_BASE}" ]; then
wheel_basename="${wheel##*/}"
distname="${wheel_basename%%-*}"
# Record name and wheel file location
mkdir -p ${SAGE_DESTDIR}${SAGE_SPKG_SCRIPTS}/${PKG_BASE}
echo "${distname} @ file://${wheel}" >> ${SAGE_DESTDIR}${SAGE_SPKG_SCRIPTS}/${PKG_BASE}/spkg-requirements.txt
fi
wheel="${SAGE_DESTDIR}${wheel}"
}
sdh_store_and_pip_install_wheel() {
# The $SAGE_PIP_INSTALL_FLAGS variable is set by sage-build-env-config.
# We skip sanity checking its contents since you should either let sage
# decide what it contains, or really know what you are doing.
local pip_options="${SAGE_PIP_INSTALL_FLAGS}"
while [ $# -gt 0 ]; do
case $1 in
-*) pip_options="$pip_options $1"
;;
*)
break
;;
esac
shift
done
sdh_store_wheel "$@"
if [ -n "$SAGE_SUDO" ]; then
# Trac #29585: Do the SAGE_DESTDIR staging of the wheel installation
# ONLY if SAGE_SUDO is set (in that case, we still do the staging so
# that we do not invoke pip as root).
if [ -n "$SAGE_DESTDIR" ]; then
# --no-warn-script-location: Suppress a warning caused by --root
local sudo=""
local root="--root=$SAGE_DESTDIR --no-warn-script-location"
else
# Trac #32361: Of course, this can only be done for normal packages,
# whose installation goes through sage-spkg.
# For script packages, we do have to invoke pip as root.
local sudo="$SAGE_SUDO"
local root=""
fi
else
local sudo=""
local root=""
fi
# Trac #32659: pip no longer reinstalls local wheels if the version is the same.
# Because neither (1) applying patches nor (2) local changes (in the case
# of sage-conf, sage-setup, etc.) bump the version number, we need to
# override this behavior. The pip install option --force-reinstall does too
# much -- it also reinstalls all dependencies, which we do not want.
wheel_basename="${wheel##*/}"
distname="${wheel_basename%%-*}"
$sudo sage-pip-uninstall "$distname"
if [ $? -ne 0 ]; then
echo "(ignoring error)" >&2
fi
$sudo sage-pip-install $root $pip_options "$wheel" || \
sdh_die "Error installing ${wheel##*/}"
}
sdh_pip_uninstall() {
sage-pip-uninstall "$@"
if [ $? -ne 0 ]; then
echo "Warning: pip exited with status $?" >&2
fi
}
sdh_cmake() {
echo "Configuring $PKG_NAME with cmake"
cmake . -DCMAKE_INSTALL_PREFIX="${SAGE_INST_LOCAL}" \
-DCMAKE_INSTALL_LIBDIR=lib \
"$@"
if [ $? -ne 0 ]; then
if [ -f "$(pwd)/CMakeFiles/CMakeOutput.log" ]; then
sdh_die <<_EOF_
Error configuring $PKG_NAME with cmake
See the file
$(pwd)/CMakeFiles/CMakeOutput.log
for details.
_EOF_
fi
sdh_die "Error configuring $PKG_NAME with cmake"
fi
}
sdh_install() {
local T=0
local src=()
if [ "$1" = "-T" ]; then
T=1
shift
fi
while [ $# -gt 1 ]; do
if [ ! \( -e "$1" -o -L "$1" \) ]; then
sdh_die "Error: source file/directory $1 does not exist"
fi
src+=("$1")
shift
done
local dest="$1"
if [ -z "$src" ]; then
sdh_die "Error: no source file(s) for sdh_install given"
fi
if [ -z "$dest" ]; then
sdh_die "Error: destination for sdh_install not given"
fi
# Prefix SAGE_DESTDIR to the destination for DESTDIR installs
dest="${SAGE_DESTDIR}$dest"
if [ $T -eq 0 -a -e "$dest" -a ! -d "$dest" ]; then
sdh_die "Error: destination $dest for sdh_install exists and is not a directory"
fi
local destdir="$dest"
if [ $T -eq 1 ]; then
destdir="$(dirname $dest)"
fi
if [ ! -d "$destdir" ]; then
mkdir -p "$destdir" || exit $?
fi
for s in "${src[@]}"; do
echo "$s -> $dest"
cp -R -p "$s" "$dest" || exit $?
done
}
sdh_preload_lib() {
local executable="$1"
local soname="$2"
if [ "$UNAME" != "Linux" ]; then
return 0
fi
local ldlibs="$(ldd $(which $executable))"
if [ $? -ne 0 ]; then
sdh_die "Could not get shared library dependencies for $executable"
fi
local lib=$(echo "$ldlibs" | sed -n 's/\s*'$soname'.* => \(.\+\) .*/\1/p')
if [ -n "$lib" ]; then
if [ -n "$LD_PRELOAD" ]; then
export LD_PRELOAD="$LD_PRELOAD:$lib"
else
export LD_PRELOAD="$lib"
fi
fi
}
set +o allexport