1
0
Fork 0
Personal wiki notebook (not under development)
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

293 lines
8.7 KiB

#!/usr/bin/python
import os
import os.path
import sys
from glob import glob
from distutils.core import setup, Distribution
from config.Version import VERSION
def files( path, excludes = None ):
if excludes is None:
excludes = []
if sys.platform.startswith( "win" ):
path = path.replace( "/", "\\" )
files_in_path = glob( path )
for exclude in excludes:
if sys.platform.startswith( "win" ):
exclude = exclude.replace( "/", "\\" )
for file_to_exclude in glob( exclude ):
try:
files_in_path.remove( file_to_exclude )
except ValueError:
pass
return files_in_path
class Luminotes( Distribution ):
def __init__( self, attrs ):
self.ctypes_com_server = []
self.com_server = []
self.services = []
self.windows = [ dict(
script = "luminotes.py",
icon_resources = [ ( 0, "static\\images\\luminotes.ico" ) ],
) ]
self.console = []
self.service = []
self.isapi = []
self.zipfile = "lib\luminotes.zip"
Distribution.__init__( self, attrs )
manifest_template = '''
<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<assembly xmlns="urn:schemas-microsoft-com:asm.v1" manifestVersion="1.0">
<assemblyIdentity
version="5.0.0.0"
processorArchitecture="x86"
name="%(prog)s"
type="win32"
/>
<description>%(prog)s Program</description>
<dependency>
<dependentAssembly>
<assemblyIdentity
type="win32"
name="Microsoft.Windows.Common-Controls"
version="6.0.0.0"
processorArchitecture="X86"
publicKeyToken="6595b64144ccf1df"
language="*"
/>
</dependentAssembly>
</dependency>
</assembly>
'''
RT_MANIFEST = 24
luminotes = dict(
script = "luminotes.py",
other_resources = [(RT_MANIFEST, 1, manifest_template % dict(prog="Luminotes"))],
dest_base = r"prog\luminotes")
class InnoScript:
def __init__(self,
name,
lib_dir,
dist_dir,
windows_exe_files = [],
lib_files = [],
version = "1.0"):
self.lib_dir = lib_dir
self.dist_dir = dist_dir
if not self.dist_dir[-1] in "\\/":
self.dist_dir += "\\"
self.name = name
self.version = version
self.windows_exe_files = [self.chop(p) for p in windows_exe_files]
self.lib_files = [self.chop(p) for p in lib_files]
def chop(self, pathname):
assert pathname.startswith(self.dist_dir)
return pathname[len(self.dist_dir):]
def create(self, pathname="dist\\luminotes.iss"):
self.pathname = pathname
ofi = self.file = open(pathname, "w")
print >> ofi, "; WARNING: This script has been created by py2exe. Changes to this script"
print >> ofi, "; will be overwritten the next time py2exe is run!"
print >> ofi, r"[Setup]"
print >> ofi, r"AppName=%s" % self.name
print >> ofi, r"AppVerName=%s %s" % (self.name, self.version)
print >> ofi, r"DefaultDirName={pf}\%s" % self.name
print >> ofi, r"DisableProgramGroupPage=yes"
print >> ofi, r"SetupIconFile=static\images\luminotes.ico"
print >> ofi, r"WizardImageFile=static\images\installer.bmp"
print >> ofi, r"WizardSmallImageFile=static\images\installer_small.bmp"
print >> ofi
print >> ofi, r"[Files]"
for path in self.windows_exe_files + self.lib_files:
if path.endswith( "README.txt" ):
extra = " isreadme"
elif path.endswith( "luminotes.exe" ):
extra = "; BeforeInstall: stop_exe()"
else:
extra = ""
print >> ofi, r'Source: "%s"; DestDir: "{app}\%s"; Flags: ignoreversion%s' % (path, os.path.dirname(path), extra)
print >> ofi
print >> ofi, r"[Icons]"
for path in self.windows_exe_files:
print >> ofi, r'Name: "{commonprograms}\%s"; Filename: "{app}\%s"' % \
(self.name, path)
print >> ofi
print >> ofi, r"[UninstallDelete]"
print >> ofi, r'Type: files; Name: "{app}\luminotes.log"'
print >> ofi, r'Type: files; Name: "{app}\luminotes_error.log"'
print >> ofi
print >> ofi, r"[UninstallRun]"
print >> ofi, r'Filename: "{app}\luminotes.exe"; Parameters: "-k"; RunOnceId: LuminotesShutdown'
print >> ofi
print >> ofi, r"[Code]"
print >> ofi, r"procedure stop_exe();"
print >> ofi, r"var"
print >> ofi, r" result_code: Integer;"
print >> ofi, r"begin"
print >> ofi, r" Exec( ExpandConstant('{app}\luminotes.exe'), '-k', '', SW_SHOW, ewWaitUntilTerminated, result_code)"
print >> ofi, r"end;"
def compile(self):
import os
if os.environ.get( "CYGWIN" ):
print "Running in Cygwin, so skipping compile"
return
try:
import ctypes
except ImportError:
try:
import win32api
except ImportError:
import os
os.startfile(self.pathname)
else:
print "Ok, using win32api."
win32api.ShellExecute(0, "compile",
self.pathname,
None,
None,
0)
else:
print "Cool, you have ctypes installed."
res = ctypes.windll.shell32.ShellExecuteA(0, "compile",
self.pathname,
None,
None,
0)
if res < 32:
raise RuntimeError, "ShellExecute failed, error %d" % res
try:
import py2exe
from py2exe.build_exe import py2exe
class Build_installer( py2exe ):
# This class first builds the exe file(s), then creates a Windows installer.
# You need InnoSetup for it.
def run( self ):
# generate an initial database file
try:
os.remove( "luminotes.db" )
except OSError:
pass
from tools import initdb
initdb.main( ( "-l", ) )
# copy the README and COPYING files to have ".txt" extensions and Windows newlines
self.copy_doc( "README" )
self.copy_doc( "COPYING" )
# First, let py2exe do it's work.
py2exe.run(self)
lib_dir = self.lib_dir
dist_dir = self.dist_dir
# create the Installer, using the files py2exe has created.
script = InnoScript("Luminotes",
lib_dir,
dist_dir,
self.windows_exe_files,
self.lib_files,
version = VERSION)
print "*** creating the inno setup script***"
script.create()
print "*** compiling the inno setup script***"
script.compile()
# Note: By default the final setup.exe will be in an Output subdirectory.
@staticmethod
def copy_doc( path ):
out = file( "%s.txt" % path, "w" )
for line in file( path ).readlines():
line = line.rstrip( "\r\n" )
out.write( "%s\r\n" % line )
out.close()
except ImportError:
class Build_installer:
pass
if "py2exe" in sys.argv[ 1: ]:
txt_extension = ".txt"
else:
txt_extension = ""
data_files = [
( "", [ "README%s" % txt_extension, ] ),
( "", [ "COPYING%s" % txt_extension, ] ),
( "", [ "luminotes.db", ] ),
( "", [ "tools/usb_luminotes.bat" ] ),
( "static/css", files( "static/css/*.*" ) ),
( "static/html", files( "static/html/*.*" ) ),
( "static/images", files( "static/images/*.*", excludes = [
"static/images/*.xcf",
"static/images/*screenshot*",
"static/images/*tour*",
"static/images/*index_card*",
"static/images/*thumb*",
"static/images/dan.png",
] ) ),
( "static/images/toolbar", files( "static/images/toolbar/*.*", excludes = [ "static/images/toolbar/*.xcf" ] ) ),
( "static/images/toolbar/small", files( "static/images/toolbar/small/*.*", excludes = [ "static/images/toolbar/small/*.xcf" ] ) ),
( "static/themes/default/images/toolbar", files( "static/themes/default/images/toolbar/*.*", excludes = [ "static/themes/default/images/toolbar/*.xcf" ] ) ),
( "static/themes/default/images/toolbar/small", files( "static/themes/default/images/toolbar/small/*.*", excludes = [ "static/themes/default/images/toolbar/small/*.xcf" ] ) ),
( "static/js", files( "static/js/*.*" ) ),
( "static/js", files( "static/js/*_LICENSE" ) ),
( "model/delta", files( "model/delta/*.sqlite" ) ),
]
package_data = { ".": sum( [ pair[ 1 ] for pair in data_files ], [] ) }
setup(
name = "Luminotes",
version = VERSION,
author = "Dan Helfman",
author_email = "support@luminotes.com",
url = "http://luminotes.com",
description = "personal wiki notebook",
distclass = Luminotes,
cmdclass = { "py2exe": Build_installer }, # override default py2exe class
scripts = [ "luminotes.py" ],
packages = [ ".", "config", "controller", "model", "tools", "view" ],
package_dir = { ".": "." },
data_files = data_files, # for py2exe
package_data = package_data, # for everything else
options = dict(
py2exe = dict(
packages = "cherrypy.filters",
includes = "email.header,simplejson,plugins.*",
compressed = 1,
optimize = 2,
)
),
)