dosage/dosagelib/plugins/t.py
2019-12-27 22:20:02 +01:00

353 lines
12 KiB
Python

# -*- coding: utf-8 -*-
# Copyright (C) 2004-2008 Tristan Seligmann and Jonathan Jacobs
# Copyright (C) 2012-2014 Bastian Kleineidam
# Copyright (C) 2015-2017 Tobias Gruetzmacher
from __future__ import absolute_import, division, print_function
from re import compile, escape, MULTILINE
try:
from functools import cached_property
except ImportError:
from cached_property import cached_property
from ..scraper import _BasicScraper, _ParserScraper
from ..helpers import indirectStarter, xpath_class
from ..util import tagre
from .common import _ComicControlScraper, _TumblrScraper, _WordPressScraper, _WPNavi, _WPNaviIn
class TalesOfTheQuestor(_ParserScraper):
stripUrl = 'http://www.rhjunior.com/%s/'
firstStripUrl = stripUrl % 'tales-of-the-questor-cover'
url = stripUrl % 'comics/tales-of-the-questor'
imageSearch = '//div[contains(@class, "entry-content")]//img'
prevSearch = ('//a[@rel="prev"]', '//a[@title="Tales of the Questor"]')
class Tamberlane(_ParserScraper):
baseUrl = 'https://www.tamberlanecomic.com/'
url = baseUrl + 'latest/'
stripUrl = baseUrl + 'tamberlane/%s/'
firstStripUrl = stripUrl % 'page-1'
imageSearch = '//div[@class="webcomic-image"]//img'
prevSearch = '//a[contains(@class, "previous-webcomic-link")]'
def namer(self, imageUrl, pageUrl):
# Fix inconsistent filenames
filename = imageUrl.rsplit('/', 1)[-1]
return filename.replace('ai4zCWaA', 'Page_152')
class TheBrads(_ParserScraper):
url = 'http://bradcolbow.com/archive/'
imageSearch = '//div[%s]//img' % xpath_class('entry')
prevSearch = '//a[%s]' % xpath_class('prev')
multipleImagesPerStrip = True
class TheClassMenagerie(_ParserScraper):
stripUrl = 'http://www.theclassm.com/d/%s.html'
url = stripUrl % '20050717'
firstStripUrl = stripUrl % '19990322'
imageSearch = '//img[@class="ksc"]'
prevSearch = '//a[@rel="prev"]'
multipleImagesPerStrip = True
endOfLife = True
class TheDevilsPanties(_WPNavi):
url = 'http://thedevilspanties.com/'
stripUrl = url + 'archives/%s'
firstStripUrl = stripUrl % '300'
help = 'Index format: number'
class TheDreamlandChronicles(_WordPressScraper):
url = 'http://www.thedreamlandchronicles.com/'
class TheGamerCat(_ParserScraper):
url = 'https://thegamercat.com/'
stripUrl = url + 'comic/%s/'
firstStripUrl = stripUrl % '06102011'
imageSearch = '//div[@id="comic"]//img'
prevSearch = '//a[contains(@class, "comic-nav-previous")]'
help = 'Index format: stripname'
class TheGentlemansArmchair(_WordPressScraper):
url = 'http://thegentlemansarmchair.com/'
class TheGentleWolf(_WordPressScraper):
url = 'https://thegentlewolf.net/'
stripUrl = url + 'comic/%s/'
firstStripUrl = stripUrl % 'tgw-001'
def namer(self, imageUrl, pageUrl):
# Fix duplicate filename
filename = imageUrl.rsplit('/', 1)[-1]
if pageUrl == self.stripUrl % 'tgw-271':
filename = filename.replace('272', '271')
return filename
class TheLandscaper(_BasicScraper):
stripUrl = 'http://landscaper.visual-assault.net/comic/%s'
url = stripUrl % 'latest'
firstStripUrl = stripUrl % '1'
imageSearch = compile(tagre("img", "src",
r'(/comics/comic/comic_page/[^"]+)'))
prevSearch = compile(tagre("a", "href", r'(/comic/[^"]+)') +
'‹ Previous')
help = 'Index format: name'
class TheMelvinChronicles(_WordPressScraper):
url = 'http://melvin.jeaniebottle.com/'
class TheNoob(_WordPressScraper):
url = 'http://thenoobcomic.com/'
stripUrl = url + 'comic/%s/'
firstStripUrl = stripUrl % '1'
help = 'Index format: n (unpadded)'
class TheOldVictorian(_ParserScraper):
url = 'http://theoldvictorianwebcomic.com/'
stripUrl = url + 'comic/%s/'
firstStripUrl = stripUrl % 'the-old-victorian-cover'
imageSearch = '//div[@id="comic"]//img'
prevSearch = '//a[contains(@class, "comic-nav-previous")]'
def namer(self, imageUrl, pageUrl):
filename = imageUrl.rsplit('/', 1)[-1].replace('_', '-')
filename = filename.replace('TOV00', 'TOV-00')
if filename.replace('oldvic', '')[0].isdigit():
filename = filename.replace('oldvic', 'TOV-00')
if 'TOV-000' in filename and len(filename) > 12:
filename = filename[:8] + '-' + filename[8:]
return filename
class TheOrderOfTheStick(_BasicScraper):
url = 'http://www.giantitp.com/'
stripUrl = url + 'comics/oots%s.html'
firstStripUrl = stripUrl % '0001'
imageSearch = compile(r'<IMG src="(/comics/images/[^"]+)">')
prevSearch = compile(r'<A href="(/comics/oots\d{4}\.html)"><IMG src="/Images/redesign/ComicNav_Back.gif"')
latestSearch = compile(r'<A href="(/comics/oots\d{4}\.html)"')
help = 'Index format: n (unpadded)'
starter = indirectStarter
def namer(self, image_url, page_url):
return page_url.rsplit('/', 1)[-1][:-5]
class TheProbabilityBomb(_ParserScraper):
stripUrl = 'http://www.rhjunior.com/%s/'
firstStripUrl = stripUrl % 'the-probability-bomb'
url = stripUrl % 'comics/the-probability-bomb'
imageSearch = '//div[contains(@class, "entry-content")]//img'
prevSearch = ('//a[@rel="prev"]', '//a[contains(@title, "Probability Bomb")]')
multipleImagesPerStrip = True
class TheRockCocks(_BasicScraper):
url = 'http://rockcocks.slipshine.net/'
rurl = escape(url)
stripUrl = url + 'comics/%s'
firstStripUrl = stripUrl % "page-1-nsfw-track-1-start"
imageSearch = compile(tagre("img", "src", r'(%scomics/[^"]+)' % rurl, after='id="cc-comic"'))
prevSearch = compile(tagre("a", "href", r'(%scomic/[^"]+)' % rurl, after='rel="prev"'))
adult = True
class TheThinHLine(_TumblrScraper):
url = 'http://thinhline.tumblr.com/'
firstStripUrl = url + 'post/4177372348/thl-1-a-cats-got-his-tongue-click-on-the'
imageSearch = '//img[@id="content-image"]/@data-src'
prevSearch = '//div[@id="pagination"]/a[text()=">"]'
latestSearch = '//a[@class="timestamp"]'
adult = True
indirectImageSearch = '//div[@id="post"]//a[not(@rel) and img]'
def getComicStrip(self, url, data):
"""The comic strip image is in a separate page."""
subPage = self.fetchUrl(url, data, self.indirectImageSearch)
pageData = self.getPage(subPage)
return super(TheThinHLine, self).getComicStrip(subPage, pageData)
class TheWhiteboard(_ParserScraper):
BROKEN_PAGE_MIDDLE = compile(r'</body></html>\n<')
url = 'http://www.the-whiteboard.com/'
stripUrl = url + 'auto%s.html'
firstStripUrl = stripUrl % 'wb001'
imageSearch = '//img[contains(@src, "auto")]'
prevSearch = '//a[.//img[contains(@src, "previous")]]'
def _parse_page(self, data):
# Ugly hack to fix broken HTML
data = self.BROKEN_PAGE_MIDDLE.sub('<', data)
return super(TheWhiteboard, self)._parse_page(data)
class TheWotch(_WordPressScraper):
url = 'http://www.thewotch.com/'
firstStripUrl = url + '?comic=enter-the-wotch'
class ThisIsIndexed(_BasicScraper):
url = 'http://thisisindexed.com/'
rurl = escape(url)
stripUrl = url + 'page/%s'
imageSearch = compile(tagre("img", "src", r'(%swp-content/uploads/\d+/\d+/card[^"]+)' % rurl))
multipleImagesPerStrip = True
prevSearch = compile(tagre("div", "class", "nav-previous") +
tagre("a", "href", r'(%spage/\d+/)[^"]*' % rurl))
help = 'Index format: number'
class ThreePanelSoul(_ComicControlScraper):
url = 'http://threepanelsoul.com/'
firstStripUrl = url + 'comic/a-test-comic'
class TinyDickAdventures(_ParserScraper):
url = 'https://www.lfg.co/'
stripUrl = url + 'tda/strip/%s/'
firstStripUrl = stripUrl % '1'
imageSearch = '//div[@id="comic-img"]//img'
prevSearch = '//a[@class="comic-nav-prev"]'
latestSearch = '//div[@id="feature-tda-footer"]/a[contains(@href, "tda/strip/")]'
starter = indirectStarter
def namer(self, imageUrl, pageUrl):
page = pageUrl.rstrip('/').rsplit('/', 1)[-1]
ext = imageUrl.rsplit('.', 1)[-1]
return page + '.' + ext
class ToonHole(_WordPressScraper):
url = 'http://toonhole.com/'
firstStripUrl = url + 'comic/toon-hole-coming-soon-2010/'
def shouldSkipUrl(self, url, data):
return url in (self.url + "comic/if-game-of-thrones-was-animated/",)
class TracesOfThePast(_WPNaviIn):
baseUrl = 'http://rickgriffinstudios.com/'
url = baseUrl + 'in-the-new-age/'
stripUrl = baseUrl + 'comic-post/%s/'
firstStripUrl = stripUrl % 'totp-page-1'
latestSearch = '//a[contains(@title, "Permanent Link")]'
starter = indirectStarter
class TracesOfThePastNSFW(_WPNaviIn):
name = 'TracesOfThePast/NSFW'
baseUrl = 'http://rickgriffinstudios.com/'
url = baseUrl + 'in-the-new-age/'
stripUrl = baseUrl + 'comic-post/%s/'
firstStripUrl = stripUrl % 'totp-page-1-nsfw'
latestSearch = '//a[contains(@title, "NSFW")]'
starter = indirectStarter
adult = True
class TrippingOverYou(_BasicScraper):
url = 'http://www.trippingoveryou.com/'
stripUrl = url + 'comic/%s'
firstStripUrl = stripUrl % 'wiggle-room'
imageSearch = compile(tagre("img", "src", r'([^"]+/comics/[^"]+)'))
prevSearch = compile(r'<a class="cc-prev" rel="prev" href="(.+?)">')
help = 'Index format: stripname'
class TumbleDryComics(_WordPressScraper):
url = 'http://tumbledrycomics.com/'
stripUrl = url + 'comic/%s/'
firstStripUrl = stripUrl % 'we-need-to-get-high-jpg'
textSearch = '//div[@id="comic"]//img/@alt'
multipleImagesPerStrip = True
adult = True
help = 'Index format: name'
def namer(self, image_url, page_url):
# Most images have the date they were posted in the filename
# For those that don't we can get the month and year from the image url
parts = image_url.rsplit('/', 3)
year = parts[1]
month = parts[2]
filename = parts[3]
if not filename.startswith(year):
filename = year + "-" + month + "-" + filename
return filename
class Turnoff(_ParserScraper):
name = 'turnoff'
url = 'https://turnoff.us/'
imageSearch = '//article[%s]//img' % xpath_class('post-content')
prevSearch = '//div[%s]//a' % xpath_class('prev')
stripUrl = url + 'geek/%s'
firstStripUrl = stripUrl % 'tcp-buddies'
multipleImagesPerStrip = True
@cached_property
def comics_order(self):
# Neither the images nor the pages contain information about dates or indices.
# However we can extract the order of the images from the JavaScript.
html = self.session.get(self.url).text
list_regex = compile(r"""^\s*"/geek/(.*)",\s*$""", flags=MULTILINE)
return list(reversed(list_regex.findall(html)))
def namer(self, image_url, page_url):
comic_name = page_url.split('/')[-1]
try:
index = self.comics_order.index(comic_name) + 1
except ValueError:
index = len(self.comics_order)
file_name = image_url.split('/')[-1]
return "%03d-%s" % (index, file_name)
class TwinDragons(_WordPressScraper):
url = 'http://www.twindragonscomic.com/'
stripUrl = url + 'comic/%s/'
firstStripUrl = stripUrl % 'the-beginning'
multipleImagesPerStrip = True
class TwoGuysAndGuy(_BasicScraper):
url = 'http://www.twogag.com/'
rurl = escape(url)
stripUrl = url + 'archives/%s'
firstStripUrl = stripUrl % '4'
imageSearch = compile(tagre('img', 'src', r'(%scomics/\d{4}-\d{2}-\d{2}[^"]*)' % rurl))
prevSearch = compile(tagre('a', 'href', r'(%sarchives/\d+)' % rurl,
after='title="Previous"'))
help = 'Index format: number'
adult = True
class Twokinds(_ParserScraper):
url = 'http://twokinds.keenspot.com/'
stripUrl = url + 'comic/%s/'
firstStripUrl = stripUrl % '1'
imageSearch = '//article[%s]//img' % xpath_class('comic')
prevSearch = '//a[%s]' % xpath_class('navprev')
help = 'Index format: n (unpadded)'
class TwoLumps(_BasicScraper):
url = 'http://www.twolumps.net/'
stripUrl = url + 'd/%s.html'
imageSearch = compile(tagre("img", "src", r'(/comics/[^"]+)'))
prevSearch = compile(tagre("a", "href", r'(/d/\d+\.html)', after="prev"))
help = 'Index format: yyyymmdd'