#!/usr/bin/env python # # The setup to create a Windows executable. # Inno Setup can then be used with the installer.iss file # in the top source directory to create an installer. # # Setuptools clashes with py2exe 0.6.8 (and probably later too). # For that reason, most of the code needs to have direct imports # that are not going through pkg_resources. # # Attention should be paid to dynamic imports. Data files can # be added to the distribution directory for that purpose. # See for example the 'images' directory below. import os, sys import platform if len(sys.argv) == 1: sys.argv.append('py2exe') # When using the SansView build script, we need to be able to pass # an extra path to be added to the python path. The extra arguments # should be removed from the list so that the setup processing doesn't # fail. try: if sys.argv.count('--extrapath'): path_flag_idx = sys.argv.index('--extrapath') extra_path = sys.argv[path_flag_idx+1] sys.path.insert(0, extra_path) del sys.argv[path_flag_idx+1] sys.argv.remove('--extrapath') except: print "Error processing extra python path needed to build SansView\n %s" % sys.exc_value from distutils.core import setup from distutils.filelist import findall import matplotlib # Solution taken from here: http://www.py2exe.org/index.cgi/win32com.shell # ModuleFinder can't handle runtime changes to __path__, but win32com uses them try: # py2exe 0.6.4 introduced a replacement modulefinder. # This means we have to add package paths there, not to the built-in # one. If this new modulefinder gets integrated into Python, then # we might be able to revert this some day. # if this doesn't work, try import modulefinder try: import py2exe.mf as modulefinder except ImportError: import modulefinder import win32com, sys for p in win32com.__path__[1:]: modulefinder.AddPackagePath("win32com", p) for extra in ["win32com.shell", "win32com.adsi", "win32com.axcontrol", "win32com.axscript", "win32com.bits", "win32com.ifilter", "win32com.internet", "win32com.mapi", "win32com.propsys", "win32com.taskscheduler"]: __import__(extra) m = sys.modules[extra] for p in m.__path__[1:]: modulefinder.AddPackagePath(extra, p) except ImportError: # no build path setup, no worries. pass import py2exe import shutil # Remove the build folder shutil.rmtree("build", ignore_errors=True) # do the same for dist folder shutil.rmtree("dist", ignore_errors=True) if sys.version_info < (2, 6): is_64bits = False origIsSystemDLL = py2exe.build_exe.isSystemDLL def isSystemDLL(pathname): if os.path.basename(pathname).lower() in ("msvcp71.dll", "comctl32.dll"): return 0 return origIsSystemDLL(pathname) py2exe.build_exe.isSystemDLL = isSystemDLL else: is_64bits = sys.maxsize > 2**32 if is_64bits and sys.version_info >= (2, 6): manifest = """ SansView """ else: manifest_for_python26 = """ SansView """ manifest_for_python25 = """ SansView """ # Select the appropriate manifest to use. py26MSdll_x86 = None if sys.version_info >= (3, 0) or sys.version_info < (2, 5): print "*** This script only works with Python 2.5, 2.6, or 2.7." sys.exit() elif sys.version_info >= (2, 6): manifest = manifest_for_python26 from glob import glob py26MSdll = glob(r"C:\Program Files\Microsoft Visual Studio 9.0\VC\redist\x86\Microsoft.VC90.CRT\*.*") try: py26MSdll_x86 = glob(r"C:\Program Files (x86)\Microsoft Visual Studio 9.0\VC\redist\x86\Microsoft.VC90.CRT\*.*") except: pass elif sys.version_info >= (2, 5): manifest = manifest_for_python25 py26MSdll = None class Target: def __init__(self, **kw): self.__dict__.update(kw) # for the versioninfo resources self.version = "2.1.0" self.company_name = "U Tennessee" self.copyright = "copyright 2009 - 2012" self.name = "SansView" # # Adapted from http://www.py2exe.org/index.cgi/MatPlotLib # to use the MatPlotLib. # path = os.getcwd() media_dir = os.path.join(path, "media") images_dir = os.path.join(path, "images") test_dir = os.path.join(path, "test") matplotlibdatadir = matplotlib.get_data_path() matplotlibdata = findall(matplotlibdatadir) data_files = [] # Copying SLD data import periodictable import logging data_files += periodictable.data_files() import sans.perspectives.fitting as fitting data_files += fitting.data_files() import sans.perspectives.calculator as calculator data_files += calculator.data_files() import sans.perspectives.invariant as invariant data_files += invariant.data_files() import sans.guiframe as guiframe data_files += guiframe.data_files() import sans.models as models data_files += models.data_files() for f in matplotlibdata: dirname = os.path.join('mpl-data', f[len(matplotlibdatadir)+1:]) data_files.append((os.path.split(dirname)[0], [f])) # Copy the settings file for the sans.dataloader file extension associations import sans.dataloader.readers f = os.path.join(sans.dataloader.readers.get_data_path(),'defaults.xml') if os.path.isfile(f): data_files.append(('.', [f])) f = 'custom_config.py' if os.path.isfile(f): data_files.append(('.', [f])) data_files.append(('config', [f])) f = 'local_config.py' if os.path.isfile(f): data_files.append(('.', [f])) if os.path.isfile("BUILD_NUMBER"): data_files.append(('.',["BUILD_NUMBER"])) # Copying the images directory to the distribution directory. for f in findall(images_dir): if os.path.split(f)[0].count('.svn')==0: data_files.append(("images", [f])) # Copying the HTML help docs for f in findall(media_dir): if os.path.split(f)[0].count('.svn')==0: data_files.append(("media", [f])) # Copying the sample data user data for f in findall(test_dir): if os.path.split(f)[0].count('.svn')==0: data_files.append(("test", [f])) if py26MSdll != None: # install the MSVC 9 runtime dll's into the application folder data_files.append(("Microsoft.VC90.CRT", py26MSdll)) if py26MSdll_x86 != None: # install the MSVC 9 runtime dll's into the application folder data_files.append(("Microsoft.VC90.CRT", py26MSdll_x86)) # packages # packages = ['matplotlib', 'scipy', 'pytz', 'encodings', 'comtypes', 'win32com'] includes = ['site'] # Exclude packages that are not needed but are often found on build systems excludes = ['Tkinter', 'PyQt4', '_ssl', '_tkagg', 'sip'] dll_excludes = ['libgdk_pixbuf-2.0-0.dll', 'libgobject-2.0-0.dll', 'libgdk-win32-2.0-0.dll', 'tcl84.dll', 'tk84.dll', 'QtGui4.dll', 'QtCore4.dll', 'msvcp90.dll', 'w9xpopen.exe', 'cygwin1.dll'] target_wx_client = Target( description = 'SansView', script = 'sansview.py', icon_resources = [(1, os.path.join(images_dir, "ball.ico"))], other_resources = [(24,1,manifest)], dest_base = "SansView" ) bundle_option = 2 if is_64bits: bundle_option = 3 setup( windows=[target_wx_client], console=[], options={ 'py2exe': { 'dll_excludes': dll_excludes, 'packages' : packages, 'includes':includes, 'excludes':excludes, "compressed": 1, "optimize": 0, "bundle_files":bundle_option, }, }, data_files=data_files, )