219 lines
7.6 KiB
Python
219 lines
7.6 KiB
Python
#!/usr/bin/env python
|
|
# -*- coding: utf-8 -*-
|
|
#
|
|
# Copyright (C) 2012-2015 Calum Lind <calumlind@gmail.com>
|
|
# Copyright (C) 2010 Damien Churchill <damoxc@gmail.com>
|
|
# Copyright (C) 2009-2010 Andrew Resch <andrewresch@gmail.com>
|
|
# Copyright (C) 2009 Jesper Lund <mail@jesperlund.com>
|
|
#
|
|
# This file is part of Deluge and is licensed under GNU General Public License 3.0, or later, with
|
|
# the additional special exception to link portions of this program with the OpenSSL library.
|
|
# See LICENSE for more details.
|
|
#
|
|
|
|
from __future__ import print_function
|
|
|
|
import glob
|
|
import os
|
|
import re
|
|
import shutil
|
|
import sys
|
|
|
|
import bbfreeze
|
|
import gtk
|
|
from win32verstamp import stamp
|
|
|
|
import deluge.common
|
|
|
|
|
|
class VersionInfo(object):
|
|
def __init__(self, version, internalname=None, originalfilename=None,
|
|
comments=None, company=None, description=None,
|
|
_copyright=None, trademarks=None, product=None, dll=False,
|
|
debug=False, verbose=True):
|
|
parts = version.split('.')
|
|
while len(parts) < 4:
|
|
parts.append('0')
|
|
self.version = '.'.join(parts)
|
|
self.internal_name = internalname
|
|
self.original_filename = originalfilename
|
|
self.comments = comments
|
|
self.company = company
|
|
self.description = description
|
|
self.copyright = _copyright
|
|
self.trademarks = trademarks
|
|
self.product = product
|
|
self.dll = dll
|
|
self.debug = debug
|
|
self.verbose = verbose
|
|
|
|
|
|
DEBUG = False
|
|
if len(sys.argv) == 2 and sys.argv[1].lower() == 'debug':
|
|
DEBUG = True
|
|
|
|
# Get build_version from installed deluge.
|
|
build_version = deluge.common.get_version()
|
|
python_path = os.path.dirname(sys.executable)
|
|
if python_path.endswith('Scripts'):
|
|
python_path = python_path[:-8]
|
|
gtk_root = os.path.join(gtk.__path__[0], '..', 'runtime')
|
|
build_dir = os.path.join('build-win32', 'deluge-bbfreeze-' + build_version)
|
|
|
|
if DEBUG:
|
|
print('Python Path: %s' % python_path)
|
|
print('Gtk Path: %s' % gtk_root)
|
|
print('bbfreeze Output Path: %s' % build_dir)
|
|
|
|
print('Freezing Deluge %s...' % build_version)
|
|
# Disable printing to console for bbfreezing.
|
|
if not DEBUG:
|
|
sys.stdout = open(os.devnull, 'w')
|
|
|
|
# Include python modules not picked up automatically by bbfreeze.
|
|
includes = ('libtorrent', 'cairo', 'pangocairo', 'atk', 'pango', 'twisted.internet.utils',
|
|
'gio', 'gzip', 'email.mime.multipart', 'email.mime.text', '_cffi_backend')
|
|
excludes = ('numpy', 'OpenGL', 'psyco', 'win32ui', 'unittest')
|
|
|
|
|
|
def recipe_gtk_override(mf):
|
|
# Override bbfreeze function so that it includes all gtk libraries
|
|
# in the installer so users don't require a separate GTK+ installation.
|
|
return True
|
|
|
|
|
|
bbfreeze.recipes.recipe_gtk_and_friends = recipe_gtk_override
|
|
|
|
# Workaround for "ImportError: The 'packaging' package is required" with setuptools > 18.8.
|
|
# (https://github.com/pypa/setuptools/issues/517)
|
|
bbfreeze.recipes.recipe_pkg_resources = bbfreeze.recipes.include_whole_package('pkg_resources')
|
|
|
|
fzr = bbfreeze.Freezer(build_dir, includes=includes, excludes=excludes)
|
|
fzr.include_py = False
|
|
fzr.setIcon(os.path.join(os.path.dirname(deluge.common.__file__), 'ui', 'data', 'pixmaps', 'deluge.ico'))
|
|
|
|
# TODO: Can/should we grab the script list from setup.py entry_points somehow.
|
|
|
|
# Hide cmd console popup for these console entries force gui_script True.
|
|
force_gui = ['deluge-web', 'deluged']
|
|
|
|
for force_script in force_gui:
|
|
script_path = os.path.join(python_path, 'Scripts', force_script + '-script.py')
|
|
shutil.copy(script_path, script_path.replace('script', 'debug-script'))
|
|
|
|
script_list = []
|
|
for script in glob.glob(os.path.join(python_path, 'Scripts\\deluge*-script.py*')):
|
|
# Copy the scripts to remove the '-script' suffix before adding to freezer.
|
|
new_script = script.replace('-script', '')
|
|
shutil.copy(script, new_script)
|
|
|
|
gui_script = False
|
|
script_splitext = os.path.splitext(os.path.basename(new_script))
|
|
if script_splitext[1] == '.pyw' or script_splitext[0] in force_gui:
|
|
gui_script = True
|
|
try:
|
|
fzr.addScript(new_script, gui_only=gui_script)
|
|
script_list.append(new_script)
|
|
except Exception:
|
|
os.remove(script)
|
|
|
|
# Start the freezing process.
|
|
fzr()
|
|
|
|
# Clean up the duplicated scripts.
|
|
for script in script_list:
|
|
os.remove(script)
|
|
|
|
# Exclude files which are already included in GTK or Windows. Also exclude unneeded pygame dlls.
|
|
excludeDlls = ('MSIMG32.dll', 'MSVCR90.dll', 'MSVCP90.dll', 'MSVCR120.dll',
|
|
'POWRPROF.dll', 'DNSAPI.dll', 'USP10.dll', 'MPR.dll',
|
|
'jpeg.dll', 'libfreetype-6.dll', 'libpng12-0.dll', 'libtiff.dll',
|
|
'SDL_image.dll', 'SDL_ttf.dll')
|
|
for exclude_dll in excludeDlls:
|
|
try:
|
|
os.remove(os.path.join(build_dir, exclude_dll))
|
|
except OSError:
|
|
pass
|
|
|
|
# Re-enable printing.
|
|
if not DEBUG:
|
|
sys.stdout = sys.__stdout__
|
|
|
|
# Copy gtk locale files.
|
|
gtk_locale = os.path.join(gtk_root, 'share/locale')
|
|
locale_include_list = ['gtk20.mo', 'locale.alias']
|
|
|
|
|
|
def ignored_files(adir, ignore_filenames):
|
|
return [
|
|
ignore_file for ignore_file in ignore_filenames
|
|
if not os.path.isdir(os.path.join(adir, ignore_file)) and
|
|
ignore_file not in locale_include_list
|
|
]
|
|
|
|
|
|
shutil.copytree(gtk_locale, os.path.join(build_dir, 'share/locale'), ignore=ignored_files)
|
|
|
|
# Copy gtk theme files.
|
|
theme_include_list = [
|
|
[gtk_root, 'share/icons/hicolor/index.theme'],
|
|
[gtk_root, 'lib/gtk-2.0/2.10.0/engines'],
|
|
[gtk_root, 'share/themes/MS-Windows'],
|
|
['DelugeStart Theme', 'lib/gtk-2.0/2.10.0/engines/libmurrine.dll'],
|
|
['DelugeStart Theme', 'share/themes/DelugeStart'],
|
|
['DelugeStart Theme', 'etc/gtk-2.0/gtkrc']
|
|
]
|
|
for path_root, path in theme_include_list:
|
|
full_path = os.path.join(path_root, path)
|
|
if os.path.isdir(full_path):
|
|
shutil.copytree(full_path, os.path.join(build_dir, path))
|
|
else:
|
|
dst_dir = os.path.join(build_dir, os.path.dirname(path))
|
|
try:
|
|
os.makedirs(dst_dir)
|
|
except OSError:
|
|
pass
|
|
shutil.copy(full_path, dst_dir)
|
|
|
|
# Add version information to exe files.
|
|
for script in script_list:
|
|
script_exe = os.path.splitext(os.path.basename(script))[0] + '.exe'
|
|
# Don't add to dev build versions.
|
|
if not re.search('[a-zA-Z_-]', build_version):
|
|
versionInfo = VersionInfo(build_version,
|
|
description='Deluge Bittorrent Client',
|
|
company='Deluge Team',
|
|
product='Deluge',
|
|
_copyright='Deluge Team')
|
|
stamp(os.path.join(build_dir, script_exe), versionInfo)
|
|
|
|
# Copy version info to file for nsis script.
|
|
with open('VERSION.tmp', 'w') as ver_file:
|
|
ver_file.write('build_version = "%s"' % build_version)
|
|
|
|
# Create the install and uninstall file list for NSIS.
|
|
filedir_list = []
|
|
for root, dirnames, filenames in os.walk(build_dir):
|
|
dirnames.sort()
|
|
filenames.sort()
|
|
filedir_list.append((root[len(build_dir):], filenames))
|
|
|
|
with open('install_files.nsh', 'w') as f:
|
|
f.write('; Files to install\n')
|
|
for dirname, files in filedir_list:
|
|
if not dirname:
|
|
dirname = os.sep
|
|
f.write('\nSetOutPath "$INSTDIR%s"\n' % dirname)
|
|
for filename in files:
|
|
f.write('File "${BBFREEZE_DIR}%s"\n' % os.path.join(dirname, filename))
|
|
|
|
with open('uninstall_files.nsh', 'w') as f:
|
|
f.write('; Files to uninstall\n')
|
|
for dirname, files in reversed(filedir_list):
|
|
f.write('\n')
|
|
if not dirname:
|
|
dirname = os.sep
|
|
for filename in files:
|
|
f.write('Delete "$INSTDIR%s"\n' % os.path.join(dirname, filename))
|
|
f.write('RMDir "$INSTDIR%s"\n' % dirname)
|