2015-07-17 21:33:25 +00:00
|
|
|
# -*- coding: utf-8 -*-
|
2012-06-20 20:41:04 +00:00
|
|
|
# Copyright (C) 2004-2005 Tristan Seligmann and Jonathan Jacobs
|
2014-01-05 15:50:57 +00:00
|
|
|
# Copyright (C) 2012-2014 Bastian Kleineidam
|
2016-03-07 00:08:57 +00:00
|
|
|
# Copyright (C) 2015-2016 Tobias Gruetzmacher
|
|
|
|
|
2012-11-13 18:12:28 +00:00
|
|
|
import re
|
2012-11-26 17:44:31 +00:00
|
|
|
import os
|
2013-02-28 20:08:43 +00:00
|
|
|
import multiprocessing
|
2013-11-07 20:22:38 +00:00
|
|
|
try:
|
|
|
|
from urllib.parse import urlsplit
|
|
|
|
except ImportError:
|
|
|
|
from urlparse import urlsplit
|
2012-06-20 19:58:13 +00:00
|
|
|
from dosagelib import scraper
|
2016-03-07 00:08:57 +00:00
|
|
|
from . import tmpdir # noqa
|
2012-06-20 19:58:13 +00:00
|
|
|
|
|
|
|
|
2013-02-28 20:08:43 +00:00
|
|
|
def get_host(url):
|
|
|
|
"""Get host part of URL."""
|
2013-11-07 20:22:38 +00:00
|
|
|
return urlsplit(url)[1].lower()
|
2013-02-28 20:08:43 +00:00
|
|
|
|
|
|
|
|
|
|
|
# Dictionary with per-host locks.
|
|
|
|
_locks = {}
|
|
|
|
# Allowed number of connections per host
|
|
|
|
MaxConnections = 4
|
2016-03-07 00:08:57 +00:00
|
|
|
# Maximum number of strips to get to test a comic
|
|
|
|
MaxStrips = 5
|
|
|
|
|
2013-02-28 20:08:43 +00:00
|
|
|
|
|
|
|
def get_lock(host):
|
|
|
|
"""Get bounded semphore for given host."""
|
|
|
|
if host not in _locks:
|
|
|
|
_locks[host] = multiprocessing.BoundedSemaphore(MaxConnections)
|
|
|
|
return _locks[host]
|
|
|
|
|
2015-07-17 21:33:25 +00:00
|
|
|
|
2016-03-07 00:08:57 +00:00
|
|
|
def _get_saved_images(outdir, scraper):
|
2015-07-17 21:33:25 +00:00
|
|
|
"""Get saved images."""
|
|
|
|
dirs = tuple(scraper.getName().split('/'))
|
2016-03-07 00:08:57 +00:00
|
|
|
files = os.listdir(os.path.join(outdir, *dirs))
|
|
|
|
files = [x for x in files if not x.endswith(".txt")]
|
2015-07-17 21:33:25 +00:00
|
|
|
return files
|
|
|
|
|
2016-03-07 00:08:57 +00:00
|
|
|
|
|
|
|
def test_comicmodule(tmpdir, scraperclass): # noqa
|
|
|
|
'''Test a scraper. It must be able to traverse backward for at least 5
|
|
|
|
strips from the start, and find strip images on at least 4 pages.'''
|
2015-07-17 21:33:25 +00:00
|
|
|
scraperobj = scraperclass()
|
|
|
|
# Limit number of connections to one host.
|
|
|
|
host = get_host(scraperobj.url)
|
|
|
|
try:
|
|
|
|
with get_lock(host):
|
|
|
|
_test_comic(tmpdir, scraperobj)
|
|
|
|
except OSError:
|
|
|
|
# interprocess lock not supported
|
|
|
|
_test_comic(tmpdir, scraperobj)
|
|
|
|
|
2016-03-07 00:08:57 +00:00
|
|
|
|
|
|
|
def _test_comic(outdir, scraperobj):
|
2015-07-17 21:33:25 +00:00
|
|
|
num_strips = 0
|
|
|
|
strip = None
|
2016-03-07 00:08:57 +00:00
|
|
|
for strip in scraperobj.getStrips(MaxStrips):
|
|
|
|
_check_strip(outdir, strip, scraperobj.multipleImagesPerStrip)
|
|
|
|
|
2015-07-17 21:33:25 +00:00
|
|
|
if num_strips > 0 and scraperobj.prevUrlMatchesStripUrl:
|
2016-03-07 00:08:57 +00:00
|
|
|
_check_stripurl(strip, scraperobj)
|
2015-07-17 21:33:25 +00:00
|
|
|
num_strips += 1
|
2016-03-07 00:08:57 +00:00
|
|
|
|
2015-07-17 21:33:25 +00:00
|
|
|
if scraperobj.prevSearch and not scraperobj.hitFirstStripUrl:
|
2016-03-07 00:08:57 +00:00
|
|
|
# subtract the number of skipped URLs with no image from the expected
|
|
|
|
# image number
|
|
|
|
num_strips_expected = MaxStrips - len(scraperobj.skippedUrls)
|
|
|
|
msg = 'Traversed %d strips instead of %d.' % (num_strips,
|
|
|
|
num_strips_expected)
|
2015-07-17 21:33:25 +00:00
|
|
|
if strip:
|
|
|
|
msg += " Check the prevSearch pattern at %s" % strip.stripUrl
|
|
|
|
assert num_strips == num_strips_expected, msg
|
|
|
|
if strip:
|
2016-03-07 00:08:57 +00:00
|
|
|
_check_scraperesult(outdir, num_strips_expected, strip, scraperobj)
|
|
|
|
|
|
|
|
|
|
|
|
def _check_strip(outdir, strip, multipleImagesPerStrip):
|
|
|
|
'''Check that a specific page yields images and the comic module correctly
|
|
|
|
declares if there are multiple images per page.'''
|
|
|
|
images = []
|
|
|
|
for image in strip.getImages():
|
|
|
|
images.append(image.url)
|
|
|
|
image.save(outdir)
|
|
|
|
assert images, 'failed to find images at %s' % strip.stripUrl
|
|
|
|
if not multipleImagesPerStrip:
|
|
|
|
assert len(images) == 1, 'found more than 1 image at %s: %s' % (
|
|
|
|
strip.stripUrl, images)
|
2015-07-17 21:33:25 +00:00
|
|
|
|
2016-03-07 00:08:57 +00:00
|
|
|
|
|
|
|
def _check_scraperesult(outdir, num_images_expected, strip, scraperobj):
|
|
|
|
'''Check that exactly or for multiple pages at least num_strips images are
|
|
|
|
saved. This checks saved files, ie. it detects duplicate filenames.'''
|
|
|
|
saved_images = _get_saved_images(outdir, scraperobj)
|
2015-07-17 21:33:25 +00:00
|
|
|
num_images = len(saved_images)
|
2016-03-07 00:08:57 +00:00
|
|
|
|
|
|
|
attrs = (num_images, saved_images, num_images_expected, outdir)
|
2015-07-17 21:33:25 +00:00
|
|
|
if scraperobj.multipleImagesPerStrip:
|
2016-03-07 00:08:57 +00:00
|
|
|
err = 'saved %d %s instead of at least %d images in %s' % attrs
|
|
|
|
assert num_images >= num_images_expected, err
|
2015-07-17 21:33:25 +00:00
|
|
|
else:
|
2016-03-07 00:08:57 +00:00
|
|
|
err = 'saved %d %s instead of %d images in %s' % attrs
|
|
|
|
assert num_images == num_images_expected, err
|
2015-07-17 21:33:25 +00:00
|
|
|
|
2016-03-07 00:08:57 +00:00
|
|
|
|
|
|
|
def _check_stripurl(strip, scraperobj):
|
2015-07-17 21:33:25 +00:00
|
|
|
if not scraperobj.stripUrl:
|
|
|
|
# no indexing support
|
|
|
|
return
|
|
|
|
# test that the stripUrl regex matches the retrieved strip URL
|
|
|
|
urlmatch = re.escape(scraperobj.stripUrl)
|
|
|
|
urlmatch = urlmatch.replace(r"\%s", r".+")
|
|
|
|
urlmatch = "^%s$" % urlmatch
|
|
|
|
ro = re.compile(urlmatch)
|
|
|
|
mo = ro.search(strip.stripUrl)
|
2016-03-07 00:08:57 +00:00
|
|
|
err = 'strip URL %r does not match stripUrl pattern %s' % (
|
|
|
|
strip.stripUrl, urlmatch)
|
|
|
|
assert mo is not None, err
|
|
|
|
|
2015-07-17 21:33:25 +00:00
|
|
|
|
|
|
|
def get_test_scraperclasses():
|
|
|
|
"""Return scrapers that should be tested."""
|
2013-11-08 17:40:55 +00:00
|
|
|
if "TESTALL" in os.environ:
|
|
|
|
# test all comics (this will take some time)
|
2013-02-15 19:39:20 +00:00
|
|
|
scraperclasses = scraper.get_scraperclasses()
|
2013-11-08 17:40:55 +00:00
|
|
|
else:
|
|
|
|
# Get limited number of scraper tests on Travis builds to make
|
|
|
|
# it faster
|
2016-03-07 00:08:57 +00:00
|
|
|
testscrapernames = [
|
|
|
|
'AbstruseGoose',
|
|
|
|
'GoComics/CalvinandHobbes',
|
|
|
|
'xkcd'
|
|
|
|
]
|
2013-11-08 17:40:55 +00:00
|
|
|
scraperclasses = [
|
|
|
|
scraperclass for scraperclass in scraper.get_scraperclasses()
|
|
|
|
if scraperclass.getName() in testscrapernames
|
|
|
|
]
|
2015-07-17 21:33:25 +00:00
|
|
|
return scraperclasses
|
2012-11-26 17:44:31 +00:00
|
|
|
|
2016-03-07 00:08:57 +00:00
|
|
|
|
2015-07-17 21:33:25 +00:00
|
|
|
def pytest_generate_tests(metafunc):
|
|
|
|
if 'scraperclass' in metafunc.fixturenames:
|
|
|
|
metafunc.parametrize('scraperclass', get_test_scraperclasses())
|