2015-06-28 22:53:52 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
2014-10-12 21:56:44 +02:00
|
|
|
|
2019-01-19 14:28:59 +01:00
|
|
|
# Copyright 2015-2019 Mike Fährmann
|
2015-06-28 22:53:52 +02:00
|
|
|
#
|
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License version 2 as
|
|
|
|
# published by the Free Software Foundation.
|
|
|
|
|
|
|
|
import re
|
|
|
|
import importlib
|
|
|
|
|
|
|
|
modules = [
|
2017-07-14 08:44:31 +02:00
|
|
|
"2chan",
|
2019-03-18 01:11:30 +01:00
|
|
|
"35photo",
|
2015-06-28 22:53:52 +02:00
|
|
|
"3dbooru",
|
|
|
|
"4chan",
|
2019-03-19 23:40:46 +01:00
|
|
|
"500px",
|
2015-06-28 22:53:52 +02:00
|
|
|
"8chan",
|
2018-03-14 14:03:53 +01:00
|
|
|
"artstation",
|
2018-08-01 21:46:55 +02:00
|
|
|
"behance",
|
2018-09-13 20:13:12 +02:00
|
|
|
"bobx",
|
2015-06-28 22:53:52 +02:00
|
|
|
"danbooru",
|
2015-10-05 20:29:48 +02:00
|
|
|
"deviantart",
|
2016-09-22 17:20:57 +02:00
|
|
|
"dynastyscans",
|
2015-06-28 22:53:52 +02:00
|
|
|
"e621",
|
2016-08-04 18:08:48 +02:00
|
|
|
"exhentai",
|
2017-02-06 20:05:58 +01:00
|
|
|
"fallenangels",
|
2017-05-30 17:43:02 +02:00
|
|
|
"flickr",
|
2016-08-04 18:08:48 +02:00
|
|
|
"gelbooru",
|
2017-05-28 17:09:54 +02:00
|
|
|
"gfycat",
|
2015-11-15 01:30:26 +01:00
|
|
|
"hbrowse",
|
2016-02-19 15:24:49 +01:00
|
|
|
"hentai2read",
|
2019-02-05 10:22:52 +01:00
|
|
|
"hentaicafe",
|
2015-11-14 03:19:44 +01:00
|
|
|
"hentaifoundry",
|
2019-01-28 18:00:32 +01:00
|
|
|
"hentaifox",
|
2016-10-05 09:20:03 +02:00
|
|
|
"hentaihere",
|
2015-10-28 16:24:35 +01:00
|
|
|
"hitomi",
|
2019-03-22 13:46:44 +01:00
|
|
|
"hypnohub",
|
2018-01-09 17:52:12 +01:00
|
|
|
"idolcomplex",
|
2015-06-28 22:53:52 +02:00
|
|
|
"imagebam",
|
2016-08-09 14:05:12 +02:00
|
|
|
"imagefap",
|
2015-06-28 22:53:52 +02:00
|
|
|
"imgbox",
|
2015-10-28 23:26:47 +01:00
|
|
|
"imgth",
|
2015-10-12 22:34:45 +02:00
|
|
|
"imgur",
|
2018-12-09 12:52:14 +01:00
|
|
|
"instagram",
|
2016-04-20 08:34:44 +02:00
|
|
|
"khinsider",
|
2017-04-05 12:16:23 +02:00
|
|
|
"kissmanga",
|
2018-02-04 16:27:44 +01:00
|
|
|
"komikcast",
|
2015-11-06 13:24:43 +01:00
|
|
|
"konachan",
|
2016-08-01 15:36:56 +02:00
|
|
|
"luscious",
|
2018-03-05 18:37:21 +01:00
|
|
|
"mangadex",
|
2017-01-14 19:39:21 +01:00
|
|
|
"mangafox",
|
2015-11-26 03:06:08 +01:00
|
|
|
"mangahere",
|
2015-11-08 00:02:37 +01:00
|
|
|
"mangapanda",
|
2015-12-08 22:29:34 +01:00
|
|
|
"mangapark",
|
2015-06-28 22:53:52 +02:00
|
|
|
"mangareader",
|
2015-11-08 00:03:14 +01:00
|
|
|
"mangastream",
|
2018-07-19 18:56:45 +02:00
|
|
|
"myportfolio",
|
2018-11-27 15:44:53 +01:00
|
|
|
"newgrounds",
|
2018-08-23 15:29:53 +02:00
|
|
|
"ngomik",
|
2015-10-28 12:08:27 +01:00
|
|
|
"nhentai",
|
2015-06-28 22:53:52 +02:00
|
|
|
"nijie",
|
2018-01-15 16:39:05 +01:00
|
|
|
"paheal",
|
2019-01-20 16:19:13 +01:00
|
|
|
"photobucket",
|
2018-08-08 10:53:01 +02:00
|
|
|
"piczel",
|
2016-09-02 19:11:16 +02:00
|
|
|
"pinterest",
|
2017-10-17 16:49:42 +02:00
|
|
|
"pixiv",
|
2019-02-25 12:37:20 +01:00
|
|
|
"pururin",
|
2019-01-07 16:59:26 +01:00
|
|
|
"reactor",
|
2016-11-14 18:29:45 +01:00
|
|
|
"readcomiconline",
|
2017-05-23 09:38:50 +02:00
|
|
|
"reddit",
|
2016-09-17 18:12:37 +02:00
|
|
|
"rule34",
|
2015-11-06 13:52:40 +01:00
|
|
|
"safebooru",
|
2015-11-09 02:29:33 +01:00
|
|
|
"sankaku",
|
2016-08-09 16:36:30 +02:00
|
|
|
"seiga",
|
2016-08-02 17:42:22 +02:00
|
|
|
"senmanga",
|
2018-05-27 15:25:04 +02:00
|
|
|
"simplyhentai",
|
2017-12-13 17:38:29 +01:00
|
|
|
"slideshare",
|
2018-04-29 21:27:25 +02:00
|
|
|
"smugmug",
|
2019-01-29 17:23:01 +01:00
|
|
|
"tsumino",
|
2016-02-20 11:29:10 +01:00
|
|
|
"tumblr",
|
2016-10-06 19:12:07 +02:00
|
|
|
"twitter",
|
2018-09-17 21:19:25 +02:00
|
|
|
"wallhaven",
|
2017-08-18 19:52:58 +02:00
|
|
|
"warosu",
|
2019-02-16 22:56:04 +01:00
|
|
|
"weibo",
|
2015-06-28 22:53:52 +02:00
|
|
|
"yandere",
|
2017-11-02 15:36:53 +01:00
|
|
|
"xvideos",
|
2018-09-28 12:46:39 +02:00
|
|
|
"yuki",
|
2019-02-03 00:40:12 +01:00
|
|
|
"foolfuuka",
|
2019-02-03 23:54:17 +01:00
|
|
|
"foolslide",
|
2019-01-19 14:28:59 +01:00
|
|
|
"mastodon",
|
2019-03-05 22:33:37 +01:00
|
|
|
"shopify",
|
2016-11-03 15:46:04 +01:00
|
|
|
"imagehosts",
|
2017-05-24 12:51:18 +02:00
|
|
|
"directlink",
|
2016-10-01 15:54:27 +02:00
|
|
|
"recursive",
|
2017-06-20 16:06:14 +02:00
|
|
|
"oauth",
|
2016-12-10 00:01:00 +01:00
|
|
|
"test",
|
2015-06-28 22:53:52 +02:00
|
|
|
]
|
|
|
|
|
2017-02-01 00:53:19 +01:00
|
|
|
|
2015-10-05 17:52:50 +02:00
|
|
|
def find(url):
|
2019-02-08 20:08:16 +01:00
|
|
|
"""Find a suitable extractor for the given URL"""
|
|
|
|
for cls in _list_classes():
|
|
|
|
match = cls.pattern.match(url)
|
|
|
|
if match and cls not in _blacklist:
|
|
|
|
return cls(match)
|
2015-11-21 00:30:31 +01:00
|
|
|
return None
|
2015-06-28 22:53:52 +02:00
|
|
|
|
2017-02-01 00:53:19 +01:00
|
|
|
|
2019-02-08 20:08:16 +01:00
|
|
|
def add(cls):
|
|
|
|
"""Add 'cls' to the list of available extractors"""
|
|
|
|
cls.pattern = re.compile(cls.pattern)
|
|
|
|
_cache.append(cls)
|
|
|
|
return cls
|
2018-02-02 00:01:41 +01:00
|
|
|
|
|
|
|
|
|
|
|
def add_module(module):
|
|
|
|
"""Add all extractors in 'module' to the list of available extractors"""
|
2019-02-08 20:08:16 +01:00
|
|
|
classes = _get_classes(module)
|
|
|
|
for cls in classes:
|
|
|
|
cls.pattern = re.compile(cls.pattern)
|
|
|
|
_cache.extend(classes)
|
|
|
|
return classes
|
2018-02-02 00:01:41 +01:00
|
|
|
|
|
|
|
|
2015-12-12 15:58:07 +01:00
|
|
|
def extractors():
|
|
|
|
"""Yield all available extractor classes"""
|
2015-12-13 04:34:15 +01:00
|
|
|
return sorted(
|
2019-02-08 20:08:16 +01:00
|
|
|
_list_classes(),
|
2015-12-13 04:34:15 +01:00
|
|
|
key=lambda x: x.__name__
|
|
|
|
)
|
2015-12-12 15:58:07 +01:00
|
|
|
|
2017-02-01 00:53:19 +01:00
|
|
|
|
2017-05-24 12:32:44 +02:00
|
|
|
class blacklist():
|
|
|
|
"""Context Manager to blacklist extractor modules"""
|
2018-01-14 18:47:22 +01:00
|
|
|
def __init__(self, categories, extractors=None):
|
|
|
|
self.extractors = extractors or []
|
2019-02-08 20:08:16 +01:00
|
|
|
for cls in _list_classes():
|
|
|
|
if cls.category in categories:
|
|
|
|
self.extractors.append(cls)
|
2017-05-24 12:32:44 +02:00
|
|
|
|
|
|
|
def __enter__(self):
|
2018-01-14 18:47:22 +01:00
|
|
|
_blacklist.update(self.extractors)
|
2017-05-24 12:32:44 +02:00
|
|
|
|
|
|
|
def __exit__(self, etype, value, traceback):
|
|
|
|
_blacklist.clear()
|
|
|
|
|
|
|
|
|
2015-06-28 22:53:52 +02:00
|
|
|
# --------------------------------------------------------------------
|
|
|
|
# internals
|
|
|
|
|
|
|
|
_cache = []
|
2018-01-14 18:47:22 +01:00
|
|
|
_blacklist = set()
|
2015-06-28 22:53:52 +02:00
|
|
|
_module_iter = iter(modules)
|
|
|
|
|
2017-02-01 00:53:19 +01:00
|
|
|
|
2019-02-08 20:08:16 +01:00
|
|
|
def _list_classes():
|
|
|
|
"""Yield all available extractor classes"""
|
2016-08-23 16:36:39 +02:00
|
|
|
yield from _cache
|
2015-10-05 18:10:18 +02:00
|
|
|
|
2015-06-28 22:53:52 +02:00
|
|
|
for module_name in _module_iter:
|
2019-02-08 20:08:16 +01:00
|
|
|
module = importlib.import_module("."+module_name, __package__)
|
|
|
|
yield from add_module(module)
|
2015-11-20 19:54:07 +01:00
|
|
|
|
2017-02-01 00:53:19 +01:00
|
|
|
|
2015-11-20 19:54:07 +01:00
|
|
|
def _get_classes(module):
|
|
|
|
"""Return a list of all extractor classes in a module"""
|
|
|
|
return [
|
2019-02-08 20:08:16 +01:00
|
|
|
cls for cls in module.__dict__.values() if (
|
|
|
|
hasattr(cls, "pattern") and cls.__module__ == module.__name__
|
2015-11-20 19:54:07 +01:00
|
|
|
)
|
|
|
|
]
|