gallery-dl/gallery_dl/extractor/sankakucomplex.py

142 lines
5.0 KiB
Python
Raw Normal View History

# -*- coding: utf-8 -*-
# Copyright 2019-2020 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.
"""Extractors for https://www.sankakucomplex.com/"""
from .common import Extractor, Message
from .. import text, util
import re
class SankakucomplexExtractor(Extractor):
"""Base class for sankakucomplex extractors"""
category = "sankakucomplex"
root = "https://www.sankakucomplex.com"
def __init__(self, match):
Extractor.__init__(self, match)
self.path = match.group(1)
class SankakucomplexArticleExtractor(SankakucomplexExtractor):
"""Extractor for articles on www.sankakucomplex.com"""
subcategory = "article"
directory_fmt = ("{category}", "{date:%Y-%m-%d} {title}")
filename_fmt = "{filename}.{extension}"
archive_fmt = "{date:%Y%m%d}_{filename}"
pattern = (r"(?:https?://)?www\.sankakucomplex\.com"
r"/(\d{4}/\d\d/\d\d/[^/?#]+)")
test = (
("https://www.sankakucomplex.com/2019/05/11/twitter-cosplayers", {
"url": "4a9ecc5ae917fbce469280da5b6a482510cae84d",
2019-08-01 21:41:16 +02:00
"keyword": "bfe08310e7d9a572f568f6900e0ed0eb295aa2b3",
}),
("https://www.sankakucomplex.com/2009/12/01/sexy-goddesses-of-2ch", {
"url": "a1e249173fd6c899a8134fcfbd9c925588a63f7c",
2019-08-01 21:41:16 +02:00
"keyword": "e78fcc23c2711befc0969a45ea5082a29efccf68",
}),
# videos (#308)
(("https://www.sankakucomplex.com/2019/06/11"
"/darling-ol-goddess-shows-off-her-plump-lower-area/"), {
"pattern": r"/wp-content/uploads/2019/06/[^/]+\d\.mp4",
"range": "26-",
"count": 5,
}),
# youtube embeds (#308)
(("https://www.sankakucomplex.com/2015/02/12"
"/snow-miku-2015-live-magical-indeed/"), {
"options": (("embeds", True),),
"pattern": r"https://www.youtube.com/embed/",
"range": "2-",
"count": 2,
}),
)
def items(self):
url = "{}/{}/?pg=X".format(self.root, self.path)
extr = text.extract_from(self.request(url).text)
data = {
"title" : text.unescape(
extr('property="og:title" content="', '"')),
"description": text.unescape(
extr('property="og:description" content="', '"')),
"date" : text.parse_datetime(
extr('property="article:published_time" content="', '"')),
}
content = extr('<div class="entry-content">', '</article>')
data["tags"] = text.split_html(extr('="meta-tags">', '</div>'))[::2]
files = self._extract_images(content)
if self.config("videos", True):
files += self._extract_videos(content)
if self.config("embeds", False):
files += self._extract_embeds(content)
data["count"] = len(files)
yield Message.Directory, data
for num, url in enumerate(files, 1):
file = text.nameext_from_url(url)
if url[0] == "/":
url = text.urljoin(self.root, url)
file["url"] = url
file["num"] = num
file.update(data)
yield Message.Url, url, file
@staticmethod
def _extract_images(content):
orig_sub = re.compile(r"-\d+x\d+\.").sub
return [
orig_sub(".", url) for url in
util.unique(text.extract_iter(content, 'data-lazy-src="', '"'))
]
@staticmethod
def _extract_videos(content):
return re.findall(r"<source [^>]*src=[\"']([^\"']+)", content)
@staticmethod
def _extract_embeds(content):
return [
"ytdl:" + url for url in
re.findall(r"<iframe [^>]*src=[\"']([^\"']+)", content)
]
class SankakucomplexTagExtractor(SankakucomplexExtractor):
"""Extractor for sankakucomplex blog articles by tag or author"""
subcategory = "tag"
pattern = (r"(?:https?://)?www\.sankakucomplex\.com"
r"/((?:tag|category|author)/[^/&?#]+)")
test = (
("https://www.sankakucomplex.com/tag/cosplay/", {
"range": "1-50",
"count": 50,
"pattern": SankakucomplexArticleExtractor.pattern,
}),
("https://www.sankakucomplex.com/category/anime/"),
("https://www.sankakucomplex.com/author/rift/page/5/"),
)
def items(self):
pnum = 1
last = None
data = {"_extractor": SankakucomplexArticleExtractor}
yield Message.Version, 1
while True:
url = "{}/{}/page/{}/".format(self.root, self.path, pnum)
response = self.request(url, fatal=False)
if response.status_code >= 400:
return
for url in text.extract_iter(response.text, 'data-direct="', '"'):
if url != last:
last = url
yield Message.Queue, url, data
pnum += 1