2007-03-29 14:01:28 -07:00
#! /usr/bin/env python
2009-09-20 11:03:43 -07:00
# -*- coding: utf-8 -*-
2006-12-05 01:29:26 -07:00
2010-05-04 02:43:40 -07:00
# Tahoe-LAFS -- secure, distributed storage grid
2008-07-14 14:53:25 -07:00
#
2011-01-28 07:07:00 -08:00
# Copyright © 2008-2011 Allmydata, Inc.
2008-07-14 14:53:25 -07:00
#
2009-08-01 19:57:10 -07:00
# This file is part of Tahoe-LAFS.
2008-07-14 14:53:25 -07:00
#
2011-05-10 12:16:50 -07:00
# See the docs/about.rst file for licensing information.
2006-12-05 01:29:26 -07:00
2011-01-21 18:07:52 -08:00
import glob , os , stat , subprocess , sys , re
2008-09-11 18:03:21 -07:00
##### sys.path management
2008-12-05 17:30:54 -07:00
def pylibdir ( prefixdir ) :
pyver = " python %d . %d " % ( sys . version_info [ : 2 ] )
if sys . platform == " win32 " :
return os . path . join ( prefixdir , " Lib " , " site-packages " )
else :
return os . path . join ( prefixdir , " lib " , pyver , " site-packages " )
2008-09-11 18:03:21 -07:00
basedir = os . path . dirname ( os . path . abspath ( __file__ ) )
2008-12-05 17:30:54 -07:00
supportlib = pylibdir ( os . path . join ( basedir , " support " ) )
2009-08-17 18:00:57 -07:00
# locate our version number
def read_version_py ( infname ) :
try :
verstrline = open ( infname , " rt " ) . read ( )
except EnvironmentError :
return None
else :
VSRE = r " ^verstr = [ ' \" ]([^ ' \" ]*)[ ' \" ] "
mo = re . search ( VSRE , verstrline , re . M )
if mo :
return mo . group ( 1 )
version = read_version_py ( " src/allmydata/_version.py " )
2010-10-29 15:28:25 -07:00
APPNAME = ' allmydata-tahoe '
APPNAMEFILE = os . path . join ( ' src ' , ' allmydata ' , ' _appname.py ' )
APPNAMEFILESTR = " __appname__ = ' %s ' " % ( APPNAME , )
try :
curappnamefilestr = open ( APPNAMEFILE , ' rU ' ) . read ( )
except EnvironmentError :
# No file, or unreadable or something, okay then let's try to write one.
open ( APPNAMEFILE , " w " ) . write ( APPNAMEFILESTR )
else :
if curappnamefilestr . strip ( ) != APPNAMEFILESTR :
print " Error -- this setup.py file is configured with the ' application name ' to be ' %s ' , but there is already a file in place in ' %s ' which contains the contents ' %s ' . If the file is wrong, please remove it and setup.py will regenerate it and write ' %s ' into it. " % ( APPNAME , APPNAMEFILE , curappnamefilestr , APPNAMEFILESTR )
sys . exit ( - 1 )
# setuptools/zetuptoolz looks in __main__.__requires__ for a list of
# requirements. When running "python setup.py test", __main__ is
# setup.py, so we put the list here so that the requirements will be
# available for tests:
# Tahoe's dependencies are managed by the find_links= entry in setup.cfg and
# the _auto_deps.install_requires list, which is used in the call to setup()
# below.
adglobals = { }
execfile ( ' src/allmydata/_auto_deps.py ' , adglobals )
install_requires = adglobals [ ' install_requires ' ]
2011-01-14 18:26:51 -08:00
if len ( sys . argv ) > 1 and sys . argv [ 1 ] == ' --fakedependency ' :
del sys . argv [ 1 ]
install_requires + = [ " fakedependency >= 1.0.0 " ]
2010-11-18 23:40:43 -08:00
__requires__ = install_requires [ : ]
2010-10-29 15:28:25 -07:00
2010-05-23 13:52:28 -07:00
egg = os . path . realpath ( glob . glob ( ' setuptools-*.egg ' ) [ 0 ] )
sys . path . insert ( 0 , egg )
2010-09-21 00:32:58 -07:00
egg = os . path . realpath ( glob . glob ( ' darcsver-*.egg ' ) [ 0 ] )
sys . path . insert ( 0 , egg )
2011-01-17 22:54:45 -08:00
egg = os . path . realpath ( glob . glob ( ' setuptools_darcs-*.egg ' ) [ 0 ] )
sys . path . insert ( 0 , egg )
2010-05-23 13:52:28 -07:00
import setuptools ; setuptools . bootstrap_install_from = egg
2007-09-13 14:51:19 -07:00
2008-09-17 16:08:29 -07:00
from setuptools import find_packages , setup
2008-08-27 11:26:44 -07:00
from setuptools . command import sdist
2009-01-29 07:00:58 -07:00
from setuptools import Command
2008-12-01 11:48:04 -07:00
2007-04-27 13:47:15 -07:00
trove_classifiers = [
2008-04-14 11:08:23 -07:00
" Development Status :: 5 - Production/Stable " ,
2007-04-27 13:47:15 -07:00
" Environment :: Console " ,
" Environment :: Web Environment " ,
2008-07-14 14:53:25 -07:00
" License :: OSI Approved :: GNU General Public License (GPL) " ,
2008-01-08 11:45:00 -07:00
" License :: DFSG approved " ,
" License :: Other/Proprietary License " ,
2008-07-14 14:53:25 -07:00
" Intended Audience :: Developers " ,
2007-04-27 13:47:15 -07:00
" Intended Audience :: End Users/Desktop " ,
" Intended Audience :: System Administrators " ,
" Operating System :: Microsoft " ,
" Operating System :: Microsoft :: Windows " ,
2008-10-21 09:32:00 -07:00
" Operating System :: Microsoft :: Windows :: Windows NT/2000 " ,
2007-04-27 13:47:15 -07:00
" Operating System :: Unix " ,
" Operating System :: POSIX :: Linux " ,
" Operating System :: POSIX " ,
" Operating System :: MacOS :: MacOS X " ,
2008-07-14 14:53:25 -07:00
" Operating System :: OS Independent " ,
" Natural Language :: English " ,
" Programming Language :: C " ,
" Programming Language :: Python " ,
2008-10-21 09:32:00 -07:00
" Programming Language :: Python :: 2 " ,
" Programming Language :: Python :: 2.4 " ,
" Programming Language :: Python :: 2.5 " ,
2009-01-28 18:26:28 -07:00
" Programming Language :: Python :: 2.6 " ,
2011-01-08 13:12:12 -08:00
" Programming Language :: Python :: 2.7 " ,
2007-04-27 13:47:15 -07:00
" Topic :: Utilities " ,
" Topic :: System :: Systems Administration " ,
" Topic :: System :: Filesystems " ,
" Topic :: System :: Distributed Computing " ,
" Topic :: Software Development :: Libraries " ,
" Topic :: Communications :: Usenet News " ,
2008-07-14 14:53:25 -07:00
" Topic :: System :: Archiving :: Backup " ,
" Topic :: System :: Archiving :: Mirroring " ,
" Topic :: System :: Archiving " ,
2007-04-27 13:47:15 -07:00
]
2006-12-14 03:25:30 -07:00
2007-05-03 20:14:07 -07:00
2008-04-09 11:30:53 -07:00
setup_requires = [ ]
2008-09-16 18:36:27 -07:00
2009-02-18 23:57:51 -07:00
# The darcsver command from the darcsver plugin is needed to initialize the
# distribution's .version attribute correctly. (It does this either by
# examining darcs history, or if that fails by reading the
2009-05-07 14:50:03 -07:00
# src/allmydata/_version.py file). darcsver will also write a new version
# stamp in src/allmydata/_version.py, with a version number derived from
2009-02-18 23:57:51 -07:00
# darcs history. Note that the setup.cfg file has an "[aliases]" section
# which enumerates commands that you might run and specifies that it will run
# darcsver before each one. If you add different commands (or if I forgot
# some that are already in use), you may need to add it to setup.cfg and
# configure it to run darcsver before your command, if you want the version
# number to be correct when that command runs.
2009-02-03 23:24:05 -07:00
# http://pypi.python.org/pypi/darcsver
2011-01-20 21:30:38 -08:00
setup_requires . append ( ' darcsver >= 1.7.2 ' )
2009-02-03 23:24:05 -07:00
2011-07-21 16:49:41 -07:00
# Nevow imports itself when building, which causes Twisted and zope.interface
# to be imported. We need to make sure that the versions of Twisted and
# zope.interface used at build time satisfy Nevow's requirements. If not
# then there are two problems:
# - prior to Nevow v0.9.33, Nevow didn't declare its dependency on Twisted
# in a way that enabled setuptools to satisfy that requirement at
# build time.
# - some versions of zope.interface, e.g. v3.6.4, are incompatible with
# Nevow, and we need to avoid those both at build and run-time.
#
# This only matters when compatible versions of Twisted and zope.interface
# are not already installed. Retire this hack when
2011-07-21 16:36:58 -07:00
# https://bugs.launchpad.net/nevow/+bug/812537 has been fixed.
2011-07-21 16:49:41 -07:00
setup_requires + = [ req for req in install_requires if req . startswith ( ' Twisted ' ) or req . startswith ( ' zope.interface ' ) ]
2008-10-25 06:50:42 -07:00
2011-01-17 22:54:45 -08:00
# setuptools_darcs is required to produce complete distributions (such
# as with "sdist" or "bdist_egg"), unless there is a
2011-01-30 16:04:20 -08:00
# src/allmydata_tahoe.egg-info/SOURCE.txt file present which contains
# a complete list of files that should be included.
2011-01-17 22:54:45 -08:00
# http://pypi.python.org/pypi/setuptools_darcs
2011-01-30 16:04:20 -08:00
# However, requiring it runs afoul of a bug in Distribute, which was
# shipped in Ubuntu Lucid, so for now you have to manually install it
# before building sdists or eggs:
# http://bitbucket.org/tarek/distribute/issue/55/revision-control-plugin-automatically-installed-as-a-build-dependency-is-not-present-when-another-build-dependency-is-being
# Note that we explicitly inject setuptools_darcs at the beginning of
# this setup.py file, so it is still in effect when building dists
# using this setup.py file even when the following requirement is
# disabled.
if False :
setup_requires . append ( ' setuptools_darcs >= 1.1.0 ' )
2008-01-22 08:35:38 -07:00
2010-05-21 05:22:26 -07:00
# trialcoverage is required if you want the "trial" unit test runner to have a
# "--reporter=bwverbose-coverage" option which produces code-coverage results.
2010-05-24 17:44:44 -07:00
# The required version is 0.3.3, because that is the latest version that only
# depends on a version of pycoverage for which binary packages are available.
2010-05-21 05:22:26 -07:00
if " --reporter=bwverbose-coverage " in sys . argv :
2010-05-24 16:37:07 -07:00
setup_requires . append ( ' trialcoverage >= 0.3.3 ' )
2010-05-21 05:22:26 -07:00
2009-07-23 08:04:07 -07:00
# stdeb is required to produce Debian files with the "sdist_dsc" command.
if " sdist_dsc " in sys . argv :
setup_requires . append ( ' stdeb >= 0.3 ' )
2011-01-18 12:51:14 -08:00
# We no longer have any requirements specific to tests.
tests_require = [ ]
2010-06-08 22:05:42 -07:00
2011-01-18 18:45:32 -08:00
class Trial ( Command ) :
description = " run trial (use ' bin %s tahoe debug trial ' for the full set of trial options) " % ( os . sep , )
# This is just a subset of the most useful options, for compatibility.
user_options = [ ( " rterrors " , " e " , " Print out tracebacks as soon as they occur. " ) ,
( " reporter= " , None , " The reporter to use for this test run. " ) ,
( " suite= " , " s " , " Specify the test suite. " ) ,
2011-01-21 00:03:28 -08:00
( " quiet " , None , " Don ' t display version numbers and paths of Tahoe dependencies. " ) ,
2011-01-18 18:45:32 -08:00
]
def initialize_options ( self ) :
self . rterrors = False
self . reporter = None
self . suite = " allmydata "
2011-01-21 00:03:28 -08:00
self . quiet = False
2011-01-18 18:45:32 -08:00
def finalize_options ( self ) :
pass
def run ( self ) :
args = [ sys . executable , os . path . join ( ' bin ' , ' tahoe ' ) ]
2011-01-21 00:03:28 -08:00
if not self . quiet :
2011-01-18 18:45:32 -08:00
args . append ( ' --version-and-path ' )
args + = [ ' debug ' , ' trial ' ]
if self . rterrors :
args . append ( ' --rterrors ' )
if self . reporter :
args . append ( ' --reporter= ' + self . reporter )
if self . suite :
args . append ( self . suite )
rc = subprocess . call ( args )
sys . exit ( rc )
2009-01-28 18:07:16 -07:00
class MakeExecutable ( Command ) :
2011-01-19 15:33:05 -08:00
description = " make the ' bin %s tahoe ' scripts " % ( os . sep , )
2008-09-11 18:03:21 -07:00
user_options = [ ]
2011-01-19 15:33:05 -08:00
2008-09-11 18:03:21 -07:00
def initialize_options ( self ) :
pass
def finalize_options ( self ) :
pass
def run ( self ) :
2009-01-28 18:07:16 -07:00
bin_tahoe_template = os . path . join ( " bin " , " tahoe-script.template " )
2011-01-19 15:31:45 -08:00
# tahoe.pyscript is really only necessary for Windows, but we also
# create it on Unix for consistency.
script_names = [ " tahoe.pyscript " , " tahoe " ]
2010-07-25 01:32:16 -07:00
# Create the tahoe script file under the 'bin' directory. This
# file is exactly the same as the 'tahoe-script.template' script
# except that the shebang line is rewritten to use our sys.executable
# for the interpreter.
2009-01-28 18:07:16 -07:00
f = open ( bin_tahoe_template , " rU " )
script_lines = f . readlines ( )
f . close ( )
2010-07-25 01:32:16 -07:00
script_lines [ 0 ] = ' #! %s \n ' % ( sys . executable , )
for script_name in script_names :
tahoe_script = os . path . join ( " bin " , script_name )
2009-01-28 18:07:16 -07:00
try :
2010-07-25 01:32:16 -07:00
os . remove ( tahoe_script )
except Exception :
if os . path . exists ( tahoe_script ) :
raise
f = open ( tahoe_script , " wb " )
for line in script_lines :
f . write ( line )
f . close ( )
2011-01-30 17:58:09 -07:00
# chmod +x
unix_script = os . path . join ( " bin " , " tahoe " )
old_mode = stat . S_IMODE ( os . stat ( unix_script ) [ stat . ST_MODE ] )
new_mode = old_mode | ( stat . S_IXUSR | stat . S_IRUSR |
stat . S_IXGRP | stat . S_IRGRP |
stat . S_IXOTH | stat . S_IROTH )
os . chmod ( unix_script , new_mode )
2010-07-25 01:32:16 -07:00
old_tahoe_exe = os . path . join ( " bin " , " tahoe.exe " )
try :
os . remove ( old_tahoe_exe )
except Exception :
if os . path . exists ( old_tahoe_exe ) :
raise
2008-09-11 18:03:21 -07:00
2008-08-27 11:26:44 -07:00
class MySdist ( sdist . sdist ) :
""" A hook in the sdist command so that we can determine whether this the
tarball should be ' SUMO ' or not , i . e . whether or not to include the
2008-09-17 13:01:19 -07:00
external dependency tarballs . Note that we always include
misc / dependencies / * in the tarball ; - - sumo controls whether tahoe - deps / *
is included as well .
2008-08-27 11:26:44 -07:00
"""
user_options = sdist . sdist . user_options + \
2008-09-17 13:01:19 -07:00
[ ( ' sumo ' , ' s ' ,
" create a ' sumo ' sdist which includes the contents of tahoe-deps/* " ) ,
]
2008-08-27 11:26:44 -07:00
boolean_options = [ ' sumo ' ]
def initialize_options ( self ) :
sdist . sdist . initialize_options ( self )
2008-09-17 13:01:19 -07:00
self . sumo = False
def make_distribution ( self ) :
# add our extra files to the list just before building the
# tarball/zipfile. We override make_distribution() instead of run()
# because setuptools.command.sdist.run() does not lend itself to
# easy/robust subclassing (the code we need to add goes right smack
# in the middle of a 12-line method). If this were the distutils
# version, we'd override get_file_list().
if self . sumo :
# If '--sumo' was specified, include tahoe-deps/* in the sdist.
# We assume that the user has fetched the tahoe-deps.tar.gz
# tarball and unpacked it already.
self . filelist . extend ( [ os . path . join ( " tahoe-deps " , fn )
for fn in os . listdir ( " tahoe-deps " ) ] )
# In addition, we want the tarball/zipfile to have -SUMO in the
# name, and the unpacked directory to have -SUMO too. The easiest
# way to do this is to patch self.distribution and override the
# get_fullname() method. (an alternative is to modify
# self.distribution.metadata.version, but that also affects the
# contents of PKG-INFO).
fullname = self . distribution . get_fullname ( )
def get_fullname ( ) :
return fullname + " -SUMO "
self . distribution . get_fullname = get_fullname
return sdist . sdist . make_distribution ( self )
2008-08-27 11:26:44 -07:00
2011-01-19 15:33:05 -08:00
2009-08-17 18:00:57 -07:00
setup_args = { }
if version :
setup_args [ " version " ] = version
versioning: include an "appname" in the application version string in the versioning protocol, and make that appname be controlled by setup.py
It is currently hardcoded in setup.py to be 'allmydata-tahoe'. Ticket #556 is to make it configurable by a runtime command-line argument to setup.py: "--appname=foo", but I suddenly wondered if we really wanted that and at the same time realized that we don't need that for tahoe-1.3.0 release, so this patch just hardcodes it in setup.py.
setup.py inspects a file named 'src/allmydata/_appname.py' and assert that it contains the string "__appname__ = 'allmydata-tahoe'", and creates it if it isn't already present. src/allmydata/__init__.py import _appname and reads __appname__ from it. The rest of the Python code imports allmydata and inspects "allmydata.__appname__", although actually every use it uses "allmydata.__full_version__" instead, where "allmydata.__full_version__" is created in src/allmydata/__init__.py to be:
__full_version__ = __appname + '-' + str(__version__).
All the code that emits an "application version string" when describing what version of a protocol it supports (introducer server, storage server, upload helper), or when describing itself in general (introducer client), usese allmydata.__full_version__.
This fixes ticket #556 at least well enough for tahoe-1.3.0 release.
2009-02-11 17:18:16 -07:00
setup ( name = APPNAME ,
2008-01-22 10:22:51 -07:00
description = ' secure, decentralized, fault-tolerant filesystem ' ,
2010-05-04 02:43:40 -07:00
long_description = open ( ' README.txt ' , ' rU ' ) . read ( ) ,
author = ' the Tahoe-LAFS project ' ,
2010-09-30 08:37:08 -07:00
author_email = ' tahoe-dev@tahoe-lafs.org ' ,
2010-05-04 02:43:40 -07:00
url = ' http://tahoe-lafs.org/ ' ,
license = ' GNU GPL ' , # see README.txt -- there is an alternative licence
2011-07-31 20:19:52 -07:00
cmdclass = { " trial " : Trial ,
2009-01-28 18:07:16 -07:00
" make_executable " : MakeExecutable ,
2008-09-11 18:03:21 -07:00
" sdist " : MySdist ,
} ,
2007-11-09 18:04:19 -07:00
package_dir = { ' ' : ' src ' } ,
packages = find_packages ( " src " ) ,
2007-04-27 13:47:15 -07:00
classifiers = trove_classifiers ,
test_suite = " allmydata.test " ,
2008-09-11 18:03:21 -07:00
install_requires = install_requires ,
2010-06-08 22:05:42 -07:00
tests_require = tests_require ,
2007-11-09 18:04:19 -07:00
include_package_data = True ,
2007-12-31 23:28:31 -07:00
setup_requires = setup_requires ,
2007-10-11 03:38:24 -07:00
entry_points = { ' console_scripts ' : [ ' tahoe = allmydata.scripts.runner:run ' ] } ,
2007-09-13 15:37:55 -07:00
zip_safe = False , # We prefer unzipped for easier access.
2010-11-13 23:40:40 -08:00
versionfiles = [ ' src/allmydata/_version.py ' , ] ,
2009-08-17 18:00:57 -07:00
* * setup_args
2007-04-27 13:47:15 -07:00
)