yet another simple static gallery generator
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

458 lines
15 KiB

6 years ago
import datetime
import logging
import os
import re
import shutil
from os.path import join
6 years ago
import pytest
from PIL import Image as PILImage
6 years ago
from sigal.gallery import Album, Gallery, Image, Media, Video
from sigal.video import SubprocessException
1 year ago
try:
from pillow_heif import HeifImagePlugin # noqa: F401
HAS_HEIF = True
except ImportError:
HAS_HEIF = False
CURRENT_DIR = os.path.dirname(__file__)
REF = {
"dir1": {
"title": "An example gallery",
"name": "dir1",
4 weeks ago
"thumbnail": "./dir1/test1/thumbnails/11.tn.jpg",
"subdirs": ["test1", "test2", "test3"],
"medias": [],
},
"dir1/test1": {
"title": "An example sub-category",
"name": "test1",
4 weeks ago
"thumbnail": "./test1/thumbnails/11.tn.jpg",
"subdirs": [],
"medias": [
"11.jpg",
"CMB_Timeline300_no_WMAP.jpg",
"flickr_jerquiaga_2394751088_cc-by-nc.jpg",
"example.gif",
5 years ago
],
},
"dir1/test2": {
"title": "test2",
"name": "test2",
4 weeks ago
"thumbnail": "./test2/thumbnails/21.tn.tiff",
"subdirs": [],
"medias": ["21.tiff", "22.jpg", "CMB_Timeline300_no_WMAP.jpg"],
},
"dir1/test3": {
"title": "01 First title alphabetically",
"name": "test3",
4 weeks ago
"thumbnail": "./test3/thumbnails/3.tn.jpg",
"subdirs": [],
"medias": ["3.jpg"],
},
"dir2": {
"title": "Another example gallery with a very long name",
"name": "dir2",
4 weeks ago
"thumbnail": "./dir2/thumbnails/m57_the_ring_nebula-587px.tn.jpg",
"subdirs": [],
"medias": [
"KeckObservatory20071020.jpg",
"Hubble Interacting Galaxy NGC 5257.jpg",
"Hubble ultra deep field.jpg",
"m57_the_ring_nebula-587px.jpg",
5 years ago
],
},
"accentué": {
"title": "accentué",
"name": "accentué",
4 weeks ago
"thumbnail": "./accentu%C3%A9/thumbnails/h%C3%A9lico%C3%AFde.tn.jpg",
"subdirs": [],
"medias": ["hélicoïde.jpg", "11.jpg"],
},
"video": {
"title": "video",
"name": "video",
4 weeks ago
"thumbnail": "./video/thumbnails/example%20video.tn.jpg",
"subdirs": [],
"medias": ["example video.ogv"],
5 years ago
},
"webp": {
"title": "webp",
"name": "webp",
4 weeks ago
"thumbnail": "./webp/thumbnails/_MG_7805_lossy80.tn.webp",
"subdirs": [],
"medias": ["_MG_7805_lossy80.webp", "_MG_7808_lossy80.webp"],
5 years ago
},
}
def test_media(settings):
m = Media("11.jpg", "dir1/test1", settings)
path = join("dir1", "test1")
file_path = join(path, "11.jpg")
thumb = join("thumbnails", "11.tn.jpg")
assert m.dst_filename == "11.jpg"
assert m.src_path == join(settings["source"], file_path)
assert m.dst_path == join(settings["destination"], file_path)
assert m.thumb_name == thumb
assert m.thumb_path == join(settings["destination"], path, thumb)
assert m.title == "Foo Bar"
2 years ago
assert m.description.startswith(
"<p>This is a <em>funny</em> <strong>description</strong> of this image</p>"
)
6 years ago
assert repr(m) == f"<Media>('{file_path}')"
assert str(m) == file_path
def test_media_orig(settings, tmpdir):
settings["keep_orig"] = False
m = Media("11.jpg", "dir1/test1", settings)
assert m.big is None
settings["keep_orig"] = True
settings["destination"] = str(tmpdir)
m = Image("11.jpg", "dir1/test1", settings)
assert m.big == "original/11.jpg"
m = Video("example video.ogv", "video", settings)
assert m.dst_filename == "example video.webm"
4 weeks ago
assert m.big_url == "./original/example%20video.ogv"
assert os.path.isfile(join(settings["destination"], m.path, m.big))
settings["use_orig"] = True
m = Image("21.jpg", "dir1/test2", settings)
assert m.big == "21.jpg"
def test_media_iptc_override(settings):
img_with_md = Image("2.jpg", "iptcTest", settings)
assert img_with_md.title == "Markdown title beats iptc"
# Markdown parsing adds formatting. Let's just focus on content
assert "Markdown description beats iptc" in img_with_md.description
img_no_md = Image("1.jpg", "iptcTest", settings)
5 years ago
assert (
img_no_md.title
== "Haemostratulus clouds over Canberra - 2005-12-28 at 03-25-07"
5 years ago
)
assert (
img_no_md.description == '"Haemo" because they look like haemoglobin '
5 years ago
'cells and "stratulus" because I can\'t work out whether '
"they're Stratus or Cumulus clouds.\nWe're driving down "
"the main drag in Canberra so it's Parliament House that "
"you can see at the end of the road."
5 years ago
)
5 years ago
def test_media_img_format(settings):
settings["img_format"] = "JPEG"
m = Image("11.tiff", "dir1/test1", settings)
path = join("dir1", "test1")
thumb = join("thumbnails", "11.tn.jpg")
assert m.dst_filename == "11.jpg"
assert m.src_path == join(settings["source"], path, "11.tiff")
assert m.dst_path == join(settings["destination"], path, "11.jpg")
5 years ago
assert m.thumb_name == thumb
assert m.thumb_path == join(settings["destination"], path, thumb)
5 years ago
assert m.title == "Foo Bar"
2 years ago
assert m.description.startswith(
"<p>This is a <em>funny</em> <strong>description</strong> of this image</p>"
)
5 years ago
file_path = join(path, "11.tiff")
5 years ago
assert repr(m) == f"<Image>('{file_path}')"
assert str(m) == file_path
def test_image(settings, tmpdir):
settings["destination"] = str(tmpdir)
settings["datetime_format"] = "%d/%m/%Y"
m = Image("11.jpg", "dir1/test1", settings)
assert m.date == datetime.datetime(2006, 1, 22, 10, 32, 42)
assert m.exif["datetime"] == "22/01/2006"
os.makedirs(join(settings["destination"], "dir1", "test1", "thumbnails"))
4 weeks ago
assert m.thumbnail == join(".", "thumbnails", "11.tn.jpg")
assert os.path.isfile(m.thumb_path)
def test_video(settings, tmpdir):
settings["destination"] = str(tmpdir)
m = Video("example video.ogv", "video", settings)
src_path = join("video", "example video.ogv")
assert str(m) == src_path
file_path = join("video", "example video.webm")
assert m.dst_path == join(settings["destination"], file_path)
os.makedirs(join(settings["destination"], "video", "thumbnails"))
4 weeks ago
assert m.thumbnail == join(".", "thumbnails", "example%20video.tn.jpg")
assert os.path.isfile(m.thumb_path)
@pytest.mark.parametrize("path,album", REF.items())
def test_album(path, album, settings, tmpdir):
gal = Gallery(settings, ncpu=1)
a = Album(path, settings, album["subdirs"], album["medias"], gal)
assert a.title == album["title"]
assert a.name == album["name"]
assert a.subdirs == album["subdirs"]
assert a.thumbnail == album["thumbnail"]
if path == "video":
assert list(a.images) == []
5 years ago
assert [m.dst_filename for m in a.medias] == [
album["medias"][0].replace(".ogv", ".webm")
5 years ago
]
else:
assert list(a.videos) == []
assert [m.dst_filename for m in a.medias] == album["medias"]
assert len(a) == len(album["medias"])
12 years ago
def test_albums_sort(settings):
gal = Gallery(settings, ncpu=1)
album = REF["dir1"]
subdirs = list(album["subdirs"])
settings["albums_sort_reverse"] = False
a = Album("dir1", settings, album["subdirs"], album["medias"], gal)
a.sort_subdirs("")
assert [alb.name for alb in a.albums] == subdirs
settings["albums_sort_reverse"] = True
a = Album("dir1", settings, album["subdirs"], album["medias"], gal)
a.sort_subdirs("")
assert [alb.name for alb in a.albums] == list(reversed(subdirs))
titles = [im.title for im in a.albums]
titles.sort()
settings["albums_sort_reverse"] = False
a = Album("dir1", settings, album["subdirs"], album["medias"], gal)
a.sort_subdirs("title")
assert [im.title for im in a.albums] == titles
settings["albums_sort_reverse"] = True
a = Album("dir1", settings, album["subdirs"], album["medias"], gal)
a.sort_subdirs("title")
assert [im.title for im in a.albums] == list(reversed(titles))
orders = ["-10", "02", "03"]
settings["albums_sort_reverse"] = False
a = Album("dir1", settings, album["subdirs"], album["medias"], gal)
a.sort_subdirs("meta.order")
assert [d.meta["order"][0] for d in a.albums] == orders
settings["albums_sort_reverse"] = True
a = Album("dir1", settings, album["subdirs"], album["medias"], gal)
a.sort_subdirs("meta.order")
assert [d.meta["order"][0] for d in a.albums] == list(reversed(orders))
settings["albums_sort_reverse"] = False
a = Album("dir1", settings, album["subdirs"], album["medias"], gal)
a.sort_subdirs(["meta.partialorder", "meta.order"])
assert [d.name for d in a.albums] == list(["test1", "test2", "test3"])
settings["albums_sort_reverse"] = False
a = Album("dir1", settings, album["subdirs"], album["medias"], gal)
a.sort_subdirs(["meta.partialorderb", "name"])
assert [d.name for d in a.albums] == list(["test2", "test3", "test1"])
settings["albums_sort_reverse"] = True
a = Album("dir1", settings, album["subdirs"], album["medias"], gal)
a.sort_subdirs(["meta.partialorderb", "name"])
assert [d.name for d in a.albums] == list(["test1", "test3", "test2"])
def test_medias_sort(settings):
gal = Gallery(settings, ncpu=1)
album = REF["dir1/test2"]
settings["medias_sort_reverse"] = True
a = Album("dir1/test2", settings, album["subdirs"], album["medias"], gal)
a.sort_medias(settings["medias_sort_attr"])
assert [im.dst_filename for im in a.images] == list(reversed(album["medias"]))
settings["medias_sort_attr"] = "date"
settings["medias_sort_reverse"] = False
a = Album("dir1/test2", settings, album["subdirs"], album["medias"], gal)
a.sort_medias(settings["medias_sort_attr"])
assert a.medias[0].src_filename == "22.jpg"
settings["medias_sort_attr"] = "meta.order"
settings["medias_sort_reverse"] = False
a = Album("dir1/test2", settings, album["subdirs"], album["medias"], gal)
a.sort_medias(settings["medias_sort_attr"])
assert [im.dst_filename for im in a.images] == [
"21.tiff",
"22.jpg",
"CMB_Timeline300_no_WMAP.jpg",
5 years ago
]
def test_gallery(settings, tmp_path, caplog):
"Test the Gallery class."
caplog.set_level("ERROR")
settings["destination"] = str(tmp_path)
settings["user_css"] = str(tmp_path / "my.css")
settings["webm_options"] = ["-missing-option", "foobar"]
gal = Gallery(settings, ncpu=1)
gal.build()
1 year ago
if HAS_HEIF:
assert re.match(r"CSS file .* could not be found", caplog.records[3].message)
else:
assert re.match(r"CSS file .* could not be found", caplog.records[4].message)
with open(tmp_path / "my.css", mode="w") as f:
f.write("color: red")
gal.build()
mycss = os.path.join(settings["destination"], "static", "my.css")
assert os.path.isfile(mycss)
out_html = os.path.join(settings["destination"], "index.html")
assert os.path.isfile(out_html)
6 years ago
with open(out_html) as f:
html = f.read()
assert "<title>Sigal test gallery - Sigal test gallery ☺</title>" in html
4 weeks ago
assert '<link rel="stylesheet" href="./static/my.css">' in html
logger = logging.getLogger("sigal")
logger.setLevel(logging.DEBUG)
try:
gal = Gallery(settings, ncpu=1)
with pytest.raises(SubprocessException):
gal.build()
finally:
logger.setLevel(logging.INFO)
def test_custom_theme(settings, tmp_path, caplog):
theme_path = tmp_path / "mytheme"
tpl_path = theme_path / "templates"
settings["destination"] = str(tmp_path / "build")
settings["source"] = os.path.join(settings["source"], "encryptTest")
settings["theme"] = str(theme_path)
settings["title"] = "My gallery"
gal = Gallery(settings, ncpu=1)
with pytest.raises(Exception, match="Impossible to find the theme"):
gal.build()
tpl_path.mkdir(parents=True)
(theme_path / "static").mkdir(parents=True)
with pytest.raises(SystemExit):
gal.build()
assert caplog.records[0].message.startswith(
"The template album.html was not found in template folder"
)
with open(tpl_path / "album.html", mode="w") as f:
f.write(""" {{ settings.title|myfilter }} """)
with open(tpl_path / "album_list.html", mode="w") as f:
f.write(""" {{ settings.title|myfilter }} """)
with open(theme_path / "filters.py", mode="w") as f:
f.write(
"""
def myfilter(value):
return f'{value} is very nice'
"""
)
gal = Gallery(settings, ncpu=1)
gal.build()
out_html = os.path.join(settings["destination"], "index.html")
assert os.path.isfile(out_html)
with open(out_html) as f:
html = f.read()
assert "My gallery is very nice" in html
def test_gallery_max_img_pixels(settings, tmpdir, monkeypatch):
"Test the Gallery class with the max_img_pixels setting."
# monkeypatch is used here to reset the value to the PIL default.
# This value does not matter, other than it is "large"
# to show that settings['max_img_pixels'] works.
monkeypatch.setattr("PIL.Image.MAX_IMAGE_PIXELS", 100_000_000)
settings["source"] = os.path.join(settings["source"], "dir2")
settings["destination"] = str(tmpdir)
settings["max_img_pixels"] = 5000
logger = logging.getLogger("sigal")
logger.setLevel(logging.DEBUG)
try:
with pytest.raises(PILImage.DecompressionBombError):
gal = Gallery(settings, ncpu=1)
gal.build()
settings["max_img_pixels"] = 100_000_000
gal = Gallery(settings, ncpu=1)
gal.build()
finally:
logger.setLevel(logging.INFO)
def test_empty_dirs(settings):
gal = Gallery(settings, ncpu=1)
assert "empty" not in gal.albums
assert "dir1/empty" not in gal.albums
def test_ignores(settings, tmp_path):
settings["source"] = os.path.join(settings["source"], "dir1")
settings["destination"] = str(tmp_path)
settings["ignore_directories"] = ["*test2"]
settings["ignore_files"] = ["*.gif", "*CMB_*"]
gal = Gallery(settings, ncpu=1)
gal.build()
assert not (tmp_path / "test2").exists()
assert not (tmp_path / "test1" / "example.gif").exists()
assert not (tmp_path / "test1" / "CMB_Timeline300_no_WMAP.jpg").exists()
@pytest.mark.parametrize("thumbnail", ["outdoor.heic", "outdoor.tn.jpg"])
def test_thumbnail_with_img_format(settings, tmp_path, thumbnail):
"""Test that outdoor.heic is correctly converted as jpg and used as thumbnail"""
src_path = tmp_path / "pictures"
src_path.mkdir()
shutil.copytree(
os.path.join(settings["source"], "dir1", "test1"), src_path / "test1"
)
desc = (src_path / "test1" / "index.md").read_text()
desc = desc.replace("Thumbnail: 11.jpg", f"Thumbnail: {thumbnail}")
(src_path / "test1" / "index.md").write_text(desc)
settings["img_format"] = "JPEG"
settings["thumb_dir"] = ""
settings["source"] = str(src_path)
settings["destination"] = str(tmp_path / "build")
gal = Gallery(settings, ncpu=1)
gal.build()
assert (tmp_path / "build" / "test1" / "outdoor.tn.jpg").is_file()
index = (tmp_path / "build" / "index.html").read_text()
4 weeks ago
assert 'src="./test1/outdoor.tn.jpg" class="album_thumb"' in index