# -*- coding: utf-8 -*- # Copyright (C) 2004-2005 Tristan Seligmann and Jonathan Jacobs # Copyright (C) 2012-2014 Bastian Kleineidam # Copyright (C) 2015-2016 Tobias Gruetzmacher from __future__ import absolute_import, division, print_function from re import compile, escape, IGNORECASE, sub from os.path import splitext import datetime from ..scraper import _BasicScraper, _ParserScraper from ..helpers import indirectStarter, bounceStarter from ..util import tagre from .common import (_ComicControlScraper, _WordPressScraper, WP_LATEST_SEARCH, xpath_class) class SabrinaOnline(_BasicScraper): url = 'http://sabrina-online.com/' stripUrl = url + '%s.html' firstStripUrl = stripUrl % '1996-01' imageSearch = compile(tagre("a", "href", r'(strips/[^"]*)')) prevSearch = compile(tagre("a", "href", r"(\d\d\d\d-\d\d.html)") + tagre("img", "src", "b_back.gif")) help = 'Index format: yyyy-qq' adult = True multipleImagesPerStrip = True def starter(self): """Pick last one in a list of archive pages.""" archive = self.url + 'archive.html' data = self.getPage(archive) search = compile(tagre("a", "href", r"(\d\d\d\d-\d\d.html)")) archivepages = self.fetchUrls(archive, data, search) return archivepages[-1] class SafelyEndangered(_WordPressScraper): url = 'http://www.safelyendangered.com/' firstStripUrl = url + 'comic/ignored/' prevSearch = '//a[%s]' % xpath_class('navi-prev') class SailorsunOrg(_WordPressScraper): url = 'http://sailorsun.org/' class SamAndFuzzy(_BasicScraper): url = 'http://www.samandfuzzy.com/' stripUrl = 'http://samandfuzzy.com/%s' firstStripUrl = stripUrl % '1' imageSearch = compile(r'(/comics/.+?)" alt') prevSearch = compile(r'"> 0: image = images[0] return [self.url + '/images/pages/' + image + '-xsmall.png'] def starter(self): """Go forward as far as possibe, then start.""" url = self.url while True: data = self.getPage(url) try: url = self.fetchUrl(url, data, self.nextSearch) except ValueError: break return url class SequentialArt(_BasicScraper): url = 'http://www.collectedcurios.com/sequentialart.php' stripUrl = url + '?s=%s' firstStripUrl = stripUrl % '1' imageSearch = compile(tagre("img", "src", r'([^"]+)', before="strip")) prevSearch = compile(tagre("a", "href", r'(/sequentialart\.php\?s=\d+)') + tagre("img", "src", "Nav_BackOne\.gif")) help = 'Index format: name' class SexyLosers(_ParserScraper): adult = True url = 'http://www.sexylosers.com/' stripUrl = url + 'comic/%s/' firstStripUrl = stripUrl % '003' imageSearch = '//div[@class="entry-content"]//img' prevSearch = '//a[@rel="prev"]' latestSearch = '//a[@rel="bookmark"]' help = 'Index format: nnn' starter = indirectStarter def namer(self, image_url, page_url): index = page_url.rsplit('/', 2)[1] title = image_url.rsplit('/', 1)[1] return index + '-' + title class Sharksplode(_WordPressScraper): url = 'http://sharksplode.com/' textSearch = '//div[@id="comic"]//img/@alt' class Sheldon(_BasicScraper): url = 'http://www.sheldoncomics.com/' rurl = escape(url) stripUrl = url + 'archive/%s.html' firstStripUrl = stripUrl % '011130' imageSearch = compile(tagre("img", "src", r'(http://cdn\.sheldoncomics\.com/strips/[^"]+)')) prevSearch = compile(tagre("a", "href", r'(%sarchive/\d+\.html)' % rurl, after="sidenav-prev")) help = 'Index format: yymmdd' class ShermansLagoon(_BasicScraper): url = 'http://shermanslagoon.com/' stripUrl = url + 'comics/%s' firstStripUrl = stripUrl % '/december-29-2003/' imageSearch = compile(tagre("img", "src", r'(https://safr\.kingfeatures\.com/idn/cnfeed/zone/js/content\.php\?file=.+?)')) prevSearch = compile(r'id="previouscomic" class="button white">]+?>') multipleImagesPerStrip = True help = 'Index format: yymmdd' class SMBC(_ParserScraper): url = 'http://www.smbc-comics.com/' rurl = escape(url) stripUrl = url + '?id=%s' firstStripUrl = stripUrl % '1' multipleImagesPerStrip = True imageSearch = ['//img[@id="comic"]', '//div[@id="aftercomic"]/img'] prevSearch = '//a[@class="prev"]' help = 'Index format: nnnn' textSearch = '//img[@id="comic"]/@title' def namer(self, image_url, page_url): """Remove random noise from name.""" return image_url.rsplit('-', 1)[-1] def shouldSkipUrl(self, url, data): """Skip promo or missing update pages.""" return url in ( self.stripUrl % '2865', self.stripUrl % '2653', self.stripUrl % '2424', self.stripUrl % '2226', self.stripUrl % '2069', self.stripUrl % '1895', self.stripUrl % '1896', self.stripUrl % '1589', ) class SnowFlame(_WordPressScraper): url = 'http://www.snowflamecomic.com/' stripUrl = url + '?comic=snowflame-%s-%s' firstStripUrl = stripUrl % ('01', '01') starter = bounceStarter nextSearch = WP_LATEST_SEARCH help = 'Index format: chapter-page' def getIndexStripUrl(self, index): return self.stripUrl % tuple(index.split('-')) def namer(self, image_url, page_url): prefix, filename = image_url.rsplit('/', 1) ro = compile(r'snowflame-([^-]+)-([^-]+)') mo = ro.search(page_url) chapter = mo.group(1) page = mo.group(2) return "%s-%s-%s" % (chapter, page, filename) class SodiumEyes(_WordPressScraper): url = 'http://sodiumeyes.com/' class SomethingPositive(_BasicScraper): url = 'http://www.somethingpositive.net/' stripUrl = url + 'sp%s.shtml' imageSearch = ( compile(tagre("img", "src", r'(sp\d+\.png)')), compile(tagre("img", "src", r'(twither\.gif)')), ) prevSearch = compile(tagre("a", "href", r'(sp\d+\.shtml)') + "(?:" + tagre("img", "src", r'images/previous\.gif') + "|Previous)") help = 'Index format: mmddyyyy' class Sorcery101(_BasicScraper): baseUrl = 'http://www.sorcery101.net/' url = baseUrl + 'sorcery-101/' rurl = escape(baseUrl) stripUrl = url + '%s/' imageSearch = compile(tagre("img", "src", r'(%swp-content/uploads/\d+/\d+/[^"]+)' % rurl)) prevSearch = compile(tagre("a", "href", r'(%ssorcery-101/[^"]+)' % rurl, after="previous-")) help = 'Index format: stripname' class SpaceTrawler(_WordPressScraper): base_url = 'http://spacetrawler.com/' url = base_url + '2013/12/24/spacetrawler-379/' firstStripUrl = base_url + '2010/01/01/spacetrawler-4/' prevSearch = '//a[%s]' % xpath_class('navi-prev') endOfLife = True class Spamusement(_BasicScraper): url = 'http://spamusement.com/' rurl = escape(url) stripUrl = url + 'index.php/comics/view/%s' imageSearch = compile(r'' % rurl, IGNORECASE) latestSearch = prevSearch help = 'Index format: n (unpadded)' starter = indirectStarter class SpareParts(_BasicScraper): baseUrl = 'http://www.sparepartscomics.com/' url = baseUrl + 'comics/?date=20080328' stripUrl = baseUrl + 'comics/index.php?date=%s' firstStripUrl = stripUrl % '20031022' imageSearch = compile(tagre("img", "src", r'(http://www\.sparepartscomics\.com/comics/[^"]+)')) prevSearch = compile(tagre("a", "href", r'(index\.php\?date=\d+)', quote="'") + "Previous Comic") help = 'Index format: yyyymmdd' class Spinnerette(_ComicControlScraper): url = 'http://www.spinnyverse.com' class SPQRBlues(_WordPressScraper): url = 'http://spqrblues.com/IV/' class StandStillStaySilent(_ParserScraper): url = 'http://www.sssscomic.com/comic.php' stripUrl = url + '?page=%s' firstStripUrl = stripUrl % '1' imageSearch = '//img[@class="comicnormal"]' prevSearch = '//a[div[@id="navprev"]]' help = 'Index Format: number' class StarCrossdDestiny(_ParserScraper): baseUrl = 'http://www.starcrossd.net/' url = baseUrl + 'comic.html' stripUrl = baseUrl + 'archives/%s.html' firstStripUrl = stripUrl % '00000001' imageSearch = '//div[@id="comic"]//img' prevSearch = '//a[text()="prev"]' help = 'Index format: nnnnnnnn' def namer(self, image_url, page_url): if image_url.find('ch1') == -1: # At first all images were stored in a strips/ directory but # that was changed with the introduction of book2 image_url = sub('(?:strips)|(?:images)', 'book1', image_url) elif not image_url.find('strips') == -1: image_url = image_url.replace('strips/', '') directory, filename = image_url.split('/')[-2:] filename, extension = splitext(filename) return directory + '-' + filename class StationV3(_ParserScraper): url = 'http://www.stationv3.com/' stripUrl = url + 'd/%s.html' imageSearch = '//img[contains(@src,"/comics2/")]' prevSearch = '//a[img[contains(@src,"/previous2")]]' help = 'Index format: yyyymmdd' class StickyDillyBuns(_BasicScraper): url = 'http://www.stickydillybuns.com/' stripUrl = url + 'strips-sdb/%s' firstStripUrl = stripUrl % 'awesome_leading_man' imageSearch = compile(tagre("img", "src", r'([^"]*/comics/[^"]+)')) prevSearch = compile(tagre("a", "href", r'([^"]*/strips-sdb/[^"]+)', before="cn[id]prev")) help = 'Index format: name' class StrawberryDeathCake(_BasicScraper): url = 'http://strawberrydeathcake.com/' rurl = escape(url) imageSearch = compile(tagre("img", "src", r'(%swp-content/webcomic/[^"]+)' % rurl)) prevSearch = compile(tagre("a", "href", r'(%sarchive/[^"]+)' % rurl, after="previous")) class StreetFighter(_ComicControlScraper): url = 'http://www.streetfightercomics.com' class StrongFemaleProtagonist(_ParserScraper): url = 'http://strongfemaleprotagonist.com/' stripUrl = url + '%s/' css = True imageSearch = 'article p img' prevSearch = 'div.nav-previous > a' help = 'Index format: issue-?/page-??' def shouldSkipUrl(self, url, data): """Skip hiatus & non-comic pages.""" return url in ( self.stripUrl % 'guest-art/tuesday', self.stripUrl % 'guest-art/friday', self.stripUrl % 'guest-art/wednesday', self.stripUrl % 'issue-5/newspaper', self.stripUrl % 'issue-5/hiatus-1', self.stripUrl % 'issue-5/hiatus-2', self.stripUrl % 'ssue-1/no-page', ) class StuffNoOneToldMe(_BasicScraper): url = 'http://www.snotm.com/' stripUrl = url + '%s.html' firstStripUrl = stripUrl % '2010/05/01' olderHref = r"(http://www\.snotm\.com/\d+/\d+/[^']+\.html)" starter = indirectStarter imageSearch = ( compile(tagre("img", "src", r'(http://i\.imgur\.com/[^"]+)') + r"(?:|
)"), compile(tagre("img", "src", r'(http://\d+\.bp\.blogspot\.com/[^"]+)') + r"(?:(?: )?|)"), compile(tagre("img", "src", r'(https://lh\d+\.googleusercontent\.com/[^"]+)') + r""), ) prevSearch = compile(tagre("a", "href", olderHref, quote="'", before="older-link")) latestSearch = compile(tagre("a", "href", olderHref, quote="'")) multipleImagesPerStrip = True help = 'Index format: yyyy/mm/stripname' def namer(self, image_url, page_url): """Use page URL to construct meaningful image name.""" parts, year, month, stripname = page_url.rsplit('/', 3) stripname = stripname.rsplit('.', 1)[0] parts, imagename = image_url.rsplit('/', 1) return '%s-%s-%s-%s' % (year, month, stripname, imagename) def shouldSkipUrl(self, url, data): """Skip pages without images.""" return url in ( self.stripUrl % '2012/08/self-rant', # no comic self.stripUrl % '2012/06/if-you-wonder-where-ive-been', # video self.stripUrl % '2011/10/i-didnt-make-this-nor-have-anything-to', # video self.stripUrl % '2010/12/first-snotm-fans-in-sao-paulo', # no comic self.stripUrl % '2010/11/ear-infection', # no comic ) class SupernormalStep(_ComicControlScraper): url = 'http://supernormalstep.com/'