#!/usr/bin/env python # -*- coding: utf-8 -*- # # Copyright (C) 2012-2015 Calum Lind # Copyright (C) 2010 Damien Churchill # Copyright (C) 2009-2010 Andrew Resch # Copyright (C) 2009 Jesper Lund # # 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)