mirror of
https://github.com/codex-storage/deluge.git
synced 2025-01-12 04:24:27 +00:00
c3a2c67b98
The usual minor fixes for unicode/bytes for library calls. The minimum Twisted version is now 16 for Python 3 support so remove old code and start replacing deprecated methods. Raised the minimum TLS version to 1.2 for the web server.
552 lines
17 KiB
Python
Executable File
552 lines
17 KiB
Python
Executable File
#!/usr/bin/env python
|
|
# -*- coding: utf-8 -*-
|
|
#
|
|
# Copyright (C) 2007 Andrew Resch <andrewresch@gmail.com>
|
|
# Copyright (C) 2009 Damien Churchill <damoxc@gmail.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 platform
|
|
import sys
|
|
from distutils import cmd
|
|
from distutils.command.build import build as _build
|
|
from distutils.command.clean import clean as _clean
|
|
from distutils.command.install_data import install_data as _install_data
|
|
from shutil import rmtree
|
|
|
|
from setuptools import find_packages, setup
|
|
from setuptools.command.test import test as _test
|
|
|
|
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')
|
|
|
|
|
|
def osx_check():
|
|
return platform.system() == 'Darwin'
|
|
|
|
|
|
desktop_data = 'deluge/ui/data/share/applications/deluge.desktop'
|
|
appdata_data = 'deluge/ui/data/share/appdata/deluge.appdata.xml'
|
|
|
|
# Variables for setuptools.setup
|
|
_package_data = {}
|
|
_exclude_package_data = {}
|
|
_entry_points = {'console_scripts': [], 'gui_scripts': [], 'deluge.ui': []}
|
|
_data_files = []
|
|
_version = get_version(prefix='deluge-', suffix='.dev0')
|
|
|
|
|
|
class PyTest(_test):
|
|
|
|
def initialize_options(self):
|
|
_test.initialize_options(self)
|
|
self.pytest_args = []
|
|
|
|
def finalize_options(self):
|
|
_test.finalize_options(self)
|
|
self.test_args = []
|
|
self.test_suite = True
|
|
|
|
def run_tests(self):
|
|
import pytest
|
|
errcode = pytest.main(self.test_args)
|
|
sys.exit(errcode)
|
|
|
|
|
|
class BuildDocs(BuildDoc):
|
|
description = 'Build the documentation'
|
|
|
|
def run(self):
|
|
print('Generating module documentation...')
|
|
os.system('sphinx-apidoc --force -o docs/source/modules/ deluge deluge/plugins')
|
|
BuildDoc.run(self)
|
|
|
|
|
|
class CleanDocs(cmd.Command):
|
|
description = 'Clean the documentation build and rst files'
|
|
user_options = []
|
|
|
|
def initialize_options(self):
|
|
pass
|
|
|
|
def finalize_options(self):
|
|
pass
|
|
|
|
def run(self):
|
|
for docs_dir in ('docs/build', 'docs/source/modules'):
|
|
try:
|
|
print('Deleting {}'.format(docs_dir))
|
|
rmtree(docs_dir)
|
|
except OSError:
|
|
pass
|
|
|
|
|
|
class BuildWebUI(cmd.Command):
|
|
description = 'Minify WebUI files'
|
|
user_options = []
|
|
|
|
JS_DIR = os.path.join('deluge', 'ui', 'web', 'js')
|
|
JS_SRC_DIRS = ('deluge-all', os.path.join('extjs', 'ext-extensions'))
|
|
|
|
def initialize_options(self):
|
|
pass
|
|
|
|
def finalize_options(self):
|
|
pass
|
|
|
|
def run(self):
|
|
js_basedir = os.path.join(os.path.dirname(__file__), self.JS_DIR)
|
|
|
|
try:
|
|
from minify_web_js import minify_js_dir
|
|
import_error = ''
|
|
except ImportError as err:
|
|
import_error = err
|
|
|
|
for js_src_dir in self.JS_SRC_DIRS:
|
|
source_dir = os.path.join(js_basedir, js_src_dir)
|
|
try:
|
|
minify_js_dir(source_dir)
|
|
except NameError:
|
|
js_file = source_dir + '.js'
|
|
if os.path.isfile(js_file):
|
|
print('Unable to minify but found existing minified: {}'.format(js_file))
|
|
else:
|
|
# Unable to minify and no existing minified file found so exiting.
|
|
print('Import error: %s' % import_error)
|
|
sys.exit(1)
|
|
|
|
# Create the gettext.js file for translations.
|
|
try:
|
|
from gen_web_gettext import create_gettext_js
|
|
except ImportError:
|
|
pass
|
|
else:
|
|
deluge_all_path = os.path.join(js_basedir, self.JS_SRC_DIRS[0])
|
|
print('Creating WebUI translation file: %s/gettext.js' % deluge_all_path)
|
|
create_gettext_js(deluge_all_path)
|
|
|
|
|
|
class CleanWebUI(cmd.Command):
|
|
description = 'Clean the documentation build and rst files'
|
|
user_options = []
|
|
|
|
def initialize_options(self):
|
|
pass
|
|
|
|
def finalize_options(self):
|
|
pass
|
|
|
|
def run(self):
|
|
js_basedir = os.path.join(os.path.dirname(__file__), BuildWebUI.JS_DIR)
|
|
for js_src_dir in BuildWebUI.JS_SRC_DIRS:
|
|
for file_type in ('.js', '-debug.js'):
|
|
js_file = os.path.join(js_basedir, js_src_dir + file_type)
|
|
print('Deleting {}'.format(js_file))
|
|
try:
|
|
os.remove(js_file)
|
|
except OSError:
|
|
pass
|
|
|
|
|
|
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():
|
|
intltool_merge = 'intltool-merge'
|
|
intltool_merge_opts = '--utf8 --quiet'
|
|
for data_file in (desktop_data, appdata_data):
|
|
# creates the translated file from .in file.
|
|
in_file = data_file + '.in'
|
|
if 'xml' in data_file:
|
|
intltool_merge_opts += ' --xml-style'
|
|
elif 'desktop' in data_file:
|
|
intltool_merge_opts += ' --desktop-style'
|
|
|
|
print('Creating file: %s' % data_file)
|
|
os.system('C_ALL=C ' + '%s ' * 5 % (
|
|
intltool_merge, intltool_merge_opts, po_dir, in_file, data_file))
|
|
|
|
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 CleanTranslations(cmd.Command):
|
|
description = 'Cleans translations files.'
|
|
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):
|
|
for path in (desktop_data, appdata_data):
|
|
if os.path.isfile(path):
|
|
print('Deleting %s' % path)
|
|
os.remove(path)
|
|
|
|
|
|
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 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'
|
|
print('Calling \'%s\'' % c)
|
|
os.system(c)
|
|
|
|
# Delete the .eggs
|
|
if path[-4:] == '.egg':
|
|
print('Deleting egg file "%s"' % path)
|
|
os.remove(path)
|
|
|
|
# Delete the .egg-link
|
|
if path[-9:] == '.egg-link':
|
|
print('Deleting egg link "%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)
|
|
|
|
|
|
class EggInfoPlugins(cmd.Command):
|
|
description = 'Create .egg-info directories for plugins'
|
|
|
|
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 Build(_build):
|
|
sub_commands = [
|
|
('build_webui', None),
|
|
('build_trans', None),
|
|
('build_plugins', 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_VERSION
|
|
print('Info: Found libtorrent ({}) installed.'.format(LT_VERSION))
|
|
except ImportError as ex:
|
|
print('Warning: libtorrent (libtorrent-rasterbar) not found: %s' % ex)
|
|
|
|
|
|
class InstallData(_install_data):
|
|
"""Custom class to fix `setup install` copying data files to incorrect location. (Bug #1389)"""
|
|
|
|
def finalize_options(self):
|
|
self.install_dir = None
|
|
self.set_undefined_options('install', ('install_data', 'install_dir'),
|
|
('root', 'root'), ('force', 'force'))
|
|
|
|
def run(self):
|
|
_install_data.run(self)
|
|
|
|
|
|
class Clean(_clean):
|
|
sub_commands = _clean.sub_commands + [
|
|
('clean_plugins', None),
|
|
('clean_trans', None),
|
|
('clean_webui', None)]
|
|
|
|
def run(self):
|
|
# Remove deluge egg-info.
|
|
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)
|
|
|
|
# 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)
|
|
|
|
|
|
cmdclass = {
|
|
'build': Build,
|
|
'build_webui': BuildWebUI,
|
|
'build_trans': BuildTranslations,
|
|
'build_plugins': BuildPlugins,
|
|
'build_docs': BuildDocs,
|
|
'install_data': InstallData,
|
|
'clean_plugins': CleanPlugins,
|
|
'clean_trans': CleanTranslations,
|
|
'clean_docs': CleanDocs,
|
|
'clean_webui': CleanWebUI,
|
|
'clean': Clean,
|
|
'egg_info_plugins': EggInfoPlugins,
|
|
'test': PyTest,
|
|
}
|
|
|
|
|
|
if not windows_check() and not osx_check():
|
|
for icon_path in glob.glob('deluge/ui/data/icons/hicolor/*x*'):
|
|
size = os.path.basename(icon_path)
|
|
_data_files.append(
|
|
('share/icons/hicolor/{}/apps'.format(size), ['{}/apps/deluge.png'.format(icon_path)]))
|
|
_data_files.extend([
|
|
('share/icons/hicolor/scalable/apps', ['deluge/ui/data/icons/hicolor/scalable/apps/deluge.svg']),
|
|
('share/pixmaps', ['deluge/ui/data/pixmaps/deluge.png']),
|
|
('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 os.path.isfile(desktop_data):
|
|
_data_files.append(('share/applications', [desktop_data]))
|
|
if os.path.isfile(appdata_data):
|
|
_data_files.append(('share/appdata', [appdata_data]))
|
|
|
|
_entry_points['console_scripts'] = [
|
|
'deluge-console = deluge.ui.console:start',
|
|
'deluge-web = deluge.ui.web:start',
|
|
'deluged = deluge.core.daemon_entry:start_daemon']
|
|
if windows_check():
|
|
_entry_points['console_scripts'].extend([
|
|
'deluge-debug = deluge.ui.ui_entry:start_ui',
|
|
'deluge-web-debug = deluge.ui.web:start',
|
|
'deluged-debug = deluge.core.daemon_entry:start_daemon'])
|
|
_entry_points['gui_scripts'] = [
|
|
'deluge = deluge.ui.ui_entry:start_ui',
|
|
'deluge-gtk = deluge.ui.gtkui:start']
|
|
_entry_points['deluge.ui'] = [
|
|
'console = deluge.ui.console:Console',
|
|
'web = deluge.ui.web:Web',
|
|
'gtk = deluge.ui.gtkui:Gtk']
|
|
|
|
|
|
_package_data['deluge'] = [
|
|
'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']
|
|
_package_data['deluge.ui.web'] = [
|
|
'index.html',
|
|
'css/*.css',
|
|
'icons/*.png',
|
|
'images/*.gif',
|
|
'images/*.png',
|
|
'js/*.js',
|
|
'js/extjs/*.js',
|
|
'render/*.html',
|
|
'themes/css/*.css',
|
|
'themes/images/*/*.gif',
|
|
'themes/images/*/*.png',
|
|
'themes/images/*/*/*.gif',
|
|
'themes/images/*/*/*.png']
|
|
_package_data['deluge.ui.gtkui'] = ['glade/*.ui']
|
|
|
|
if 'dev' not in _version:
|
|
_exclude_package_data['deluge.ui.web'] = ['*-debug.js', '*-debug.css']
|
|
|
|
docs_require = [
|
|
'Sphinx',
|
|
'recommonmark',
|
|
'sphinx-rtd-theme',
|
|
'sphinxcontrib-spelling',
|
|
]
|
|
tests_require = [
|
|
'coverage',
|
|
'flake8',
|
|
'flake8-blind-except',
|
|
'flake8-builtins',
|
|
'flake8-commas',
|
|
'flake8-comprehensions',
|
|
'flake8-debugger',
|
|
'flake8-isort',
|
|
'flake8-mock',
|
|
'flake8-mutable',
|
|
'flake8-quotes',
|
|
'pre-commit',
|
|
'pre-commit-hooks',
|
|
'pytest',
|
|
'detox',
|
|
'tox',
|
|
]
|
|
|
|
# Main setup
|
|
setup(
|
|
name='deluge',
|
|
version=_version,
|
|
fullname='Deluge BitTorrent Client',
|
|
description='BitTorrent Client',
|
|
author='Deluge Team',
|
|
maintainer='Calum Lind',
|
|
maintainer_email='calumlind+deluge@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 GTK-UI, the Web-UI and a Console-UI. Deluge uses
|
|
libtorrent in it's backend to handle the BitTorrent protocol.""",
|
|
url='https://deluge-torrent.org',
|
|
classifiers=[
|
|
'Development Status :: 4 - Beta',
|
|
'Environment :: Console',
|
|
'Environment :: Web Environment',
|
|
'Environment :: X11 Applications :: GTK',
|
|
'Framework :: Twisted',
|
|
'Intended Audience :: End Users/Desktop',
|
|
('License :: OSI Approved :: '
|
|
'GNU General Public License v3 or later (GPLv3+)'),
|
|
'Programming Language :: Python',
|
|
'Operating System :: MacOS :: MacOS X',
|
|
'Operating System :: Microsoft :: Windows',
|
|
'Operating System :: POSIX',
|
|
'Topic :: Internet'],
|
|
license='GPLv3',
|
|
cmdclass=cmdclass,
|
|
extras_require={
|
|
'docs': docs_require,
|
|
'tests': tests_require,
|
|
'dev': docs_require + tests_require,
|
|
},
|
|
tests_require=tests_require,
|
|
data_files=_data_files,
|
|
package_data=_package_data,
|
|
exclude_package_data=_exclude_package_data,
|
|
packages=find_packages(exclude=['deluge.plugins.*', 'deluge.tests']),
|
|
namespace_packages=['deluge', 'deluge.plugins'],
|
|
entry_points=_entry_points,
|
|
)
|