Merge branch 'master' into development
# Conflicts: # cps/server.py # test/Calibre-Web TestSummary_Linux.html
This commit is contained in:
commit
d87ccae6c9
|
@ -154,7 +154,8 @@ def get_comic_info(tmp_file_path, original_file_name, original_file_extension, r
|
||||||
tags="",
|
tags="",
|
||||||
series=loadedMetadata.series or "",
|
series=loadedMetadata.series or "",
|
||||||
series_id=loadedMetadata.issue or "",
|
series_id=loadedMetadata.issue or "",
|
||||||
languages=loadedMetadata.language)
|
languages=loadedMetadata.language,
|
||||||
|
publisher="")
|
||||||
|
|
||||||
return BookMeta(
|
return BookMeta(
|
||||||
file_path=tmp_file_path,
|
file_path=tmp_file_path,
|
||||||
|
@ -166,4 +167,5 @@ def get_comic_info(tmp_file_path, original_file_name, original_file_extension, r
|
||||||
tags="",
|
tags="",
|
||||||
series="",
|
series="",
|
||||||
series_id="",
|
series_id="",
|
||||||
languages="")
|
languages="",
|
||||||
|
publisher="")
|
||||||
|
|
|
@ -152,7 +152,7 @@ def selected_roles(dictionary):
|
||||||
|
|
||||||
# :rtype: BookMeta
|
# :rtype: BookMeta
|
||||||
BookMeta = namedtuple('BookMeta', 'file_path, extension, title, author, cover, description, tags, series, '
|
BookMeta = namedtuple('BookMeta', 'file_path, extension, title, author, cover, description, tags, series, '
|
||||||
'series_id, languages')
|
'series_id, languages, publisher')
|
||||||
|
|
||||||
STABLE_VERSION = {'version': '0.6.12 Beta'}
|
STABLE_VERSION = {'version': '0.6.12 Beta'}
|
||||||
|
|
||||||
|
|
|
@ -444,10 +444,10 @@ def edit_book_languages(languages, book, upload=False):
|
||||||
return modify_database_object(input_l, book.languages, db.Languages, calibre_db.session, 'languages')
|
return modify_database_object(input_l, book.languages, db.Languages, calibre_db.session, 'languages')
|
||||||
|
|
||||||
|
|
||||||
def edit_book_publisher(to_save, book):
|
def edit_book_publisher(publishers, book):
|
||||||
changed = False
|
changed = False
|
||||||
if to_save["publisher"]:
|
if publishers:
|
||||||
publisher = to_save["publisher"].rstrip().strip()
|
publisher = publishers.rstrip().strip()
|
||||||
if len(book.publishers) == 0 or (len(book.publishers) > 0 and publisher != book.publishers[0].name):
|
if len(book.publishers) == 0 or (len(book.publishers) > 0 and publisher != book.publishers[0].name):
|
||||||
changed |= modify_database_object([publisher], book.publishers, db.Publishers, calibre_db.session,
|
changed |= modify_database_object([publisher], book.publishers, db.Publishers, calibre_db.session,
|
||||||
'publisher')
|
'publisher')
|
||||||
|
@ -740,7 +740,7 @@ def edit_book(book_id):
|
||||||
book.pubdate = db.Books.DEFAULT_PUBDATE
|
book.pubdate = db.Books.DEFAULT_PUBDATE
|
||||||
|
|
||||||
# handle book publisher
|
# handle book publisher
|
||||||
modif_date |= edit_book_publisher(to_save, book)
|
modif_date |= edit_book_publisher(to_save['publisher'], book)
|
||||||
|
|
||||||
# handle book languages
|
# handle book languages
|
||||||
modif_date |= edit_book_languages(to_save['languages'], book)
|
modif_date |= edit_book_languages(to_save['languages'], book)
|
||||||
|
@ -867,6 +867,9 @@ def create_book_on_upload(modif_date, meta):
|
||||||
# handle tags
|
# handle tags
|
||||||
modif_date |= edit_book_tags(meta.tags, db_book)
|
modif_date |= edit_book_tags(meta.tags, db_book)
|
||||||
|
|
||||||
|
# handle publisher
|
||||||
|
modif_date |= edit_book_publisher(meta.publisher, db_book)
|
||||||
|
|
||||||
# handle series
|
# handle series
|
||||||
modif_date |= edit_book_series(meta.series, db_book)
|
modif_date |= edit_book_series(meta.series, db_book)
|
||||||
|
|
||||||
|
|
|
@ -142,4 +142,5 @@ def get_epub_info(tmp_file_path, original_file_name, original_file_extension):
|
||||||
tags=epub_metadata['subject'].encode('utf-8').decode('utf-8'),
|
tags=epub_metadata['subject'].encode('utf-8').decode('utf-8'),
|
||||||
series=epub_metadata['series'].encode('utf-8').decode('utf-8'),
|
series=epub_metadata['series'].encode('utf-8').decode('utf-8'),
|
||||||
series_id=epub_metadata['series_id'].encode('utf-8').decode('utf-8'),
|
series_id=epub_metadata['series_id'].encode('utf-8').decode('utf-8'),
|
||||||
languages=epub_metadata['language'])
|
languages=epub_metadata['language'],
|
||||||
|
publisher="")
|
||||||
|
|
27
cps/fb2.py
27
cps/fb2.py
|
@ -30,51 +30,52 @@ def get_fb2_info(tmp_file_path, original_file_extension):
|
||||||
}
|
}
|
||||||
|
|
||||||
fb2_file = open(tmp_file_path)
|
fb2_file = open(tmp_file_path)
|
||||||
tree = etree.fromstring(fb2_file.read())
|
tree = etree.fromstring(fb2_file.read().encode())
|
||||||
|
|
||||||
authors = tree.xpath('/fb:FictionBook/fb:description/fb:title-info/fb:author', namespaces=ns)
|
authors = tree.xpath('/fb:FictionBook/fb:description/fb:title-info/fb:author', namespaces=ns)
|
||||||
|
|
||||||
def get_author(element):
|
def get_author(element):
|
||||||
last_name = element.xpath('fb:last-name/text()', namespaces=ns)
|
last_name = element.xpath('fb:last-name/text()', namespaces=ns)
|
||||||
if len(last_name):
|
if len(last_name):
|
||||||
last_name = last_name[0].encode('utf-8')
|
last_name = last_name[0]
|
||||||
else:
|
else:
|
||||||
last_name = u''
|
last_name = u''
|
||||||
middle_name = element.xpath('fb:middle-name/text()', namespaces=ns)
|
middle_name = element.xpath('fb:middle-name/text()', namespaces=ns)
|
||||||
if len(middle_name):
|
if len(middle_name):
|
||||||
middle_name = middle_name[0].encode('utf-8')
|
middle_name = middle_name[0]
|
||||||
else:
|
else:
|
||||||
middle_name = u''
|
middle_name = u''
|
||||||
first_name = element.xpath('fb:first-name/text()', namespaces=ns)
|
first_name = element.xpath('fb:first-name/text()', namespaces=ns)
|
||||||
if len(first_name):
|
if len(first_name):
|
||||||
first_name = first_name[0].encode('utf-8')
|
first_name = first_name[0]
|
||||||
else:
|
else:
|
||||||
first_name = u''
|
first_name = u''
|
||||||
return (first_name.decode('utf-8') + u' '
|
return (first_name + u' '
|
||||||
+ middle_name.decode('utf-8') + u' '
|
+ middle_name + u' '
|
||||||
+ last_name.decode('utf-8')).encode('utf-8')
|
+ last_name)
|
||||||
|
|
||||||
author = str(", ".join(map(get_author, authors)))
|
author = str(", ".join(map(get_author, authors)))
|
||||||
|
|
||||||
title = tree.xpath('/fb:FictionBook/fb:description/fb:title-info/fb:book-title/text()', namespaces=ns)
|
title = tree.xpath('/fb:FictionBook/fb:description/fb:title-info/fb:book-title/text()', namespaces=ns)
|
||||||
if len(title):
|
if len(title):
|
||||||
title = str(title[0].encode('utf-8'))
|
title = str(title[0])
|
||||||
else:
|
else:
|
||||||
title = u''
|
title = u''
|
||||||
description = tree.xpath('/fb:FictionBook/fb:description/fb:publish-info/fb:book-name/text()', namespaces=ns)
|
description = tree.xpath('/fb:FictionBook/fb:description/fb:publish-info/fb:book-name/text()', namespaces=ns)
|
||||||
if len(description):
|
if len(description):
|
||||||
description = str(description[0].encode('utf-8'))
|
description = str(description[0])
|
||||||
else:
|
else:
|
||||||
description = u''
|
description = u''
|
||||||
|
|
||||||
return BookMeta(
|
return BookMeta(
|
||||||
file_path=tmp_file_path,
|
file_path=tmp_file_path,
|
||||||
extension=original_file_extension,
|
extension=original_file_extension,
|
||||||
title=title.decode('utf-8'),
|
title=title,
|
||||||
author=author.decode('utf-8'),
|
author=author,
|
||||||
cover=None,
|
cover=None,
|
||||||
description=description.decode('utf-8'),
|
description=description,
|
||||||
tags="",
|
tags="",
|
||||||
series="",
|
series="",
|
||||||
series_id="",
|
series_id="",
|
||||||
languages="")
|
languages="",
|
||||||
|
publisher="")
|
||||||
|
|
|
@ -57,28 +57,30 @@ def get_language_name(locale, lang_code):
|
||||||
|
|
||||||
def get_language_codes(locale, language_names, remainder=None):
|
def get_language_codes(locale, language_names, remainder=None):
|
||||||
language_names = set(x.strip().lower() for x in language_names if x)
|
language_names = set(x.strip().lower() for x in language_names if x)
|
||||||
languages = list()
|
lang = list()
|
||||||
for k, v in get_language_names(locale).items():
|
for k, v in get_language_names(locale).items():
|
||||||
v = v.lower()
|
v = v.lower()
|
||||||
if v in language_names:
|
if v in language_names:
|
||||||
languages.append(k)
|
lang.append(k)
|
||||||
language_names.remove(v)
|
language_names.remove(v)
|
||||||
if remainder is not None:
|
if remainder is not None:
|
||||||
remainder.extend(language_names)
|
remainder.extend(language_names)
|
||||||
return languages
|
return lang
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
def get_valid_language_codes(locale, language_names, remainder=None):
|
def get_valid_language_codes(locale, language_names, remainder=None):
|
||||||
languages = list()
|
lang = list()
|
||||||
if "" in language_names:
|
if "" in language_names:
|
||||||
language_names.remove("")
|
language_names.remove("")
|
||||||
for k, __ in get_language_names(locale).items():
|
for k, __ in get_language_names(locale).items():
|
||||||
if k in language_names:
|
if k in language_names:
|
||||||
languages.append(k)
|
lang.append(k)
|
||||||
language_names.remove(k)
|
language_names.remove(k)
|
||||||
if remainder is not None and len(language_names):
|
if remainder is not None and len(language_names):
|
||||||
remainder.extend(language_names)
|
remainder.extend(language_names)
|
||||||
return languages
|
return lang
|
||||||
|
|
||||||
|
|
||||||
def get_lang3(lang):
|
def get_lang3(lang):
|
||||||
try:
|
try:
|
||||||
|
|
|
@ -251,7 +251,7 @@ class WebServer(object):
|
||||||
finally:
|
finally:
|
||||||
self.wsgiserver = None
|
self.wsgiserver = None
|
||||||
|
|
||||||
# prevent irritiating log of pending tasks message from asyncio
|
# prevent irritating log of pending tasks message from asyncio
|
||||||
logger.get('asyncio').setLevel(logger.logging.CRITICAL)
|
logger.get('asyncio').setLevel(logger.logging.CRITICAL)
|
||||||
|
|
||||||
if not self.restart:
|
if not self.restart:
|
||||||
|
|
|
@ -330,6 +330,7 @@ class Updater(threading.Thread):
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def _load_nightly_data(repository_url, commit, status):
|
def _load_nightly_data(repository_url, commit, status):
|
||||||
|
update_data = dict()
|
||||||
try:
|
try:
|
||||||
headers = {'Accept': 'application/vnd.github.v3+json'}
|
headers = {'Accept': 'application/vnd.github.v3+json'}
|
||||||
r = requests.get(repository_url + '/git/commits/' + commit['object']['sha'],
|
r = requests.get(repository_url + '/git/commits/' + commit['object']['sha'],
|
||||||
|
|
200
cps/uploader.py
200
cps/uploader.py
|
@ -22,11 +22,10 @@ import hashlib
|
||||||
from tempfile import gettempdir
|
from tempfile import gettempdir
|
||||||
from flask_babel import gettext as _
|
from flask_babel import gettext as _
|
||||||
|
|
||||||
from . import logger, comic
|
from . import logger, comic, isoLanguages, get_locale
|
||||||
from .constants import BookMeta
|
from .constants import BookMeta
|
||||||
from .helper import split_authors
|
from .helper import split_authors
|
||||||
|
|
||||||
|
|
||||||
log = logger.create()
|
log = logger.create()
|
||||||
|
|
||||||
|
|
||||||
|
@ -45,12 +44,17 @@ except (ImportError, RuntimeError) as e:
|
||||||
use_generic_pdf_cover = True
|
use_generic_pdf_cover = True
|
||||||
|
|
||||||
try:
|
try:
|
||||||
from PyPDF2 import PdfFileReader
|
from PyPDF3 import PdfFileReader
|
||||||
from PyPDF2 import __version__ as PyPdfVersion
|
from PyPDF3 import __version__ as PyPdfVersion
|
||||||
use_pdf_meta = True
|
use_pdf_meta = True
|
||||||
except ImportError as e:
|
except ImportError as ex:
|
||||||
log.debug('Cannot import PyPDF2, extracting pdf metadata will not work: %s', e)
|
try:
|
||||||
use_pdf_meta = False
|
from PyPDF2 import PdfFileReader
|
||||||
|
from PyPDF2 import __version__ as PyPdfVersion
|
||||||
|
use_pdf_meta = True
|
||||||
|
except ImportError as e:
|
||||||
|
log.debug('Cannot import PyPDF3/PyPDF2, extracting pdf metadata will not work: %s / %s', e)
|
||||||
|
use_pdf_meta = False
|
||||||
|
|
||||||
try:
|
try:
|
||||||
from . import epub
|
from . import epub
|
||||||
|
@ -98,39 +102,199 @@ def default_meta(tmp_file_path, original_file_name, original_file_extension):
|
||||||
extension=original_file_extension,
|
extension=original_file_extension,
|
||||||
title=original_file_name,
|
title=original_file_name,
|
||||||
author=_(u'Unknown'),
|
author=_(u'Unknown'),
|
||||||
cover=None,
|
cover=None, #pdf_preview(tmp_file_path, original_file_name),
|
||||||
description="",
|
description="",
|
||||||
tags="",
|
tags="",
|
||||||
series="",
|
series="",
|
||||||
series_id="",
|
series_id="",
|
||||||
languages="")
|
languages="",
|
||||||
|
publisher="")
|
||||||
|
|
||||||
|
|
||||||
|
def parse_xmp(pdf_file):
|
||||||
|
"""
|
||||||
|
Parse XMP Metadata and prepare for BookMeta object
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
xmp_info = pdf_file.getXmpMetadata()
|
||||||
|
except Exception as e:
|
||||||
|
log.debug('Can not read XMP metadata', e)
|
||||||
|
return None
|
||||||
|
|
||||||
|
if xmp_info:
|
||||||
|
try:
|
||||||
|
xmp_author = xmp_info.dc_creator # list
|
||||||
|
except AttributeError:
|
||||||
|
xmp_author = ['']
|
||||||
|
|
||||||
|
if xmp_info.dc_title:
|
||||||
|
xmp_title = xmp_info.dc_title['x-default']
|
||||||
|
else:
|
||||||
|
xmp_title = ''
|
||||||
|
|
||||||
|
if xmp_info.dc_description:
|
||||||
|
xmp_description = xmp_info.dc_description['x-default']
|
||||||
|
else:
|
||||||
|
xmp_description = ''
|
||||||
|
|
||||||
|
languages = []
|
||||||
|
try:
|
||||||
|
for i in xmp_info.dc_language:
|
||||||
|
#calibre-web currently only takes one language.
|
||||||
|
languages.append(isoLanguages.get_lang3(i))
|
||||||
|
except:
|
||||||
|
languages.append('')
|
||||||
|
|
||||||
|
xmp_tags = ', '.join(xmp_info.dc_subject)
|
||||||
|
xmp_publisher = ', '.join(xmp_info.dc_publisher)
|
||||||
|
|
||||||
|
return {'author': xmp_author,
|
||||||
|
'title': xmp_title,
|
||||||
|
'subject': xmp_description,
|
||||||
|
'tags': xmp_tags, 'languages': languages,
|
||||||
|
'publisher': xmp_publisher
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def parse_xmp(pdf_file):
|
||||||
|
"""
|
||||||
|
Parse XMP Metadata and prepare for BookMeta object
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
xmp_info = pdf_file.getXmpMetadata()
|
||||||
|
except Exception as e:
|
||||||
|
log.debug('Can not read XMP metadata', e)
|
||||||
|
return None
|
||||||
|
|
||||||
|
if xmp_info:
|
||||||
|
try:
|
||||||
|
xmp_author = xmp_info.dc_creator # list
|
||||||
|
except:
|
||||||
|
xmp_author = ['']
|
||||||
|
|
||||||
|
if xmp_info.dc_title:
|
||||||
|
xmp_title = xmp_info.dc_title['x-default']
|
||||||
|
else:
|
||||||
|
xmp_title = ''
|
||||||
|
|
||||||
|
if xmp_info.dc_description:
|
||||||
|
xmp_description = xmp_info.dc_description['x-default']
|
||||||
|
else:
|
||||||
|
xmp_description = ''
|
||||||
|
|
||||||
|
languages = []
|
||||||
|
try:
|
||||||
|
for i in xmp_info.dc_language:
|
||||||
|
languages.append(isoLanguages.get_lang3(i))
|
||||||
|
except AttributeError:
|
||||||
|
languages= [""]
|
||||||
|
|
||||||
|
xmp_tags = ', '.join(xmp_info.dc_subject)
|
||||||
|
xmp_publisher = ', '.join(xmp_info.dc_publisher)
|
||||||
|
|
||||||
|
return {'author': xmp_author,
|
||||||
|
'title': xmp_title,
|
||||||
|
'subject': xmp_description,
|
||||||
|
'tags': xmp_tags,
|
||||||
|
'languages': languages,
|
||||||
|
'publisher': xmp_publisher
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
def parse_xmp(pdf_file):
|
||||||
|
"""
|
||||||
|
Parse XMP Metadata and prepare for BookMeta object
|
||||||
|
"""
|
||||||
|
try:
|
||||||
|
xmp_info = pdf_file.getXmpMetadata()
|
||||||
|
except Exception as e:
|
||||||
|
log.debug('Can not read XMP metadata', e)
|
||||||
|
return None
|
||||||
|
|
||||||
|
if xmp_info:
|
||||||
|
try:
|
||||||
|
xmp_author = xmp_info.dc_creator # list
|
||||||
|
except AttributeError:
|
||||||
|
xmp_author = ['Unknown']
|
||||||
|
|
||||||
|
if xmp_info.dc_title:
|
||||||
|
xmp_title = xmp_info.dc_title['x-default']
|
||||||
|
else:
|
||||||
|
xmp_title = ''
|
||||||
|
|
||||||
|
if xmp_info.dc_description:
|
||||||
|
xmp_description = xmp_info.dc_description['x-default']
|
||||||
|
else:
|
||||||
|
xmp_description = ''
|
||||||
|
|
||||||
|
languages = []
|
||||||
|
try:
|
||||||
|
for i in xmp_info.dc_language:
|
||||||
|
languages.append(isoLanguages.get_lang3(i))
|
||||||
|
except AttributeError:
|
||||||
|
languages.append('')
|
||||||
|
|
||||||
|
xmp_tags = ', '.join(xmp_info.dc_subject)
|
||||||
|
xmp_publisher = ', '.join(xmp_info.dc_publisher)
|
||||||
|
|
||||||
|
return {'author': xmp_author,
|
||||||
|
'title': xmp_title,
|
||||||
|
'subject': xmp_description,
|
||||||
|
'tags': xmp_tags,
|
||||||
|
'languages': languages,
|
||||||
|
'publisher': xmp_publisher
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
def pdf_meta(tmp_file_path, original_file_name, original_file_extension):
|
def pdf_meta(tmp_file_path, original_file_name, original_file_extension):
|
||||||
doc_info = None
|
doc_info = None
|
||||||
|
xmp_info = None
|
||||||
|
|
||||||
if use_pdf_meta:
|
if use_pdf_meta:
|
||||||
with open(tmp_file_path, 'rb') as f:
|
with open(tmp_file_path, 'rb') as f:
|
||||||
doc_info = PdfFileReader(f).getDocumentInfo()
|
pdf_file = PdfFileReader(f)
|
||||||
if doc_info:
|
doc_info = pdf_file.getDocumentInfo()
|
||||||
author = doc_info.author if doc_info.author else u'Unknown'
|
xmp_info = parse_xmp(pdf_file)
|
||||||
title = doc_info.title if doc_info.title else original_file_name
|
|
||||||
subject = doc_info.subject
|
if xmp_info:
|
||||||
|
author = ' & '.join(split_authors(xmp_info['author']))
|
||||||
|
title = xmp_info['title']
|
||||||
|
subject = xmp_info['subject']
|
||||||
|
tags = xmp_info['tags']
|
||||||
|
languages = xmp_info['languages']
|
||||||
|
publisher = xmp_info['publisher']
|
||||||
else:
|
else:
|
||||||
author = u'Unknown'
|
author = u'Unknown'
|
||||||
title = original_file_name
|
title = ''
|
||||||
|
languages = [""]
|
||||||
|
publisher = ""
|
||||||
subject = ""
|
subject = ""
|
||||||
|
tags = ""
|
||||||
|
|
||||||
|
if doc_info:
|
||||||
|
if author == '':
|
||||||
|
author = ' & '.join(split_authors([doc_info.author])) if doc_info.author else u'Unknown'
|
||||||
|
if title == '':
|
||||||
|
title = doc_info.title if doc_info.title else original_file_name
|
||||||
|
if subject == '':
|
||||||
|
subject = doc_info.subject
|
||||||
|
if tags == '' and '/Keywords' in doc_info:
|
||||||
|
tags = doc_info['/Keywords']
|
||||||
|
else:
|
||||||
|
title = original_file_name
|
||||||
|
|
||||||
return BookMeta(
|
return BookMeta(
|
||||||
file_path=tmp_file_path,
|
file_path=tmp_file_path,
|
||||||
extension=original_file_extension,
|
extension=original_file_extension,
|
||||||
title=title,
|
title=title,
|
||||||
author=' & '.join(split_authors([author])),
|
author=author,
|
||||||
cover=pdf_preview(tmp_file_path, original_file_name),
|
cover=pdf_preview(tmp_file_path, original_file_name),
|
||||||
description=subject,
|
description=subject,
|
||||||
tags="",
|
tags=tags,
|
||||||
series="",
|
series="",
|
||||||
series_id="",
|
series_id="",
|
||||||
languages="")
|
languages=','.join(languages),
|
||||||
|
publisher=publisher)
|
||||||
|
|
||||||
|
|
||||||
def pdf_preview(tmp_file_path, tmp_dir):
|
def pdf_preview(tmp_file_path, tmp_dir):
|
||||||
|
|
|
@ -30,7 +30,7 @@ rarfile>=2.7
|
||||||
|
|
||||||
# other
|
# other
|
||||||
natsort>=2.2.0,<7.1.0
|
natsort>=2.2.0,<7.1.0
|
||||||
comicapi>= 2.1.3,<2.2.0
|
comicapi>= 2.2.0,<2.3.0
|
||||||
|
|
||||||
#Kobo integration
|
#Kobo integration
|
||||||
jsonschema>=3.2.0,<3.3.0
|
jsonschema>=3.2.0,<3.3.0
|
||||||
|
|
|
@ -6,7 +6,7 @@ singledispatch>=3.4.0.0,<3.5.0.0
|
||||||
backports_abc>=0.4
|
backports_abc>=0.4
|
||||||
Flask>=1.0.2,<1.2.0
|
Flask>=1.0.2,<1.2.0
|
||||||
iso-639>=0.4.5,<0.5.0
|
iso-639>=0.4.5,<0.5.0
|
||||||
PyPDF2>=1.26.0,<1.27.0
|
PyPDF3>=1.0.0,<1.0.4
|
||||||
pytz>=2016.10
|
pytz>=2016.10
|
||||||
requests>=2.11.1,<2.25.0
|
requests>=2.11.1,<2.25.0
|
||||||
SQLAlchemy>=1.3.0,<1.4.0
|
SQLAlchemy>=1.3.0,<1.4.0
|
||||||
|
|
14
setup.cfg
14
setup.cfg
|
@ -42,7 +42,7 @@ install_requires =
|
||||||
backports_abc>=0.4
|
backports_abc>=0.4
|
||||||
Flask>=1.0.2,<1.2.0
|
Flask>=1.0.2,<1.2.0
|
||||||
iso-639>=0.4.5,<0.5.0
|
iso-639>=0.4.5,<0.5.0
|
||||||
PyPDF2>=1.26.0,<1.27.0
|
PyPDF3>=1.0.0,<1.0.4
|
||||||
pytz>=2016.10
|
pytz>=2016.10
|
||||||
requests>=2.11.1,<2.25.0
|
requests>=2.11.1,<2.25.0
|
||||||
SQLAlchemy>=1.3.0,<1.4.0
|
SQLAlchemy>=1.3.0,<1.4.0
|
||||||
|
@ -52,9 +52,9 @@ install_requires =
|
||||||
|
|
||||||
[options.extras_require]
|
[options.extras_require]
|
||||||
gdrive =
|
gdrive =
|
||||||
google-api-python-client>=1.7.11,<1.8.0
|
google-api-python-client>=1.7.11,<1.13.0
|
||||||
gevent>=1.2.1,<20.6.0
|
gevent>20.6.0,<21.2.0
|
||||||
greenlet>=0.4.12,<0.4.17
|
greenlet>=0.4.17,<1.1.0
|
||||||
httplib2>=0.9.2,<0.18.0
|
httplib2>=0.9.2,<0.18.0
|
||||||
oauth2client>=4.0.0,<4.1.4
|
oauth2client>=4.0.0,<4.1.4
|
||||||
uritemplate>=3.0.0,<3.1.0
|
uritemplate>=3.0.0,<3.1.0
|
||||||
|
@ -68,16 +68,16 @@ goodreads =
|
||||||
goodreads>=0.3.2,<0.4.0
|
goodreads>=0.3.2,<0.4.0
|
||||||
python-Levenshtein>=0.12.0,<0.13.0
|
python-Levenshtein>=0.12.0,<0.13.0
|
||||||
ldap =
|
ldap =
|
||||||
python-ldap>=3.0.0,<3.3.0
|
python-ldap>=3.0.0,<3.4.0
|
||||||
Flask-SimpleLDAP>=1.4.0,<1.5.0
|
Flask-SimpleLDAP>=1.4.0,<1.5.0
|
||||||
oauth =
|
oauth =
|
||||||
Flask-Dance>=1.4.0,<3.1.0
|
Flask-Dance>=1.4.0,<3.1.0
|
||||||
SQLAlchemy-Utils>=0.33.5,<0.37.0
|
SQLAlchemy-Utils>=0.33.5,<0.37.0
|
||||||
metadata =
|
metadata =
|
||||||
lxml>=3.8.0,<4.6.0
|
lxml>=3.8.0,<4.7.0
|
||||||
rarfile>=2.7
|
rarfile>=2.7
|
||||||
comics =
|
comics =
|
||||||
natsort>=2.2.0
|
natsort>=2.2.0,<7.1.0
|
||||||
comicapi>= 2.1.3,<2.2.0
|
comicapi>= 2.1.3,<2.2.0
|
||||||
kobo =
|
kobo =
|
||||||
jsonschema>=3.2.0,<3.3.0
|
jsonschema>=3.2.0,<3.3.0
|
||||||
|
|
Loading…
Reference in New Issue
Block a user