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