Marcel Telka
2022-11-25 e2076b24257c237e5de8a342155d9e141b5ba7bc
commit | author | age
053684 1 #! /usr/bin/ksh
MT 2 #
3 #
4 # This file and its contents are supplied under the terms of the
5 # Common Development and Distribution License ("CDDL"), version 1.0.
6 # You may only use this file in accordance with the terms of version
7 # 1.0 of the CDDL.
8 #
9 # A full copy of the text of the CDDL should have accompanied this
10 # source.  A copy of the CDDL is also available via the Internet at
11 # http://www.illumos.org/license/CDDL.
12 #
13
14 #
15 # Copyright 2022 Marcel Telka
16 #
17
18
19 THIS="python-integrate-project"
20 CONF="$THIS.conf"
21 SNIPPET="$THIS.snippet"
f06a31 22 APIURL="https://pypi.org/pypi"
053684 23 CURL="/usr/bin/curl -s"
MT 24
25
26 function usage
27 {
28     [[ -n "$1" ]] && printf "ERROR: %s\n\n" "$1" >&2
29     printf "Usage: %s [-l VERSION] [-o OBSOLETE].. [-u] PROJECT\n" "$THIS" >&2
30     [[ -n "$1" ]] && exit 1
31     exit 0
32 }
33
34
35 VERSION=
36 OBSOLETE=
37 UPGRADE_ONLY=0
38 while getopts ":hl:o:u" OPT ; do
39     case "$OPT" in
40     "?"|"h")    usage ;;
41     "l")        VERSION="$OPTARG" ;;
42     "o")        OBSOLETE="$OBSOLETE $OPTARG" ;;
43     "u")        UPGRADE_ONLY=1 ;;
44     esac
45 done
46 shift $((OPTIND - 1))
47
48 (($# == 0)) && usage
49 (($# > 1)) && usage "Too many arguments"
50
51 PROJECT="$1"
52
53
9db594 54 # Prevent user's environment to affect the integration.
MT 55 # Allow one exception only: USERLAND_ARCHIVES
56 GMAKE="env -"
57 [[ -n "$USERLAND_ARCHIVES" ]] && GMAKE="$GMAKE USERLAND_ARCHIVES=$USERLAND_ARCHIVES"
58 GMAKE="$GMAKE gmake"
59
60
053684 61 WS_TOP=$(git rev-parse --show-toplevel 2>/dev/null)
MT 62 [[ -z "$WS_TOP" ]] && usage "The script must be run in git repo"
63
64 DIR="$WS_TOP/components/python"
65 [[ -d "$DIR" ]] || usage "Directory $DIR not found"
66
67
68 # Get data from pypi
69 PYPI_PROJECT=$($CURL "$APIURL/$PROJECT/json")
70 if (($? != 0)) || [[ -z "$PYPI_PROJECT" ]] ; then
71     printf "FATAL: Failed to get data from pypi\n" >&2
72     exit 1
73 fi
74
f06a31 75 # Distribution match project
MT 76 DISTRIBUTION="$PROJECT"
77
a18785 78 function get_PKGINFO_entry
MT 79 {
80     typeset ENTRY="$1"
81
82     [[ -f "$SOURCE_DIR/PKG-INFO" ]] || return
83
84     cat "$SOURCE_DIR/PKG-INFO" \
85         | sed -e '/^$/,$d' \
86         | awk 'END{printf("\n")}/^[^:]+: /{$0="\n"$0}1' ORS=' ' \
87         | grep "^$ENTRY: " \
88         | sed -e "s/^$ENTRY: //" -e 's/ *$//'
89 }
90
053684 91 # Get project homepage
MT 92 HOMEPAGE=$(printf "%s" "$PYPI_PROJECT" | /usr/bin/jq -r '.info.home_page')
93 if (($? != 0)) || [[ -z "$HOMEPAGE" || "$HOMEPAGE" == "null" ]] ; then
a18785 94     HOMEPAGE=$(printf "%s" "$PYPI_PROJECT" | /usr/bin/jq -r '.info.project_urls.Homepage')
MT 95     if (($? != 0)) || [[ -z "$HOMEPAGE" || "$HOMEPAGE" == "null" ]] ; then
96         printf "WARNING: Failed to get homepage for project %s from pypi\n" "$PROJECT" >&2
97         HOMEPAGE=$(get_PKGINFO_entry "Home-page")
98     fi
053684 99 fi
MT 100
101 # Find the latest version if not provided by user
102 if [[ -z "$VERSION" ]] ; then
103     VERSION=$(printf "%s" "$PYPI_PROJECT" | /usr/bin/jq -r '.info.version')
104     if (($? != 0)) || [[ -z "$VERSION" || "$VERSION" == "null" ]] ; then
105         printf "FATAL: Failed to get version for project %s from pypi\n" "$PROJECT" >&2
106         exit 1
107     fi
108 fi
109
110 # Get release data from pypi
111 PYPI_PROJECT_RELEASE=$($CURL "$APIURL/$PROJECT/$VERSION/json")
112 if (($? != 0)) || [[ -z "$PYPI_PROJECT_RELEASE" ]] ; then
113     printf "FATAL: Failed to get data for version %s from pypi\n" "$VERSION" >&2
114     exit 1
115 fi
116
53ac61 117 # Get download url
MT 118 DOWNLOAD_URL=$(printf "%s" "$PYPI_PROJECT_RELEASE" | /usr/bin/jq -r '.urls[]|select(.packagetype=="sdist")|.url')
119 if (($? != 0)) || [[ -z "$DOWNLOAD_URL" || "$DOWNLOAD_URL" == "null" ]] ; then
120     printf "WARNING: Failed to get download url for project %s, version %s from pypi\n" "$PROJECT" "$VERSION" >&2
121     DOWNLOAD_URL="TODO"
122 fi
123
053684 124
MT 125 # Prepare the directory
f06a31 126 DIR="$DIR/$DISTRIBUTION"
053684 127 mkdir -p "$DIR"
MT 128 cd "$DIR"
129 git restore --staged . > /dev/null 2>&1
130 git checkout . > /dev/null 2>&1
131
132 # Is this new project, or just a rebuild?
133 NEW=1
134 REBUILD=0
135 PREV_VER=
136 PREV_HVER=
137 PREV_REV=0
138 if git ls-files --error-unmatch Makefile > /dev/null 2>&1 ; then
139     NEW=0
140     REBUILD=1
9db594 141     PREV_VER=$($GMAKE print-value-COMPONENT_VERSION 2>/dev/null)
4e8881 142     (($? != 0)) && printf "FATAL: 'gmake print-value-COMPONENT_VERSION' failed!\n" >&2 && exit 1
9db594 143     PREV_REV=$($GMAKE print-value-COMPONENT_REVISION 2>/dev/null)
053684 144
MT 145     # If we were asked to do version upgrade, but we do not have new
146     # version, then we are done.
9db594 147     PREV_HVER=$($GMAKE print-value-HUMAN_VERSION 2>/dev/null)
053684 148     ((UPGRADE_ONLY)) && [[ "$PREV_HVER" == "$VERSION" ]] && exit 0
MT 149
9db594 150     $GMAKE clobber > /dev/null 2>&1
053684 151 fi
MT 152
153 # Remove everything from git (except known patches, history, and $CONF)
154 touch "$CONF"
155 grep "^%patch%" "$CONF" | while read TAG PATCH ; do rm -f "patches/$PATCH" ; done
156 rm -f history "$CONF"
157 find . -type f | while read f ; do git rm "$f" > /dev/null 2>&1 ; done
158 rm -rf "$DIR" 2>/dev/null
159 git checkout history > /dev/null 2>&1
160 git checkout "$CONF" > /dev/null 2>&1
161 touch "$CONF"
162 grep "^%patch%" "$CONF" | while read TAG PATCH ; do
163     git checkout "patches/$PATCH" > /dev/null 2>&1
164     [[ -f "patches/$PATCH" ]] || printf "WARNING: Patch %s not found\n" "$PATCH" >&2
165 done
166
167
168 # Makefile template
169 (
170 cat $WS_TOP/transforms/copyright-template | sed -e '/^$/,$d'
171 cat <<EOF
172
173 #
174 # This file was automatically generated using the following command:
175 #   \$WS_TOOLS/$THIS $PROJECT
176 #
177
b3e12e 178 BUILD_STYLE = pyproject
345a62 179 USE_COMMON_TEST_MASTER = no
053684 180 EOF
MT 181 gsed -e '0,/^%include-1%/d' -e '/^%/,$d' < "$CONF"
182 cat <<EOF
183
184 include ../../../make-rules/shared-macros.mk
185
f06a31 186 COMPONENT_NAME =        $DISTRIBUTION
053684 187 COMPONENT_VERSION =        $VERSION
MT 188 COMPONENT_REVISION =        $((PREV_REV + 1))
189 COMPONENT_SUMMARY =        $PROJECT - TODO
74d2fd 190 EOF
MT 191 [[ -n "$HOMEPAGE" ]] && printf "COMPONENT_PROJECT_URL =\t\t%s\n" "$HOMEPAGE"
192 cat <<EOF
53ac61 193 COMPONENT_ARCHIVE_URL =        \\
MT 194     $DOWNLOAD_URL
053684 195 COMPONENT_ARCHIVE_HASH =    \\
MT 196     sha256:TODO
197 COMPONENT_LICENSE =        license:TODO
198 COMPONENT_LICENSE_FILE =    licfile:TODO
5e7447 199
MT 200 _TEST_STYLE = TODO
053684 201 EOF
MT 202 cat "$CONF" | gsed -e '0,/^%include-2%/d' -e '/^%/,$d' | gsed -e '1s/^./\n&/'
203 printf "\ninclude \$(WS_MAKE_RULES)/common.mk\n"
204 cat "$CONF" | gsed -e '0,/^%include-3%/d' -e '/^%/,$d' | gsed -e '1s/^./\n&/'
205 printf "\n"
206 ) > Makefile
207
208 # Remove COMPONENT_REVISION if not needed
9db594 209 COMPONENT_VERSION=$($GMAKE print-value-COMPONENT_VERSION)
053684 210 [[ "$PREV_VER" != "$COMPONENT_VERSION" ]] && REBUILD=0 && sed -i -e '/^COMPONENT_REVISION/d' Makefile
5d2e17 211 git add Makefile
053684 212
MT 213 # Calculate sham256 sum for source package
9db594 214 $GMAKE fetch > /dev/null 2>&1
MT 215 USERLAND_ARCHIVES=$($GMAKE print-value-USERLAND_ARCHIVES)
216 COMPONENT_ARCHIVE=$($GMAKE print-value-COMPONENT_ARCHIVE)
53b4dd 217 [[ ! -f "$USERLAND_ARCHIVES$COMPONENT_ARCHIVE" ]] && printf "FATAL: 'gmake fetch' failed!\n" >&2 && exit 1
MT 218 SHA256=$(digest -a sha256 "$USERLAND_ARCHIVES$COMPONENT_ARCHIVE")
053684 219 sed -i -e 's/sha256:TODO/sha256:'"$SHA256"'/g' Makefile
MT 220 git add Makefile
221
222 # Unpack sources
9db594 223 ! $GMAKE prep > /dev/null 2>&1 && printf "FATAL: 'gmake prep' failed!\n" >&2 && exit 1
MT 224 SOURCE_DIR=$($GMAKE print-value-SOURCE_DIR)
b3e12e 225
MT 226 if [[ ! -f "$SOURCE_DIR/pyproject.toml" ]] ; then
227     [[ ! -f "$SOURCE_DIR/setup.py" ]] && printf "FATAL: Neither pyproject.toml nor setup.py found!\n" >&2 && exit 1
228     sed -i -e 's/^\(BUILD_STYLE = \).*$/\1setup.py/' Makefile
229 fi
053684 230
MT 231 # Get summary
232 SUMMARY=$(printf "%s" "$PYPI_PROJECT" | /usr/bin/jq -r '.info.summary')
233 if (($? != 0)) || [[ -z "$SUMMARY" || "$SUMMARY" == "null" ]] ; then
234     printf "WARNING: Failed to get summary for project %s from pypi\n" "$PROJECT" >&2
a18785 235     SUMMARY=$(get_PKGINFO_entry "Summary")
MT 236     [[ -z "$SUMMARY" ]] && SUMMARY="TODO"
053684 237 fi
MT 238 # Summary needs to be sanitized
239 SUMMARY="${SUMMARY//\`/\\\\\`}"
240 SUMMARY="${SUMMARY//\"/\\\\\\\\\\\\\\\\\\\\\\\\\\\\\"}"
241 SUMMARY="${SUMMARY//\//\/}"
242 SUMMARY="${SUMMARY//\$/\\\\\$\$}"
243 sed -i -e 's/\(COMPONENT_SUMMARY.*\)TODO$/\1'"$SUMMARY"'/g' Makefile
244
245
246 # Try to detect license type(s)
247 function detect_license
248 {
249     typeset -n L="$1"
250     typeset F="$2"
251     typeset D
252
253     D=$("$WS_TOP/tools/license-detector" "$F")
254     [[ -n "$L" ]] && L="$L OR " ; L="$L$D"
43f587 255 }
MT 256
053684 257 LICENSE=
MT 258 LICFILE=
7f15d5 259 for f in $(get_PKGINFO_entry "License-File") LICENSE LICENSE.rst LICENSE.txt ; do
053684 260     [[ -f "$SOURCE_DIR/$f" ]] || continue
MT 261     LICFILE="$f"
262
263     detect_license LICENSE "$SOURCE_DIR/$LICFILE"
264     [[ -n "$LICENSE" ]] && break
265
266     printf "WARNING: Failed to detect license type in %s file\n" "$f" >&2
267 done
268 if [[ -z "$LICFILE" ]] ; then
269     printf "WARNING: No license file found\n" >&2
270 else
271     sed -i -e 's|licfile:TODO|'"$LICFILE"'|g' Makefile
272 fi
273
274 if [[ -z "$LICENSE" ]] ; then
275     # Execute hook-no-license snippet
276     gsed -e '0,/^%hook-no-license%/d' -e '/^%/,$d' < "$CONF" > "$SNIPPET"
277     . "./$SNIPPET"
278     rm -f "$SNIPPET"
279
f06a31 280     if [[ -f "$DISTRIBUTION.license" ]] ; then
053684 281         sed -i -e '/^COMPONENT_LICENSE_FILE/d' Makefile
f06a31 282         git add "$DISTRIBUTION.license"
MT 283         [[ -z "$LICENSE" ]] && detect_license LICENSE "$DISTRIBUTION.license"
053684 284     fi
MT 285     [[ -z "$LICENSE" ]] && LICENSE="TODO"
286 fi
287
288 # Store the detected license into the Makefile
289 sed -i -e 's/license:TODO/'"$LICENSE"'/g' Makefile
290
291
5e7447 292 # detect TEST_STYLE
MT 293 TEST_STYLE=
294 cd "$SOURCE_DIR"
295 while true ; do
296     tox -l && TEST_STYLE="tox" && break
297
298     pytest --setup-plan
299     (($? != 5)) && TEST_STYLE="pytest" && break
300
b3e12e 301     [[ -f setup.py ]] && python setup.py test --help && TEST_STYLE="setup.py" && break
5e7447 302
MT 303     TEST_STYLE="none"
304     break
305 done > /dev/null 2>&1
306 cd "$DIR"
307
9db594 308 if [[ "$TEST_STYLE" == "$($GMAKE print-value-TEST_STYLE)" ]] ; then
5e7447 309     # If the detected TEST_STYLE is same as the default value or the value
MT 310     # forced by the component, then we do not need to set the detected
311     # value.
75946c 312     sed -i -e '/^_TEST_STYLE = TODO$/,+1d' Makefile
5e7447 313 else
MT 314     # Set the detected TEST_STYLE value
315     sed -i -e 's/^_\(TEST_STYLE = \)TODO$/\1'"$TEST_STYLE/" Makefile
316
317     # If the component forces different test style than detected, then drop
318     # the detected value
9db594 319     if [[ "$TEST_STYLE" != "$($GMAKE print-value-TEST_STYLE)" ]] ; then
75946c 320         sed -i -e '/^TEST_STYLE = '"${TEST_STYLE//./\\.}/,+1d" Makefile
5e7447 321     fi
MT 322 fi
323
0477cc 324 # Warn if pytest is called directly by tox
MT 325 [[ "$($GMAKE print-value-TEST_STYLE)" == "tox" && -f "$SOURCE_DIR/tox.ini" ]] \
e2076b 326     && grep -q '^\(commands *=\)\? *pytest' "$SOURCE_DIR/tox.ini" \
0477cc 327     && printf "WARNING: pytest is called directly in tox.ini\n" >&2
MT 328
5e7447 329
053684 330 # Create manifests
9db594 331 if ! $GMAKE sample-manifest > /dev/null 2>&1 ; then
053684 332     printf "ERROR: 'gmake sample-manifest' failed!\n" >&2
MT 333 else
334     cat manifests/sample-manifest.p5m \
f06a31 335         | sed -e 's/^#.*Copyright.*<contributor>.*$/# This file was automatically generated using '"$THIS"'/g' \
MT 336         > "$DISTRIBUTION-PYVER.p5m"
053684 337
MT 338     # Execute hook-manifest snippet
339     gsed -e '0,/^%hook-manifest%/d' -e '/^%/,$d' < "$CONF" > "$SNIPPET"
340     . "./$SNIPPET"
341     rm -f "$SNIPPET"
342
f06a31 343     git add manifests/sample-manifest.p5m "$DISTRIBUTION-PYVER.p5m"
053684 344 fi
MT 345
346
347 # $CONF is no longer needed
348 rm -f "$CONF"
349 git checkout "$CONF" > /dev/null 2>&1
350
351
352 # Generate REQUIRED_PACKAGES
9db594 353 $GMAKE REQUIRED_PACKAGES > /dev/null 2>&1 || printf "ERROR: 'gmake REQUIRED_PACKAGES' failed!\n" >&2
053684 354 git add Makefile
MT 355
356
357 # Check for Makefile completeness
358 grep -q "TODO" Makefile && printf "ERROR: Makefile is not complete (TODO found)\n" >&2
359
360
361 # Make sure the build environment is setup properly and we do have all
362 # requirements installed.  Otherwise we cannot continue.
9db594 363 ! $GMAKE env-check > /dev/null 2>&1 && printf "FATAL: 'gmake env-check' failed!\n" >&2 && exit 1
053684 364
MT 365
345a62 366 # Publish packages and create pkg5 file
9db594 367 $GMAKE publish > /dev/null 2>&1 || printf "ERROR: 'gmake publish' failed!\n" >&2
345a62 368 git add pkg5 2>/dev/null
ae9d97 369
MT 370
9db594 371 PYTHON_VERSIONS=$($GMAKE print-value-PYTHON_VERSIONS)
feebc5 372 PYTHON_TEST_BOOTSTRAP=$($GMAKE print-value-PYTHON_TEST_BOOTSTRAP)
053684 373
MT 374
345a62 375 # Run tests to make sure they pass and to create result snapshots
c335f2 376 TESTED_VERSIONS=
f4886c 377 for v in $PYTHON_VERSIONS ; do
90c79e 378     # Check the test environment
feebc5 379     if ! $GMAKE PYTHON_VERSIONS=$v test-env-check > /dev/null 2>&1 ; then
MT 380         if [[ "$PYTHON_TEST_BOOTSTRAP" == "yes" ]] ; then
381             printf "WARNING: Test environment for %s is not ready yet (bootstrap)\n" "$v" >&2
382         else
383             printf "ERROR: 'gmake test-env-check' failed for %s!\n" "$v" >&2
384         fi
385         continue
386     fi
90c79e 387
MT 388     # Run the test
9db594 389     ! $GMAKE PYTHON_VERSIONS=$v test > /dev/null 2>&1 && printf "ERROR: Testing failed for %s!\n" "$v" >&2 && continue
053684 390
f4886c 391     # If there is no snapshot produced the component likely does not support tests
9db594 392     COMPONENT_TEST_SNAPSHOT=$($GMAKE PYTHON_VERSION=$v print-value-COMPONENT_TEST_SNAPSHOT)
345a62 393     [[ ! -f "$COMPONENT_TEST_SNAPSHOT" ]] && printf "WARNING: Testing unsupported for %s\n" "$v" >&2 && continue
MT 394
395     # Empty result snapshot is suspicious
396     [[ -s "$COMPONENT_TEST_SNAPSHOT" ]] || printf "WARNING: Empty test results for %s\n" "$v" >&2
c335f2 397
MT 398     TESTED_VERSIONS="$TESTED_VERSIONS $v"
f4886c 399 done
053684 400
345a62 401 # Save result snapshots and detect USE_COMMON_TEST_MASTER value
MT 402 TEST_MASTERS=
403 for common_results in yes no ; do
c335f2 404     for v in $TESTED_VERSIONS ; do
9db594 405         COMPONENT_TEST_SNAPSHOT=$($GMAKE PYTHON_VERSION=$v print-value-COMPONENT_TEST_SNAPSHOT)
MT 406         COMPONENT_TEST_MASTER=$($GMAKE PYTHON_VERSION=$v USE_COMMON_TEST_MASTER=$common_results print-value-COMPONENT_TEST_MASTER)
053684 407
345a62 408         if [[ -f "$COMPONENT_TEST_MASTER" ]] ; then
MT 409             # Switch to 'USE_COMMON_TEST_MASTER = no' if test results differ
410             if ! diff "$COMPONENT_TEST_SNAPSHOT" "$COMPONENT_TEST_MASTER" > /dev/null ; then
f37c36 411                 printf "WARNING: Test results differ so switch to 'USE_COMMON_TEST_MASTER = no'\n" >&2
345a62 412                 rm -f $TEST_MASTERS
MT 413                 TEST_MASTERS=
414                 continue 2
415             fi
416         else
417             mkdir -p $(dirname "$COMPONENT_TEST_MASTER")
418             cp -p "$COMPONENT_TEST_SNAPSHOT" "$COMPONENT_TEST_MASTER"
419             TEST_MASTERS="$TEST_MASTERS $COMPONENT_TEST_MASTER"
420         fi
421     done
422     break
423 done
424 [[ -n "$TEST_MASTERS" ]] && git add $TEST_MASTERS
425
426 # Run tests again to confirm the results are reproducible
c335f2 427 for v in $TESTED_VERSIONS ; do
9db594 428     $GMAKE PYTHON_VERSIONS=$v USE_COMMON_TEST_MASTER=$common_results test > /dev/null 2>&1 || printf "ERROR: Testing for %s is not reproducible!\n" "$v" >&2
345a62 429 done
MT 430
431 # Remove USE_COMMON_TEST_MASTER from Makefile if it should be set to (default) 'yes'
432 if [[ "$common_results" == "yes" ]] ; then
433     sed -i -e '/^USE_COMMON_TEST_MASTER/d' Makefile
434     git add Makefile
435 fi
053684 436
MT 437
438 # Handle history
9db594 439 COMPONENT_FMRI=$($GMAKE print-value-COMPONENT_FMRI)
MT 440 PYTHON_VERSIONS_OBSOLETING=$($GMAKE print-value-PYTHON_VERSIONS_OBSOLETING)
053684 441 OV=
MT 442 OV_PLURAL=
443 for o in $(echo $OBSOLETE $PYTHON_VERSIONS_OBSOLETING | LC_ALL=C sort -u) ; do
444     PYV=${o//.}
c13d3c 445     FMRI=$(pkg list -nvH "$COMPONENT_FMRI-$PYV" 2>/dev/null | egrep -v '(o|r)$' | sed -e 's|^.*\('"$COMPONENT_FMRI"'\)|\1|g' -e 's/:[^:]*$//g' -e 's/\(-[^-]*\)$/,5.11\1/g')
053684 446     [[ -n "$FMRI" ]] || continue
MT 447     FMRI_H=${FMRI%.*}
448     FMRI_T=${FMRI##*.}
449     if [[ "$FMRI_H" == "$FMRI" ]] ; then
450         printf "WARNING: Wrong fmri format: %s\n" "$FMRI" >&2
451         continue
452     fi
453     FMRI_T=$((FMRI_T + 1))
454     printf "%s.%s noincorporate\n" "$FMRI_H" "$FMRI_T" >> history
455
456     [[ -n "$OV" ]] && OV="$OV and " && OV_PLURAL="s"
457     OV="$OV$o"
458 done
459 if [[ -f history ]] ; then
460     LC_ALL=C sort -u history > history.new
461     mv history.new history
462     git add history
463 fi
464
465
466 # Construct the commit message
467 MSG=
468 if ((NEW)) ; then
469     MSG="Add $PROJECT python project"
470 else
471     if ((REBUILD == 0)) ; then
472         [[ "$PREV_HVER" != "$VERSION" ]] && MSG="update to $VERSION" || MSG="change version format"
473     fi
474
475     NV=
476     for v in $PYTHON_VERSIONS ; do
477         PYV=${v//.}
478         pkg list -avH "$COMPONENT_FMRI-$PYV" 2>/dev/null | egrep -q -v '(o|r)$' && continue
479         [[ -n "$NV" ]] && NV="$NV and "
480         NV="$NV$v"
481     done
482
483     REBUILDMSG=
484     [[ -n "$NV" ]] && REBUILDMSG="rebuild for python $NV"
485     if [[ -n "$OV" ]] ; then
486         [[ -n "$REBUILDMSG" ]] && REBUILDMSG="$REBUILDMSG and" || REBUILDMSG="rebuild"
487         REBUILDMSG="$REBUILDMSG to get package$OV_PLURAL for python $OV obsoleted"
488     fi
489
490     if [[ -n "$REBUILDMSG" ]] ; then
491         [[ -n "$MSG" ]] && MSG="$MSG; "
492         MSG="$MSG$REBUILDMSG"
493     fi
494     [[ -z "$MSG" ]] && MSG="rebuild"
495
f06a31 496     MSG="python/$DISTRIBUTION: $MSG"
053684 497 fi
MT 498
499 # Commit the results
500 ! git commit -m "$MSG" > /dev/null 2>&1 && printf "FATAL: 'git commit' failed!\n" >&2 && exit 1