6284731107
- single strings for URL patterns - tuples instead of lists for 'directory_fmt' and 'test' - single-tuple tests where applicable
167 lines
3.4 KiB
Python
167 lines
3.4 KiB
Python
# -*- coding: utf-8 -*-
|
|
|
|
# Copyright 2015-2019 Mike Fährmann
|
|
#
|
|
# 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 = [
|
|
"2chan",
|
|
"3dbooru",
|
|
"4chan",
|
|
"8chan",
|
|
"artstation",
|
|
"behance",
|
|
"bobx",
|
|
"danbooru",
|
|
"deviantart",
|
|
"dynastyscans",
|
|
"e621",
|
|
"exhentai",
|
|
"fallenangels",
|
|
"flickr",
|
|
"gelbooru",
|
|
"gfycat",
|
|
"hbrowse",
|
|
"hentai2read",
|
|
"hentaicafe",
|
|
"hentaifoundry",
|
|
"hentaifox",
|
|
"hentaihere",
|
|
"hitomi",
|
|
"idolcomplex",
|
|
"imagebam",
|
|
"imagefap",
|
|
"imgbox",
|
|
"imgth",
|
|
"imgur",
|
|
"instagram",
|
|
"khinsider",
|
|
"kissmanga",
|
|
"komikcast",
|
|
"konachan",
|
|
"luscious",
|
|
"mangadex",
|
|
"mangafox",
|
|
"mangahere",
|
|
"mangapanda",
|
|
"mangapark",
|
|
"mangareader",
|
|
"mangastream",
|
|
"myportfolio",
|
|
"newgrounds",
|
|
"ngomik",
|
|
"nhentai",
|
|
"nijie",
|
|
"paheal",
|
|
"photobucket",
|
|
"piczel",
|
|
"pinterest",
|
|
"pixiv",
|
|
"reactor",
|
|
"readcomiconline",
|
|
"reddit",
|
|
"rule34",
|
|
"safebooru",
|
|
"sankaku",
|
|
"seiga",
|
|
"senmanga",
|
|
"simplyhentai",
|
|
"slideshare",
|
|
"smugmug",
|
|
"tsumino",
|
|
"tumblr",
|
|
"twitter",
|
|
"wallhaven",
|
|
"warosu",
|
|
"yandere",
|
|
"xvideos",
|
|
"yuki",
|
|
"foolfuuka",
|
|
"foolslide",
|
|
"mastodon",
|
|
"imagehosts",
|
|
"directlink",
|
|
"recursive",
|
|
"oauth",
|
|
"test",
|
|
]
|
|
|
|
|
|
def find(url):
|
|
"""Find suitable extractor for the given url"""
|
|
for pattern, klass in _list_patterns():
|
|
match = pattern.match(url)
|
|
if match and klass not in _blacklist:
|
|
return klass(match)
|
|
return None
|
|
|
|
|
|
def add(klass):
|
|
"""Add 'klass' to the list of available extractors"""
|
|
_cache.append((re.compile(klass.pattern), klass))
|
|
|
|
|
|
def add_module(module):
|
|
"""Add all extractors in 'module' to the list of available extractors"""
|
|
tuples = [
|
|
(re.compile(klass.pattern), klass)
|
|
for klass in _get_classes(module)
|
|
]
|
|
_cache.extend(tuples)
|
|
return tuples
|
|
|
|
|
|
def extractors():
|
|
"""Yield all available extractor classes"""
|
|
return sorted(
|
|
set(klass for _, klass in _list_patterns()),
|
|
key=lambda x: x.__name__
|
|
)
|
|
|
|
|
|
class blacklist():
|
|
"""Context Manager to blacklist extractor modules"""
|
|
def __init__(self, categories, extractors=None):
|
|
self.extractors = extractors or []
|
|
for _, klass in _list_patterns():
|
|
if klass.category in categories:
|
|
self.extractors.append(klass)
|
|
|
|
def __enter__(self):
|
|
_blacklist.update(self.extractors)
|
|
|
|
def __exit__(self, etype, value, traceback):
|
|
_blacklist.clear()
|
|
|
|
|
|
# --------------------------------------------------------------------
|
|
# internals
|
|
|
|
_cache = []
|
|
_blacklist = set()
|
|
_module_iter = iter(modules)
|
|
|
|
|
|
def _list_patterns():
|
|
"""Yield all available (pattern, class) tuples"""
|
|
yield from _cache
|
|
|
|
for module_name in _module_iter:
|
|
yield from add_module(
|
|
importlib.import_module("."+module_name, __package__)
|
|
)
|
|
|
|
|
|
def _get_classes(module):
|
|
"""Return a list of all extractor classes in a module"""
|
|
return [
|
|
klass for klass in module.__dict__.values() if (
|
|
hasattr(klass, "pattern") and klass.__module__ == module.__name__
|
|
)
|
|
]
|