381 lines
14 KiB
Python
Executable File
381 lines
14 KiB
Python
Executable File
#!/usr/bin/env python
|
|
#
|
|
# setup.py
|
|
#
|
|
# Copyright (C) 2007 Andrew Resch <andrewresch@gmail.com>
|
|
# 2009 Damien Churchill <damoxc@gmail.com>
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation; either version 3, or (at your option)
|
|
# any later version.
|
|
#
|
|
# This program is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program. If not, write to:
|
|
# The Free Software Foundation, Inc.,
|
|
# 51 Franklin Street, Fifth Floor
|
|
# Boston, MA 02110-1301, USA.
|
|
#
|
|
|
|
import glob
|
|
import os
|
|
import platform
|
|
import sys
|
|
from distutils import cmd
|
|
from distutils.command.build import build as _build
|
|
from distutils.command.clean import clean as _clean
|
|
|
|
from setuptools import find_packages, setup
|
|
|
|
import msgfmt
|
|
from version import get_version
|
|
|
|
try:
|
|
from sphinx.setup_command import BuildDoc
|
|
except ImportError:
|
|
class BuildDoc(object):
|
|
pass
|
|
|
|
|
|
def windows_check():
|
|
return platform.system() in ('Windows', 'Microsoft')
|
|
|
|
desktop_data = 'deluge/ui/data/share/applications/deluge.desktop'
|
|
|
|
|
|
class BuildTranslations(cmd.Command):
|
|
description = 'Compile .po files into .mo files & create .desktop file'
|
|
|
|
user_options = [
|
|
('build-lib', None, "lib build folder"),
|
|
('develop', 'D', 'Compile translations in develop mode (deluge/i18n)')
|
|
]
|
|
boolean_options = ['develop']
|
|
|
|
def initialize_options(self):
|
|
self.build_lib = None
|
|
self.develop = False
|
|
|
|
def finalize_options(self):
|
|
self.set_undefined_options('build', ('build_lib', 'build_lib'))
|
|
|
|
def run(self):
|
|
po_dir = os.path.join(os.path.dirname(__file__), 'deluge/i18n/')
|
|
|
|
if self.develop:
|
|
basedir = po_dir
|
|
else:
|
|
basedir = os.path.join(self.build_lib, 'deluge', 'i18n')
|
|
|
|
if not windows_check():
|
|
# creates the translated desktop file
|
|
intltool_merge = 'intltool-merge'
|
|
intltool_merge_opts = '--utf8 --quiet --desktop-style'
|
|
desktop_in = 'deluge/ui/data/share/applications/deluge.desktop.in'
|
|
print('Creating desktop file: %s' % desktop_data)
|
|
os.system('C_ALL=C ' + '%s ' * 5 % (intltool_merge, intltool_merge_opts,
|
|
po_dir, desktop_in, desktop_data))
|
|
|
|
print('Compiling po files from %s...' % po_dir),
|
|
for path, names, filenames in os.walk(po_dir):
|
|
for f in filenames:
|
|
upto_date = False
|
|
if f.endswith('.po'):
|
|
lang = f[:len(f) - 3]
|
|
src = os.path.join(path, f)
|
|
dest_path = os.path.join(basedir, lang, 'LC_MESSAGES')
|
|
dest = os.path.join(dest_path, 'deluge.mo')
|
|
if not os.path.exists(dest_path):
|
|
os.makedirs(dest_path)
|
|
if not os.path.exists(dest):
|
|
sys.stdout.write('%s, ' % lang)
|
|
sys.stdout.flush()
|
|
msgfmt.make(src, dest)
|
|
else:
|
|
src_mtime = os.stat(src)[8]
|
|
dest_mtime = os.stat(dest)[8]
|
|
if src_mtime > dest_mtime:
|
|
sys.stdout.write('%s, ' % lang)
|
|
sys.stdout.flush()
|
|
msgfmt.make(src, dest)
|
|
else:
|
|
upto_date = True
|
|
|
|
if upto_date:
|
|
sys.stdout.write(' po files already upto date. ')
|
|
sys.stdout.write('\b\b \nFinished compiling translation files. \n')
|
|
|
|
|
|
class BuildPlugins(cmd.Command):
|
|
description = "Build plugins into .eggs"
|
|
|
|
user_options = [
|
|
('install-dir=', None, "develop install folder"),
|
|
('develop', 'D', 'Compile plugins in develop mode')
|
|
]
|
|
boolean_options = ['develop']
|
|
|
|
def initialize_options(self):
|
|
self.install_dir = None
|
|
self.develop = False
|
|
|
|
def finalize_options(self):
|
|
pass
|
|
|
|
def run(self):
|
|
# Build the plugin eggs
|
|
plugin_path = "deluge/plugins/*"
|
|
|
|
for path in glob.glob(plugin_path):
|
|
if os.path.exists(os.path.join(path, "setup.py")):
|
|
if self.develop and self.install_dir:
|
|
os.system("cd " + path + "&& " + sys.executable +
|
|
" setup.py develop --install-dir=%s" % self.install_dir)
|
|
elif self.develop:
|
|
os.system("cd " + path + "&& " + sys.executable + " setup.py develop")
|
|
else:
|
|
os.system("cd " + path + "&& " + sys.executable + " setup.py bdist_egg -d ..")
|
|
|
|
|
|
class EggInfoPlugins(cmd.Command):
|
|
description = "create a distribution's .egg-info directory"
|
|
|
|
user_options = []
|
|
|
|
def initialize_options(self):
|
|
pass
|
|
|
|
def finalize_options(self):
|
|
pass
|
|
|
|
def run(self):
|
|
# Build the plugin eggs
|
|
plugin_path = "deluge/plugins/*"
|
|
|
|
for path in glob.glob(plugin_path):
|
|
if os.path.exists(os.path.join(path, "setup.py")):
|
|
os.system("cd " + path + "&& " + sys.executable + " setup.py egg_info")
|
|
|
|
|
|
class BuildDocs(BuildDoc):
|
|
def run(self):
|
|
class Mock(object):
|
|
def __init__(self, *args, **kwargs):
|
|
pass
|
|
|
|
def __call__(self, *args, **kwargs):
|
|
return Mock()
|
|
|
|
def __getattr__(self, key):
|
|
return Mock()
|
|
|
|
def __setattr__(self, key, value):
|
|
self.__dict__[key] = value
|
|
|
|
def __len__(self):
|
|
return 0
|
|
|
|
def __getitem__(self, ii):
|
|
return " "
|
|
|
|
old_import = __builtins__.__import__
|
|
|
|
def new_import(name, globals={}, locals={}, fromlist=[], level=-1):
|
|
try:
|
|
return old_import(name, globals, locals, fromlist, level)
|
|
except ImportError as ex:
|
|
#sys.stdout.write("ImportError: %s\n" % ex)
|
|
return Mock()
|
|
|
|
except Exception as ex:
|
|
#sys.stdout.write("Skipping Exception: %s\n" % ex)
|
|
return Mock()
|
|
__builtins__.__import__ = new_import
|
|
|
|
BuildDoc.run(self)
|
|
|
|
|
|
class Build(_build):
|
|
sub_commands = [('BuildTranslations', None), ('BuildPlugins', None)] + _build.sub_commands
|
|
|
|
def run(self):
|
|
# Run all sub-commands (at least those that need to be run)
|
|
_build.run(self)
|
|
try:
|
|
from deluge._libtorrent import lt
|
|
print "Found libtorrent version: %s" % lt.version
|
|
except ImportError, e:
|
|
print "Warning libtorrent not found: %s" % e
|
|
|
|
|
|
class CleanPlugins(cmd.Command):
|
|
description = "Cleans the plugin folders"
|
|
user_options = [
|
|
('all', 'a', "remove all build output, not just temporary by-products")
|
|
]
|
|
boolean_options = ['all']
|
|
|
|
def initialize_options(self):
|
|
self.all = None
|
|
|
|
def finalize_options(self):
|
|
self.set_undefined_options('clean', ('all', 'all'))
|
|
|
|
def run(self):
|
|
print("Cleaning the plugin's folders..")
|
|
|
|
plugin_path = "deluge/plugins/*"
|
|
|
|
for path in glob.glob(plugin_path):
|
|
if os.path.exists(os.path.join(path, "setup.py")):
|
|
c = "cd " + path + "&& " + sys.executable + " setup.py clean"
|
|
if self.all:
|
|
c += " -a"
|
|
os.system(c)
|
|
|
|
# Delete the .eggs
|
|
if path[-4:] == ".egg":
|
|
print("Deleting %s" % path)
|
|
os.remove(path)
|
|
|
|
egg_info_dir_path = "deluge/plugins/*/*.egg-info"
|
|
|
|
for path in glob.glob(egg_info_dir_path):
|
|
# Delete the .egg-info's directories
|
|
if path[-9:] == ".egg-info":
|
|
print("Deleting %s" % path)
|
|
for fpath in os.listdir(path):
|
|
os.remove(os.path.join(path, fpath))
|
|
os.removedirs(path)
|
|
|
|
root_egg_info_dir_path = "deluge*.egg-info"
|
|
|
|
for path in glob.glob(root_egg_info_dir_path):
|
|
print("Deleting %s" % path)
|
|
for fpath in os.listdir(path):
|
|
os.remove(os.path.join(path, fpath))
|
|
os.removedirs(path)
|
|
|
|
|
|
class Clean(_clean):
|
|
sub_commands = _clean.sub_commands + [('CleanPlugins', None)]
|
|
|
|
def run(self):
|
|
# Run all sub-commands (at least those that need to be run)
|
|
for cmd_name in self.get_sub_commands():
|
|
self.run_command(cmd_name)
|
|
_clean.run(self)
|
|
|
|
if os.path.exists(desktop_data):
|
|
print("Deleting %s" % desktop_data)
|
|
os.remove(desktop_data)
|
|
|
|
cmdclass = {
|
|
'build': Build,
|
|
'build_trans': BuildTranslations,
|
|
'build_plugins': BuildPlugins,
|
|
'build_docs': BuildDocs,
|
|
'clean_plugins': CleanPlugins,
|
|
'clean': Clean,
|
|
'egg_info_plugins': EggInfoPlugins
|
|
}
|
|
|
|
# Data files to be installed to the system
|
|
_data_files = [
|
|
('share/icons/hicolor/scalable/apps', ['deluge/ui/data/icons/hicolor/scalable/apps/deluge.svg']),
|
|
('share/icons/hicolor/128x128/apps', ['deluge/ui/data/icons/hicolor/128x128/apps/deluge.png']),
|
|
('share/icons/hicolor/16x16/apps', ['deluge/ui/data/icons/hicolor/16x16/apps/deluge.png']),
|
|
('share/icons/hicolor/192x192/apps', ['deluge/ui/data/icons/hicolor/192x192/apps/deluge.png']),
|
|
('share/icons/hicolor/22x22/apps', ['deluge/ui/data/icons/hicolor/22x22/apps/deluge.png']),
|
|
('share/icons/hicolor/24x24/apps', ['deluge/ui/data/icons/hicolor/24x24/apps/deluge.png']),
|
|
('share/icons/hicolor/256x256/apps', ['deluge/ui/data/icons/hicolor/256x256/apps/deluge.png']),
|
|
('share/icons/hicolor/32x32/apps', ['deluge/ui/data/icons/hicolor/32x32/apps/deluge.png']),
|
|
('share/icons/hicolor/36x36/apps', ['deluge/ui/data/icons/hicolor/36x36/apps/deluge.png']),
|
|
('share/icons/hicolor/48x48/apps', ['deluge/ui/data/icons/hicolor/48x48/apps/deluge.png']),
|
|
('share/icons/hicolor/64x64/apps', ['deluge/ui/data/icons/hicolor/64x64/apps/deluge.png']),
|
|
('share/icons/hicolor/72x72/apps', ['deluge/ui/data/icons/hicolor/72x72/apps/deluge.png']),
|
|
('share/icons/hicolor/96x96/apps', ['deluge/ui/data/icons/hicolor/96x96/apps/deluge.png']),
|
|
('share/pixmaps', ['deluge/ui/data/pixmaps/deluge.png', 'deluge/ui/data/pixmaps/deluge.xpm']),
|
|
('share/man/man1', [
|
|
'docs/man/deluge.1',
|
|
'docs/man/deluged.1',
|
|
'docs/man/deluge-gtk.1',
|
|
'docs/man/deluge-web.1',
|
|
'docs/man/deluge-console.1'])
|
|
]
|
|
|
|
if not windows_check() and os.path.exists(desktop_data):
|
|
_data_files.append(('share/applications', [desktop_data]))
|
|
|
|
entry_points = {
|
|
"console_scripts": [
|
|
"deluge-console = deluge.ui.console:start"
|
|
],
|
|
"gui_scripts": [
|
|
"deluge = deluge.main:start_ui",
|
|
"deluge-gtk = deluge.ui.gtkui:start",
|
|
"deluge-web = deluge.ui.web:start",
|
|
"deluged = deluge.main:start_daemon"
|
|
]
|
|
}
|
|
|
|
if windows_check():
|
|
entry_points["console_scripts"].extend([
|
|
"deluge-debug = deluge.main:start_ui",
|
|
"deluge-web-debug = deluge.ui.web:start",
|
|
"deluged-debug = deluge.main:start_daemon"])
|
|
|
|
# Main setup
|
|
setup(
|
|
name="deluge",
|
|
version=get_version(prefix='deluge-', suffix='.dev0'),
|
|
fullname="Deluge Bittorrent Client",
|
|
description="Bittorrent Client",
|
|
author="Andrew Resch, Damien Churchill",
|
|
author_email="andrewresch@gmail.com, damoxc@gmail.com",
|
|
keywords="torrent bittorrent p2p fileshare filesharing",
|
|
long_description="""Deluge is a bittorrent client that utilizes a
|
|
daemon/client model. There are various user interfaces available for
|
|
Deluge such as the GTKui, the webui and a console ui. Deluge uses
|
|
libtorrent in it's backend to handle the bittorrent protocol.""",
|
|
url="http://deluge-torrent.org",
|
|
license="GPLv3",
|
|
cmdclass=cmdclass,
|
|
data_files=_data_files,
|
|
package_data={"deluge": ["ui/gtkui/glade/*.glade",
|
|
"ui/gtkui/glade/*.ui",
|
|
"ui/data/pixmaps/*.png",
|
|
"ui/data/pixmaps/*.svg",
|
|
"ui/data/pixmaps/*.ico",
|
|
"ui/data/pixmaps/*.gif",
|
|
"ui/data/pixmaps/flags/*.png",
|
|
"plugins/*.egg",
|
|
"i18n/*/LC_MESSAGES/*.mo",
|
|
"ui/web/index.html",
|
|
"ui/web/css/*.css",
|
|
"ui/web/icons/*.png",
|
|
"ui/web/images/*.gif",
|
|
"ui/web/images/*.png",
|
|
"ui/web/js/*.js",
|
|
"ui/web/js/*/*.js",
|
|
"ui/web/js/*/.order",
|
|
"ui/web/js/*/*/*.js",
|
|
"ui/web/js/*/*/.order",
|
|
"ui/web/js/*/*/*/*.js",
|
|
"ui/web/render/*.html",
|
|
"ui/web/themes/css/*.css",
|
|
"ui/web/themes/images/*/*.gif",
|
|
"ui/web/themes/images/*/*.png",
|
|
"ui/web/themes/images/*/*/*.gif",
|
|
"ui/web/themes/images/*/*/*.png"
|
|
]},
|
|
packages=find_packages(exclude=["plugins", "docs", "tests"]),
|
|
namespace_packages=["deluge", "deluge.plugins"],
|
|
entry_points=entry_points
|
|
)
|