482 lines
16 KiB
Python
482 lines
16 KiB
Python
# SPDX-License-Identifier: MIT
|
|
# SPDX-FileCopyrightText: © 2004 Tristan Seligmann and Jonathan Jacobs
|
|
# SPDX-FileCopyrightText: © 2012 Bastian Kleineidam
|
|
# SPDX-FileCopyrightText: © 2015 Tobias Gruetzmacher
|
|
# SPDX-FileCopyrightText: © 2019 Daniel Ring
|
|
from re import compile, escape
|
|
from typing import List
|
|
|
|
from ..scraper import _BasicScraper, _ParserScraper, ParserScraper
|
|
from ..helpers import bounceStarter, indirectStarter, joinPathPartsNamer
|
|
from ..util import tagre
|
|
from .common import ComicControlScraper, WordPressScraper, WordPressNavi, WordPressWebcomic
|
|
|
|
|
|
class CampComic(_ParserScraper):
|
|
url = 'http://campcomic.com/comic/'
|
|
stripUrl = url + '%s'
|
|
firstStripUrl = stripUrl % '6'
|
|
imageSearch = '//div[@id="comic"]/img'
|
|
prevSearch = '//a[d:class("btnPrev")]'
|
|
help = 'Index Format: number'
|
|
|
|
|
|
class CaptainSNES(_BasicScraper):
|
|
url = 'http://www.captainsnes.com/'
|
|
rurl = escape(url)
|
|
stripUrl = url + '%s/'
|
|
firstStripUrl = stripUrl % '2001/07/10/the-mistake'
|
|
imageSearch = compile(tagre("img", "src", r"(%scomics/[^']+)" % rurl,
|
|
quote="'"))
|
|
prevSearch = compile(tagre("a", "href", r'(%s[^"]+)' % rurl) +
|
|
tagre("span", "class", "prev"))
|
|
multipleImagesPerStrip = True
|
|
help = 'Index format: yyyy/mm/dd/nnn-stripname'
|
|
|
|
|
|
class CarryOn(_ParserScraper):
|
|
url = 'http://www.hirezfox.com/km/co/'
|
|
stripUrl = url + 'd/%s.html'
|
|
firstStripUrl = stripUrl % '20040701'
|
|
imageSearch = '//div[@class="strip"]/img'
|
|
prevSearch = '//a[text()="Previous Day"]'
|
|
multipleImagesPerStrip = True
|
|
|
|
def namer(self, imageUrl, pageUrl):
|
|
# Fix filenames of early comics
|
|
filename = imageUrl.rsplit('/', 1)[-1]
|
|
if filename[0].isdigit():
|
|
filename = 'co' + filename
|
|
return filename
|
|
|
|
|
|
class CarryOnAliceBlueAndTheGardensOfQ(CarryOn):
|
|
name = 'CarryOn/AliceBlueAndTheGardensOfQ'
|
|
url = 'http://www.hirezfox.com/km/abgq/abgq1024/'
|
|
stripUrl = url + 'd/%s.html'
|
|
firstStripUrl = stripUrl % '20050401'
|
|
endOfLife = True
|
|
|
|
def namer(self, imageUrl, pageUrl):
|
|
# Fix filenames
|
|
return 'abgq' + imageUrl.rsplit('/', 1)[-1]
|
|
|
|
|
|
class CarryOnLegendOfAnneBunny(CarryOn):
|
|
name = 'CarryOn/LegendOfAnneBunny'
|
|
url = 'http://www.hirezfox.com/km/loab/loab1024/'
|
|
stripUrl = url + 'd/%s.html'
|
|
firstStripUrl = stripUrl % '20040701'
|
|
endOfLife = True
|
|
|
|
def namer(self, imageUrl, pageUrl):
|
|
# Fix filenames of early comics
|
|
filename = imageUrl.rsplit('/', 1)[-1]
|
|
if filename[0].isdigit():
|
|
filename = 'ab' + filename
|
|
return filename
|
|
|
|
|
|
class CaseyAndAndy(_BasicScraper):
|
|
url = 'http://www.galactanet.com/comic/'
|
|
stripUrl = url + 'view.php?strip=%s'
|
|
firstStripUrl = stripUrl % '1'
|
|
imageSearch = compile(tagre("img", "src", r'(Strip\d+\.gif)'))
|
|
prevSearch = compile(tagre("a", "href", r'(view\.php\?strip=\d+)') +
|
|
tagre("img", "src", r'previous\.gif'))
|
|
help = 'Index format: number'
|
|
|
|
|
|
class CassiopeiaQuinn(ComicControlScraper):
|
|
url = 'https://www.cassiopeiaquinn.com/'
|
|
firstStripUrl = url + 'comic/the-prize-cover'
|
|
|
|
|
|
class CasuallyKayla(_BasicScraper):
|
|
url = 'http://casuallykayla.com/'
|
|
stripUrl = url + '?p=%s'
|
|
firstStripUrl = stripUrl % '89'
|
|
imageSearch = compile(tagre("img", "src",
|
|
r'(http://casuallykayla\.com/comics/[^"]+)'))
|
|
prevSearch = compile(tagre("div", "class", r'nav-previous') +
|
|
tagre("a", "href", r'([^"]+)'))
|
|
help = 'Index format: nnn'
|
|
|
|
|
|
class Catalyst(_BasicScraper):
|
|
baseUrl = "http://catalyst.spiderforest.com/"
|
|
rurl = escape(baseUrl)
|
|
url = baseUrl + "comic.php?comic_id=415"
|
|
stripUrl = baseUrl + "comic.php?comic_id=%s"
|
|
firstStripUrl = stripUrl % '1'
|
|
imageSearch = compile(tagre("img", "src", r'((?:%s)?comics/[^"]+)' % rurl))
|
|
prevSearch = compile("<center>" +
|
|
tagre("a", "href",
|
|
r'(%scomic\.php\?comic_id=\d+)' % rurl))
|
|
help = 'Index format: number'
|
|
|
|
|
|
class CatAndGirl(_ParserScraper):
|
|
url = 'https://catandgirl.com/'
|
|
imageSearch = '//div[@id="comic"]//img'
|
|
prevSearch = '//a[d:class("pager--prev")]'
|
|
|
|
|
|
class CatenaManor(_ParserScraper):
|
|
baseUrl = ('https://web.archive.org/web/20141027141116/'
|
|
'http://catenamanor.com/')
|
|
url = baseUrl + 'archives'
|
|
stripUrl = baseUrl + '%s/'
|
|
firstStripUrl = stripUrl % '2003/07'
|
|
imageSearch = '//img[@class="comicthumbnail"]'
|
|
multipleImagesPerStrip = True
|
|
endOfLife = True
|
|
strips: List[str] = []
|
|
|
|
def starter(self):
|
|
# Retrieve archive links and select valid range
|
|
archivePage = self.getPage(self.url)
|
|
archiveStrips = archivePage.xpath('//div[@id="archivepage"]//a')
|
|
valid = False
|
|
for link in archiveStrips:
|
|
if self.stripUrl % '2012/01' in link.get('href'):
|
|
valid = True
|
|
elif self.stripUrl % '2003/06' in link.get('href'):
|
|
valid = False
|
|
if valid:
|
|
self.strips.append(link.get('href'))
|
|
return self.strips.pop(0)
|
|
|
|
def getPrevUrl(self, url, data):
|
|
return self.strips.pop(0)
|
|
|
|
|
|
class CatNine(WordPressScraper):
|
|
url = 'https://classic.cat-nine.net/'
|
|
firstStripUrl = url + 'comic/day-first'
|
|
imageSearch = '//div[@id="comic"]//img/@data-src-img'
|
|
endOfLife = True
|
|
|
|
|
|
class CatNineTakeTwo(WordPressScraper):
|
|
name = 'CatNine/TakeTwo'
|
|
url = 'https://cat-nine.net/'
|
|
firstStripUrl = url + 'comic/episode-1/1-first-day-for-everything/'
|
|
|
|
|
|
class CatsAndCameras(WordPressScraper):
|
|
url = 'https://catsncameras.com/cnc/'
|
|
stripUrl = url + 'comic/%s'
|
|
firstStripUrl = stripUrl % 'cnc-begins'
|
|
adult = True
|
|
|
|
|
|
class CatVersusHuman(_ParserScraper):
|
|
url = 'http://www.catversushuman.com'
|
|
imageSearch = '//div[@class="post-body entry-content"]//img'
|
|
prevSearch = '//a[@id="Blog1_blog-pager-older-link"]'
|
|
latestSearch = '//a[@rel="bookmark"]'
|
|
starter = indirectStarter
|
|
|
|
|
|
class CavesAndCritters(WordPressWebcomic):
|
|
url = 'https://cavesandcritters.com/?ao_confirm'
|
|
stripUrl = 'https://cavesandcritters.com/cnc_webcomic/%s/'
|
|
firstStripUrl = stripUrl % '01_000'
|
|
adult = True
|
|
|
|
|
|
class Centralia2050(_ParserScraper):
|
|
url = 'https://centralia2050.com/'
|
|
stripUrl = url + 'comic/%s/'
|
|
firstStripUrl = stripUrl % 'ch1cover'
|
|
imageSearch = '//div[@id="spliced-comic"]//img'
|
|
prevSearch = '//a[@class="previous-comic"]'
|
|
nextSearch = '//a[@class="next-comic"]'
|
|
starter = bounceStarter
|
|
|
|
def namer(self, imageUrl, pageUrl):
|
|
page = pageUrl.rstrip('/').rsplit('/', 1)[-1].replace('chapter', 'ch')
|
|
if 'page-' in page and 'ch-' not in page:
|
|
page = 'ch-1-' + page
|
|
ext = imageUrl.rsplit('.', 1)[-1]
|
|
return page + '.' + ext
|
|
|
|
|
|
class ChannelAte(WordPressNavi):
|
|
url = 'http://www.channelate.com/'
|
|
|
|
|
|
class ChasingTheSunset(_BasicScraper):
|
|
url = 'http://www.fantasycomic.com/'
|
|
stripUrl = url + 'index.php?p=%s'
|
|
firstStripUrl = stripUrl % 'c1'
|
|
imageSearch = compile(r'(/cmsimg/.+?)".+?comic-img')
|
|
prevSearch = compile(r'<a href="(.+?)" title="" ><img src="(images/eye-prev.png|images/cn-prev.png)"')
|
|
help = 'Index format: n'
|
|
|
|
|
|
class Chester5000XYV(WordPressScraper):
|
|
url = 'http://jessfink.com/Chester5000XYV/'
|
|
stripUrl = url + '?p=%s'
|
|
firstStripUrl = stripUrl % '34'
|
|
prevSearch = '//a[@rel="prev"]'
|
|
adult = True
|
|
help = 'Index format: n (unpadded)'
|
|
|
|
def link_modifier(self, fromurl, tourl):
|
|
"""Bugfix for link to blog"""
|
|
if tourl == self.stripUrl % '714':
|
|
return self.stripUrl % '710'
|
|
return tourl
|
|
|
|
|
|
class Chisuji(WordPressScraper):
|
|
url = 'http://www.chisuji.com/'
|
|
stripUrl = url + '?p=%s'
|
|
firstStripUrl = stripUrl % '266'
|
|
prevSearch = '//div[@class="nav-previous"]/a'
|
|
help = 'Index format: nnn'
|
|
|
|
|
|
class CigarroAndCerveja(_ParserScraper):
|
|
url = 'http://www.cigarro.ca/'
|
|
stripUrl = url + 'comic/%s/'
|
|
firstStripUrl = stripUrl % 'reacquaintance'
|
|
imageSearch = '//div[@id="comic"]//img'
|
|
prevSearch = '//a[contains(text()," Prev")]'
|
|
|
|
|
|
class ClanOfTheCats(WordPressScraper):
|
|
url = 'http://www.cotclassic.com/'
|
|
stripUrl = url + 'comic/%s/'
|
|
firstStripUrl = stripUrl % 'coming-home-2'
|
|
|
|
def link_modifier(self, fromurl, tourl):
|
|
# Fix broken navigation link
|
|
return tourl.replace('/2954/', '/2002-06-22/')
|
|
|
|
|
|
class ClanOfTheCatsReunion(WordPressScraper):
|
|
name = 'ClanOfTheCats/Reunion'
|
|
url = 'http://www.clanofthecats.com/'
|
|
stripUrl = url + 'comic/%s/'
|
|
firstStripUrl = stripUrl % 'cotc-reunion'
|
|
|
|
|
|
class Cloudscratcher(_ParserScraper):
|
|
url = 'http://www.cloudscratcher.com/'
|
|
stripUrl = url + 'comic.php?page=%s'
|
|
firstStripUrl = stripUrl % '1'
|
|
imageSearch = '//img[contains(@src, "pages/")]'
|
|
prevSearch = '//a[./img[@alt="Previous Page"]]'
|
|
latestSearch = '//a[./img[@alt="Comic"]]'
|
|
starter = indirectStarter
|
|
|
|
|
|
class CollegeCatastrophe(_ParserScraper):
|
|
url = 'https://www.tigerknight.com/cc'
|
|
stripUrl = url + '/%s'
|
|
firstStripUrl = stripUrl % '2000-11-10'
|
|
imageSearch = '//img[d:class("comic-image")]'
|
|
prevSearch = '//a[./span[contains(text(), "Previous")]]'
|
|
endOfLife = True
|
|
multipleImagesPerStrip = True
|
|
|
|
|
|
class Comedity(_BasicScraper):
|
|
url = 'http://www.comedity.com/'
|
|
stripUrl = url + 'index.php?strip_id=%s'
|
|
firstStripUrl = stripUrl % '1'
|
|
imageSearch = compile(r'<img src="(Comedity_files/.+?)"')
|
|
prevSearch = compile(r'<a href="(/?index.php\?strip_id=\d+?)"> *<img alt=\"Prior Strip')
|
|
help = 'Index format: n (no padding)'
|
|
|
|
|
|
class CommanderKitty(WordPressNavi):
|
|
url = 'http://www.commanderkitty.com/'
|
|
stripUrl = url + '%s/'
|
|
firstStripUrl = stripUrl % '2009/01/03/good-to-be-back'
|
|
endOfLife = True
|
|
|
|
|
|
class CommitStrip(_ParserScraper):
|
|
baseUrl = 'https://www.commitstrip.com/en/'
|
|
url = baseUrl + '?setLocale=1' # ensure the language cookie is set
|
|
stripUrl = baseUrl + '%s/'
|
|
firstStripUrl = stripUrl % '2012/02/22/interview'
|
|
|
|
latestSearch = '//section//a'
|
|
starter = indirectStarter
|
|
imageSearch = '//article/div//img'
|
|
prevSearch = '//span[@class="nav-previous"]/a'
|
|
help = 'Index format: yyyy/mm/dd/strip-name'
|
|
|
|
def namer(self, image_url, page_url):
|
|
parts = page_url.rstrip('/').rsplit('/')[-4:]
|
|
return '-'.join(parts)
|
|
|
|
def link_modifier(self, fromurl, tourl):
|
|
return tourl.replace('http:', 'https:')
|
|
|
|
|
|
class CommitStripFr(CommitStrip):
|
|
baseUrl = 'https://www.commitstrip.com/fr/'
|
|
url = baseUrl + '?setLocale=1' # ensure the language cookie is set
|
|
stripUrl = baseUrl + '%s/'
|
|
firstStripUrl = stripUrl % '2012/02/22/interview'
|
|
lang = 'fr'
|
|
|
|
|
|
class CompanyY(_BasicScraper):
|
|
url = 'http://company-y.com/'
|
|
rurl = escape(url)
|
|
stripUrl = url + '%s/'
|
|
firstStripUrl = stripUrl % '2009/08/14/coming-soon'
|
|
imageSearch = compile(tagre("img", "src", r'(%scomics/[^"]+)' % rurl))
|
|
prevSearch = compile(tagre("div", "class", r"nav-previous") +
|
|
tagre("a", "href", r'(%s[^"]+)' % rurl))
|
|
help = 'Index format: yyyy/mm/dd/strip-name'
|
|
|
|
|
|
class Concession(_ParserScraper):
|
|
url = 'http://concessioncomic.com/'
|
|
stripUrl = url + 'index.php?pid=%s'
|
|
firstStripUrl = stripUrl % '20060701'
|
|
imageSearch = '//div[@id="comic"]/img[not(@class="preload")]'
|
|
prevSearch = '//a[@class="nav-prev"]'
|
|
adult = True
|
|
endOfLife = True
|
|
|
|
|
|
class CorydonCafe(_ParserScraper):
|
|
url = 'http://corydoncafe.com/'
|
|
imageSearch = "//center[2]//img"
|
|
prevSearch = '//a[@title="prev"]'
|
|
multipleImagesPerStrip = True
|
|
|
|
|
|
class CourtingDisaster(WordPressScraper):
|
|
url = 'https://web.archive.org/web/20201127150157/http://www.courting-disaster.com/'
|
|
firstStripUrl = 'http://www.courting-disaster.com/comic/courting-disaster-17/'
|
|
endOfLife = True
|
|
|
|
|
|
class CraftedFables(WordPressScraper):
|
|
url = 'https://web.archive.org/web/20191126025641/http://www.caf-fiends.net/comicpress/'
|
|
prevSearch = '//a[@rel="prev"]'
|
|
endOfLife = True
|
|
|
|
|
|
class CrimsonDark(_BasicScraper):
|
|
url = 'http://www.davidcsimon.com/crimsondark/'
|
|
stripUrl = url + 'index.php?view=comic&strip_id=%s'
|
|
firstStripUrl = stripUrl % '1'
|
|
imageSearch = compile(r'src="(.+?strips/.+?)"')
|
|
prevSearch = compile(r'<a href=[\'"](/crimsondark/index\.php\?view=comic&strip_id=\d+)[\'"]><img src=[\'"]themes/cdtheme/images/active_prev.png[\'"]')
|
|
help = 'Index format: n (unpadded)'
|
|
|
|
|
|
class CrimsonFlag(_ParserScraper):
|
|
url = 'http://crimsonflagcomic.com/'
|
|
stripUrl = url + 'comic.php?comicID=%s'
|
|
firstStripUrl = stripUrl % '1'
|
|
imageSearch = '//img[@class="comicimage"]'
|
|
prevSearch = '//a[contains(@class, "prev")]'
|
|
|
|
|
|
class CritterCoven(WordPressScraper):
|
|
url = 'http://crittercoven.com/'
|
|
stripUrl = url + 'comic/%s/'
|
|
firstStripUrl = stripUrl % 'critter-coven'
|
|
|
|
|
|
class CrossTimeCafe(_ParserScraper):
|
|
stripUrl = 'http://www.whiteponyproductions.com/ctc/%s.htm'
|
|
url = stripUrl % 'present'
|
|
firstStripUrl = stripUrl % 'ctc0001'
|
|
imageSearch = '//img[not(contains(@src, "graphics/"))]'
|
|
prevSearch = '//a[.//text()="Back"]'
|
|
multipleImagesPerStrip = True
|
|
endOfLife = True
|
|
|
|
|
|
class CSectionComics(WordPressScraper):
|
|
url = 'https://www.csectioncomics.com/'
|
|
firstStripUrl = url + 'comics/one-day-in-country'
|
|
namer = joinPathPartsNamer((), (-3, -2, -1))
|
|
multipleImagesPerStrip = True
|
|
|
|
|
|
class CtrlAltDel(WordPressWebcomic):
|
|
url = 'https://cad-comic.com/'
|
|
stripUrl = url + 'comic/%s/'
|
|
firstStripUrl = stripUrl % 'nice-melon'
|
|
imageSearch = '//div[@class="comicpage"]//img[contains (@src,"/uploads/")]'
|
|
prevSearch = '//a[@rel="prev"]'
|
|
|
|
|
|
class CucumberQuest(WordPressWebcomic):
|
|
baseUrl = 'http://cucumber.gigidigi.com/'
|
|
stripUrl = baseUrl + 'cq/%s/'
|
|
firstStripUrl = stripUrl % 'page-1'
|
|
url = firstStripUrl
|
|
starter = indirectStarter
|
|
help = 'Index format: stripname'
|
|
|
|
|
|
class Curtailed(WordPressScraper):
|
|
url = 'https://www.curtailedcomic.com/'
|
|
stripUrl = url + 'comic/%s/'
|
|
firstStripUrl = stripUrl % '001-sneeze'
|
|
|
|
def shouldSkipUrl(self, url, data):
|
|
"""Skip pages without images."""
|
|
return 'comic/sitrep-1' in url or 'comic/be-right-back' in url
|
|
|
|
|
|
class Curvy(_ParserScraper):
|
|
url = 'http://www.c.urvy.org/'
|
|
stripUrl = url + '?date=%s'
|
|
firstStripUrl = stripUrl % '20080329'
|
|
imageSearch = '//div[@id="theActualComic"]//img'
|
|
prevSearch = '//div[@class="aNavbar"]//p[2]/a'
|
|
help = 'Index format: yyyymmdd'
|
|
|
|
|
|
class CutLoose(_ParserScraper):
|
|
url = 'https://www.cutloosecomic.com/'
|
|
stripUrl = url + 'archive/comic/%s'
|
|
firstStripUrl = stripUrl % '2016/02/02'
|
|
imageSearch = '//img[@id="comic-container"]'
|
|
prevSearch = '//a[@title="Previous Comic"]'
|
|
nextSearch = '//a[@title="Next Comic"]'
|
|
starter = bounceStarter
|
|
adult = True
|
|
|
|
def namer(self, imageUrl, pageUrl):
|
|
postDate = pageUrl.rsplit('/', 3)
|
|
filename = imageUrl.rsplit('/', 1)[-1]
|
|
return '%s-%s-%s_%s' % (postDate[1], postDate[2], postDate[3], filename)
|
|
|
|
|
|
class CyanideAndHappiness(ParserScraper):
|
|
url = 'https://explosm.net/'
|
|
imageSearch = '//div[@id="comic"]//div[contains(@class,"ComicImage")]/span//img'
|
|
prevSearch = '//div[@type="comic"]//a[*[local-name()="svg" and @rotate="180deg"]]'
|
|
nextSearch = '//div[@type="comic"]//a[*[local-name()="svg" and @rotate="0deg"]]'
|
|
starter = bounceStarter
|
|
namer = joinPathPartsNamer((), range(-4, 0))
|
|
|
|
|
|
class CynWolf(_ParserScraper):
|
|
url = 'https://cynwolf.net/'
|
|
stripUrl = url + '%s/'
|
|
firstStripUrl = stripUrl % '2008/because'
|
|
imageSearch = '//section[contains(@class, "comic")]//img'
|
|
prevSearch = '//a[text()="\u2190"]'
|
|
multipleImagesPerStrip = True
|
|
endOfLife = True
|
|
|
|
def shouldSkipUrl(self, url, data):
|
|
return '2016/the-end' in url # video
|