# -*- coding: utf-8 -*- # Copyright 2016-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. """Extractors for FoOlSlide based sites""" from .common import ( Extractor, ChapterExtractor, MangaExtractor, SharedConfigMixin, Message, generate_extractors, ) from .. import text, util import base64 import json class FoolslideBase(SharedConfigMixin): """Base class for FoOlSlide extractors""" basecategory = "foolslide" def request(self, url): return Extractor.request( self, url, encoding="utf-8", method="POST", data={"adult": "true"}) @staticmethod def parse_chapter_url(url, data): info = url.partition("/read/")[2].rstrip("/").split("/") lang = info[1].partition("-")[0] data["lang"] = lang data["language"] = util.code_to_language(lang) data["volume"] = text.parse_int(info[2]) data["chapter"] = text.parse_int(info[3]) data["chapter_minor"] = "." + info[4] if len(info) >= 5 else "" data["title"] = data["chapter_string"].partition(":")[2].strip() return data class FoolslideChapterExtractor(FoolslideBase, ChapterExtractor): """Base class for chapter extractors for FoOlSlide based sites""" directory_fmt = ("{category}", "{manga}", "{chapter_string}") archive_fmt = "{id}" pattern_fmt = r"(/read/[^/?&#]+/[a-z-]+/\d+/\d+(?:/\d+)?)" decode = "default" def items(self): page = self.request(self.gallery_url).text data = self.metadata(page) imgs = self.images(page) data["count"] = len(imgs) data["chapter_id"] = text.parse_int(imgs[0]["chapter_id"]) yield Message.Version, 1 yield Message.Directory, data for data["page"], image in enumerate(imgs, 1): try: url = image["url"] del image["url"] del image["chapter_id"] del image["thumb_url"] except KeyError: pass for key in ("height", "id", "size", "width"): image[key] = text.parse_int(image[key]) data.update(image) text.nameext_from_url(data["filename"], data) yield Message.Url, url, data def metadata(self, page): extr = text.extract_from(page) extr('

', '') return self.parse_chapter_url(self.gallery_url, { "manga" : text.unescape(extr('title="', '"')).strip(), "chapter_string": text.unescape(extr('title="', '"')), }) def images(self, page): data = None if self.decode == "base64": base64_data = text.extract(page, 'atob("', '"')[0] if base64_data: data = base64.b64decode(base64_data.encode()).decode() elif self.decode == "double": pos = page.find("[{") if pos >= 0: data = text.extract(page, " = ", ";", pos)[0] if not data: data = text.extract(page, "var pages = ", ";")[0] return json.loads(data) class FoolslideMangaExtractor(FoolslideBase, MangaExtractor): """Base class for manga extractors for FoOlSlide based sites""" pattern_fmt = r"(/series/[^/?&#]+)" def chapters(self, page): extr = text.extract_from(page) manga = text.unescape(extr('

', '

')).strip() author = extr('Author: ', 'Artist: ', '= 1", "keyword": { "chapter": int, "chapter_minor": str, "chapter_string": str, "group": "PowerManga", "lang": "en", "language": "English", "manga": "One Piece Digital Colour Comics", "title": str, "volume": int, }, }), }, "sensescans": { "root": "http://sensescans.com/reader", "pattern": r"(?:(?:www\.)?sensescans\.com/reader" r"|reader\.sensescans\.com)", "test-chapter": ( (("http://sensescans.com/reader/read/" "magi__labyrinth_of_magic/en/37/369/"), { "url": "a399ef037cdfbc25b09d435cc2ea1e3e454a6812", "keyword": "07acd84fb18a9f1fd6dff5befe711bcca0ff9988", }), (("http://reader.sensescans.com/read/" "magi__labyrinth_of_magic/en/37/369/"), { "url": "a399ef037cdfbc25b09d435cc2ea1e3e454a6812", "keyword": "07acd84fb18a9f1fd6dff5befe711bcca0ff9988", }), ), "test-manga": ("http://sensescans.com/reader/series/hakkenden/", { "url": "2360ccb0ead0ff2f5e27b7aef7eb17b9329de2f2", "keyword": "4919f2bfed38e3a34dc984ec8d1dbd7a03044e23", }), }, "worldthree": { "root": "http://www.slide.world-three.org", "pattern": r"(?:www\.)?slide\.world-three\.org", "test-chapter": ( (("http://www.slide.world-three.org" "/read/black_bullet/en/2/7/page/1"), { "url": "be2f04f6e2d311b35188094cfd3e768583271584", "keyword": "967d536a65de4d52478d5b666a1760b181eddb6e", }), (("http://www.slide.world-three.org" "/read/idolmster_cg_shuffle/en/0/4/2/"), { "url": "6028ea5ca282744f925dfad92eeb98509f9cc78c", "keyword": "f3cfe2ad3388991f1d045c85d0fa94795a7694dc", }), ), "test-manga": ("http://www.slide.world-three.org/series/black_bullet/", { "url": "5743b93512d26e6b540d90a7a5d69208b6d4a738", "keyword": "3a24f1088b4d7f3b798a96163f21ca251293a120", }), }, "_ckey": "chapterclass", } generate_extractors(EXTRACTORS, globals(), ( FoolslideChapterExtractor, FoolslideMangaExtractor, ))