#!/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)