2016-09-10 00:04:55 -07:00
#!/usr/bin/python
2016-10-04 08:56:39 -07:00
import os , sys , argparse , random
2016-10-04 17:51:08 -07:00
from shell import shell , ssh
2016-09-10 00:04:55 -07:00
2016-10-04 08:56:39 -07:00
description = " Build release packages for pyqtgraph. "
2016-09-10 00:04:55 -07:00
2016-10-04 08:56:39 -07:00
epilog = """
Package build is done in several steps :
2016-09-10 00:04:55 -07:00
2016-10-17 09:20:27 -07:00
* Attempt to clone branch release - x . y . z from source - repo
2016-10-04 08:56:39 -07:00
* Merge release branch into master
* Write new version numbers into the source
* Roll over unreleased CHANGELOG entries
* Commit and tag new release
* Build HTML documentation
* Build source package
* Build deb packages ( if running on Linux )
2016-10-05 09:24:24 -07:00
* Build Windows exe installers
2016-10-04 08:56:39 -07:00
2016-10-06 09:07:22 -07:00
Release packages may be published by using the - - publish flag :
* Uploads release files to website
* Pushes tagged git commit to github
* Uploads source package to pypi
2016-10-04 08:56:39 -07:00
Building source packages requires :
*
*
* python - sphinx
Building deb packages requires several dependencies :
* build - essential
* python - all , python3 - all
* python - stdeb , python3 - stdeb
2016-10-17 09:20:27 -07:00
Note : building windows . exe files should be possible on any OS . However ,
2016-10-06 09:07:22 -07:00
Debian / Ubuntu systems do not include the necessary wininst * . exe files ; these
2016-10-17 09:20:27 -07:00
must be manually copied from the Python source to the distutils / command
submodule path ( / usr / lib / pythonX . X / distutils / command ) . Additionally , it may be
necessary to rename ( or copy / link ) wininst - 9.0 - amd64 . exe to
wininst - 6.0 - amd64 . exe .
2016-10-06 09:07:22 -07:00
2016-10-04 08:56:39 -07:00
"""
2016-10-06 09:07:22 -07:00
path = os . path . abspath ( os . path . join ( os . path . dirname ( __file__ ) , ' .. ' ) )
2016-10-04 08:56:39 -07:00
build_dir = os . path . join ( path , ' release-build ' )
pkg_dir = os . path . join ( path , ' release-packages ' )
ap = argparse . ArgumentParser ( description = description , epilog = epilog , formatter_class = argparse . RawDescriptionHelpFormatter )
ap . add_argument ( ' version ' , help = ' The x.y.z version to generate release packages for. '
' There must be a corresponding pyqtgraph-x.y.z branch in the source repository. ' )
2016-10-06 09:07:22 -07:00
ap . add_argument ( ' --publish ' , metavar = ' ' , help = ' Publish previously built package files (must be stored in pkg-dir/version) and tagged release commit (from build-dir). ' , action = ' store_const ' , const = True , default = False )
2016-10-04 08:56:39 -07:00
ap . add_argument ( ' --source-repo ' , metavar = ' ' , help = ' Repository from which release and master branches will be cloned. Default is the repo containing this script. ' , default = path )
ap . add_argument ( ' --build-dir ' , metavar = ' ' , help = ' Directory where packages will be staged and built. Default is source_root/release-build. ' , default = build_dir )
ap . add_argument ( ' --pkg-dir ' , metavar = ' ' , help = ' Directory where packages will be stored. Default is source_root/release-packages. ' , default = pkg_dir )
ap . add_argument ( ' --skip-pip-test ' , metavar = ' ' , help = ' Skip testing pip install. ' , action = ' store_const ' , const = True , default = False )
ap . add_argument ( ' --no-deb ' , metavar = ' ' , help = ' Skip building Debian packages. ' , action = ' store_const ' , const = True , default = False )
ap . add_argument ( ' --no-exe ' , metavar = ' ' , help = ' Skip building Windows exe installers. ' , action = ' store_const ' , const = True , default = False )
2016-10-05 09:24:24 -07:00
2016-10-04 08:56:39 -07:00
2016-10-06 09:07:22 -07:00
def build ( args ) :
if os . path . exists ( args . build_dir ) :
sys . stderr . write ( " Please remove the build directory %s before proceeding, or specify a different path with --build-dir. \n " % args . build_dir )
sys . exit ( - 1 )
if os . path . exists ( args . pkg_dir ) :
sys . stderr . write ( " Please remove the package directory %s before proceeding, or specify a different path with --pkg-dir. \n " % args . pkg_dir )
sys . exit ( - 1 )
# Clone source repository and tag the release branch
shell ( '''
# Clone and merge release branch into previous master
mkdir - p { build_dir }
cd { build_dir }
rm - rf pyqtgraph
2016-11-07 17:57:23 -08:00
git clone - - depth 1 - - branch master - - single - branch { source_repo } pyqtgraph
2016-10-06 09:07:22 -07:00
cd pyqtgraph
git checkout - b release - { version }
git pull { source_repo } release - { version }
git checkout master
git merge - - no - ff - - no - commit release - { version }
# Write new version number into the source
sed - i " s/__version__ = .*/__version__ = ' {version} ' / " pyqtgraph / __init__ . py
sed - i " s/version = .*/version = ' {version} ' / " doc / source / conf . py
sed - i " s/release = .*/release = ' {version} ' / " doc / source / conf . py
# make sure changelog mentions unreleased changes
grep " pyqtgraph- {version} .*unreleased.* " CHANGELOG
sed - i " s/pyqtgraph- {version} .*unreleased.*/pyqtgraph- {version} / " CHANGELOG
2016-09-10 00:04:55 -07:00
2016-10-06 09:07:22 -07:00
# Commit and tag new release
git commit - a - m " PyQtGraph release {version} "
git tag pyqtgraph - { version }
2016-09-10 00:04:55 -07:00
2016-10-06 09:07:22 -07:00
# Build HTML documentation
cd doc
make clean
make html
cd . .
find . / - name " *.pyc " - delete
2016-09-10 00:04:55 -07:00
2016-10-06 09:07:22 -07:00
# package source distribution
python setup . py sdist
2016-09-10 00:04:55 -07:00
2016-10-06 09:07:22 -07:00
mkdir - p { pkg_dir }
cp dist / * . tar . gz { pkg_dir }
2016-09-10 00:04:55 -07:00
2016-10-06 09:07:22 -07:00
# source package build complete.
''' .format(**args.__dict__))
2016-09-10 00:04:55 -07:00
2016-10-04 08:56:39 -07:00
2016-10-06 09:07:22 -07:00
if args . skip_pip_test :
args . pip_test = ' skipped '
else :
shell ( '''
# test pip install source distribution
rm - rf release - { version } - virtenv
virtualenv - - system - site - packages release - { version } - virtenv
. release - { version } - virtenv / bin / activate
echo " PATH: $PATH "
echo " ENV: $VIRTUAL_ENV "
pip install - - no - index - - no - deps dist / pyqtgraph - { version } . tar . gz
deactivate
# pip install test passed
''' .format(**args.__dict__))
args . pip_test = ' passed '
if ' linux ' in sys . platform and not args . no_deb :
shell ( '''
# build deb packages
cd { build_dir } / pyqtgraph
python setup . py - - command - packages = stdeb . command sdist_dsc
cd deb_dist / pyqtgraph - { version }
sed - i " s/^Depends:.*/Depends: python (>= 2.6), python-qt4 | python-pyside, python-numpy/ " debian / control
dpkg - buildpackage
cd . . / . . /
mv deb_dist { pkg_dir } / pyqtgraph - { version } - deb
# deb package build complete.
''' .format(**args.__dict__))
args . deb_status = ' built '
else :
args . deb_status = ' skipped '
2016-09-10 00:04:55 -07:00
2016-10-06 09:07:22 -07:00
if not args . no_exe :
shell ( """
# Build windows executables
cd { build_dir } / pyqtgraph
python setup . py build bdist_wininst - - plat - name = win32
2016-10-17 09:20:27 -07:00
python setup . py build bdist_wininst - - plat - name = win - amd64
2016-10-06 09:07:22 -07:00
cp dist / * . exe { pkg_dir }
""" .format(**args.__dict__))
args . exe_status = ' built '
else :
args . exe_status = ' skipped '
2016-09-10 00:04:55 -07:00
2016-10-06 09:07:22 -07:00
print ( unindent ( """
== == == == Build complete . == == == == =
* Source package : built
* Pip install test : { pip_test }
* Debian packages : { deb_status }
* Windows installers : { exe_status }
* Package files in { pkg_dir }
Next steps to publish :
2016-10-04 08:56:39 -07:00
2016-10-06 09:07:22 -07:00
* Test all packages
* Run script again with - - publish
""" ).format(**args.__dict__))
2016-10-04 08:56:39 -07:00
2016-10-06 09:07:22 -07:00
def publish ( args ) :
if not os . path . isfile ( os . path . expanduser ( ' ~/.pypirc ' ) ) :
print ( unindent ( """
Missing ~ / . pypirc file . Should look like :
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
[ distutils ]
index - servers =
pypi
[ pypi ]
username : your_username
password : your_password
""" ))
sys . exit ( - 1 )
### Upload everything to server
2016-10-05 09:24:24 -07:00
shell ( """
2016-10-06 09:07:22 -07:00
cd { build_dir } / pyqtgraph
2016-11-07 17:57:23 -08:00
# Uploading documentation.. (disabled; now hosted by readthedocs.io)
#rsync -rv doc/build/* pyqtgraph.org:/www/code/pyqtgraph/pyqtgraph/documentation/build/
2016-10-06 09:07:22 -07:00
# Uploading release packages to website
2016-11-07 17:57:23 -08:00
rsync - v { pkg_dir } pyqtgraph . org : / www / code / pyqtgraph / downloads /
2016-10-06 09:07:22 -07:00
2016-11-07 17:57:23 -08:00
# Push master to github
git push https : / / github . com / pyqtgraph / pyqtgraph master : master
# Push tag to github
git push https : / / github . com / pyqtgraph / pyqtgraph pyqtgraph - { version }
2016-10-06 09:07:22 -07:00
# Upload to pypi..
python setup . py sdist upload
""" .format(**args.__dict__))
print ( unindent ( """
== == == == Upload complete . == == == == =
Next steps to publish :
- update website
- mailing list announcement
- new conda recipe ( http : / / conda . pydata . org / docs / build . html )
- contact deb maintainer ( gianfranco costamagna )
- other package maintainers ?
""" ).format(**args.__dict__))
def unindent ( msg ) :
ind = 1e6
lines = msg . split ( ' \n ' )
for line in lines :
if len ( line . strip ( ) ) == 0 :
continue
ind = min ( ind , len ( line ) - len ( line . lstrip ( ) ) )
return ' \n ' . join ( [ line [ ind : ] for line in lines ] )
if __name__ == ' __main__ ' :
args = ap . parse_args ( )
args . build_dir = os . path . abspath ( args . build_dir )
args . pkg_dir = os . path . join ( os . path . abspath ( args . pkg_dir ) , args . version )
if args . publish :
publish ( args )
else :
build ( args )