mirror of
https://github.com/python-pillow/Pillow.git
synced 2025-10-24 12:41:11 +03:00
444 lines
15 KiB
Python
444 lines
15 KiB
Python
from __future__ import annotations
|
|
|
|
import io
|
|
from pathlib import Path
|
|
|
|
import pytest
|
|
|
|
from PIL import EpsImagePlugin, Image, UnidentifiedImageError, features
|
|
|
|
from .helper import (
|
|
assert_image_equal_tofile,
|
|
assert_image_similar,
|
|
assert_image_similar_tofile,
|
|
hopper,
|
|
is_win32,
|
|
mark_if_feature_version,
|
|
skip_unless_feature,
|
|
timeout_unless_slower_valgrind,
|
|
)
|
|
|
|
HAS_GHOSTSCRIPT = EpsImagePlugin.has_ghostscript()
|
|
|
|
# Our two EPS test files (they are identical except for their bounding boxes)
|
|
FILE1 = "Tests/images/eps/zero_bb.eps"
|
|
FILE2 = "Tests/images/eps/non_zero_bb.eps"
|
|
|
|
# Due to palletization, we'll need to convert these to RGB after load
|
|
FILE1_COMPARE = "Tests/images/eps/zero_bb.png"
|
|
FILE1_COMPARE_SCALE2 = "Tests/images/eps/zero_bb_scale2.png"
|
|
|
|
FILE2_COMPARE = "Tests/images/eps/non_zero_bb.png"
|
|
FILE2_COMPARE_SCALE2 = "Tests/images/eps/non_zero_bb_scale2.png"
|
|
|
|
# EPS test files with binary preview
|
|
FILE3 = "Tests/images/eps/binary_preview_map.eps"
|
|
|
|
# Three unsigned 32bit little-endian values:
|
|
# 0xC6D3D0C5 magic number
|
|
# byte position of start of postscript section (12)
|
|
# byte length of postscript section (0)
|
|
# this byte length isn't valid, but we don't read it
|
|
simple_binary_header = b"\xc5\xd0\xd3\xc6\x0c\x00\x00\x00\x00\x00\x00\x00"
|
|
|
|
# taken from page 8 of the specification
|
|
# https://web.archive.org/web/20220120164601/https://www.adobe.com/content/dam/acom/en/devnet/actionscript/articles/5002.EPSF_Spec.pdf
|
|
simple_eps_file = (
|
|
b"%!PS-Adobe-3.0 EPSF-3.0",
|
|
b"%%BoundingBox: 5 5 105 105",
|
|
b"10 setlinewidth",
|
|
b"10 10 moveto",
|
|
b"0 90 rlineto 90 0 rlineto 0 -90 rlineto closepath",
|
|
b"stroke",
|
|
)
|
|
simple_eps_file_with_comments = (
|
|
simple_eps_file[:1]
|
|
+ (
|
|
b"%%Comment1: Some Value",
|
|
b"%%SecondComment: Another Value",
|
|
)
|
|
+ simple_eps_file[1:]
|
|
)
|
|
simple_eps_file_without_version = simple_eps_file[1:]
|
|
simple_eps_file_without_boundingbox = simple_eps_file[:1] + simple_eps_file[2:]
|
|
simple_eps_file_with_invalid_boundingbox = (
|
|
simple_eps_file[:1] + (b"%%BoundingBox: a b c d",) + simple_eps_file[2:]
|
|
)
|
|
simple_eps_file_with_invalid_boundingbox_valid_imagedata = (
|
|
simple_eps_file_with_invalid_boundingbox + (b"%ImageData: 100 100 8 3",)
|
|
)
|
|
simple_eps_file_with_long_ascii_comment = (
|
|
simple_eps_file[:2] + (b"%%Comment: " + b"X" * 300,) + simple_eps_file[2:]
|
|
)
|
|
simple_eps_file_with_long_binary_data = (
|
|
simple_eps_file[:2]
|
|
+ (
|
|
b"%%BeginBinary: 300",
|
|
b"\0" * 300,
|
|
b"%%EndBinary",
|
|
)
|
|
+ simple_eps_file[2:]
|
|
)
|
|
|
|
|
|
@pytest.mark.skipif(not HAS_GHOSTSCRIPT, reason="Ghostscript not available")
|
|
@pytest.mark.parametrize("filename, size", ((FILE1, (460, 352)), (FILE2, (360, 252))))
|
|
@pytest.mark.parametrize("scale", (1, 2))
|
|
def test_sanity(filename: str, size: tuple[int, int], scale: int) -> None:
|
|
expected_size = tuple(s * scale for s in size)
|
|
with Image.open(filename) as image:
|
|
assert isinstance(image, EpsImagePlugin.EpsImageFile)
|
|
|
|
image.load(scale=scale)
|
|
assert image.mode == "RGB"
|
|
assert image.size == expected_size
|
|
assert image.format == "EPS"
|
|
|
|
|
|
@pytest.mark.skipif(not HAS_GHOSTSCRIPT, reason="Ghostscript not available")
|
|
def test_load() -> None:
|
|
with Image.open(FILE1) as im:
|
|
px = im.load()
|
|
assert px is not None
|
|
assert px[0, 0] == (255, 255, 255)
|
|
|
|
# Test again now that it has already been loaded once
|
|
px = im.load()
|
|
assert px is not None
|
|
assert px[0, 0] == (255, 255, 255)
|
|
|
|
|
|
def test_binary() -> None:
|
|
if HAS_GHOSTSCRIPT:
|
|
assert EpsImagePlugin.gs_binary is not None
|
|
else:
|
|
assert EpsImagePlugin.gs_binary is False
|
|
|
|
if not is_win32():
|
|
assert EpsImagePlugin.gs_windows_binary is None
|
|
elif not HAS_GHOSTSCRIPT:
|
|
assert EpsImagePlugin.gs_windows_binary is False
|
|
else:
|
|
assert EpsImagePlugin.gs_windows_binary is not None
|
|
|
|
|
|
def test_invalid_file() -> None:
|
|
invalid_file = "Tests/images/flower.jpg"
|
|
with pytest.raises(SyntaxError):
|
|
EpsImagePlugin.EpsImageFile(invalid_file)
|
|
|
|
|
|
def test_binary_header_only() -> None:
|
|
data = io.BytesIO(simple_binary_header)
|
|
with pytest.raises(SyntaxError, match='EPS header missing "%!PS-Adobe" comment'):
|
|
EpsImagePlugin.EpsImageFile(data)
|
|
|
|
|
|
@pytest.mark.parametrize("prefix", (b"", simple_binary_header))
|
|
def test_simple_eps_file(prefix: bytes) -> None:
|
|
data = io.BytesIO(prefix + b"\n".join(simple_eps_file))
|
|
with Image.open(data) as img:
|
|
assert img.mode == "RGB"
|
|
assert img.size == (100, 100)
|
|
assert img.format == "EPS"
|
|
|
|
|
|
@pytest.mark.parametrize("prefix", (b"", simple_binary_header))
|
|
def test_missing_version_comment(prefix: bytes) -> None:
|
|
data = io.BytesIO(prefix + b"\n".join(simple_eps_file_without_version))
|
|
with pytest.raises(SyntaxError):
|
|
EpsImagePlugin.EpsImageFile(data)
|
|
|
|
|
|
@pytest.mark.parametrize("prefix", (b"", simple_binary_header))
|
|
def test_missing_boundingbox_comment(prefix: bytes) -> None:
|
|
data = io.BytesIO(prefix + b"\n".join(simple_eps_file_without_boundingbox))
|
|
with pytest.raises(SyntaxError, match='EPS header missing "%%BoundingBox" comment'):
|
|
EpsImagePlugin.EpsImageFile(data)
|
|
|
|
|
|
@pytest.mark.parametrize("prefix", (b"", simple_binary_header))
|
|
@pytest.mark.parametrize(
|
|
"file_lines",
|
|
(
|
|
simple_eps_file_with_invalid_boundingbox,
|
|
simple_eps_file_with_invalid_boundingbox_valid_imagedata,
|
|
),
|
|
)
|
|
def test_invalid_boundingbox_comment(
|
|
prefix: bytes, file_lines: tuple[bytes, ...]
|
|
) -> None:
|
|
data = io.BytesIO(prefix + b"\n".join(file_lines))
|
|
with pytest.raises(OSError, match="cannot determine EPS bounding box"):
|
|
EpsImagePlugin.EpsImageFile(data)
|
|
|
|
|
|
@pytest.mark.parametrize("prefix", (b"", simple_binary_header))
|
|
def test_ascii_comment_too_long(prefix: bytes) -> None:
|
|
data = io.BytesIO(prefix + b"\n".join(simple_eps_file_with_long_ascii_comment))
|
|
with pytest.raises(SyntaxError, match="not an EPS file"):
|
|
EpsImagePlugin.EpsImageFile(data)
|
|
|
|
|
|
@pytest.mark.parametrize("prefix", (b"", simple_binary_header))
|
|
def test_long_binary_data(prefix: bytes) -> None:
|
|
data = io.BytesIO(prefix + b"\n".join(simple_eps_file_with_long_binary_data))
|
|
EpsImagePlugin.EpsImageFile(data)
|
|
|
|
|
|
@pytest.mark.skipif(not HAS_GHOSTSCRIPT, reason="Ghostscript not available")
|
|
@pytest.mark.parametrize("prefix", (b"", simple_binary_header))
|
|
def test_load_long_binary_data(prefix: bytes) -> None:
|
|
data = io.BytesIO(prefix + b"\n".join(simple_eps_file_with_long_binary_data))
|
|
with Image.open(data) as img:
|
|
img.load()
|
|
assert img.mode == "1"
|
|
assert img.size == (100, 100)
|
|
assert img.format == "EPS"
|
|
|
|
|
|
def test_begin_binary() -> None:
|
|
with open("Tests/images/eps/binary_preview_map.eps", "rb") as fp:
|
|
data = bytearray(fp.read())
|
|
data[76875 : 76875 + 11] = b"%" * 11
|
|
with Image.open(io.BytesIO(data)) as img:
|
|
assert img.size == (399, 480)
|
|
|
|
|
|
@mark_if_feature_version(
|
|
pytest.mark.valgrind_known_error, "libjpeg_turbo", "2.0", reason="Known Failing"
|
|
)
|
|
@pytest.mark.skipif(not HAS_GHOSTSCRIPT, reason="Ghostscript not available")
|
|
def test_cmyk() -> None:
|
|
with Image.open("Tests/images/eps/pil_sample_cmyk.eps") as cmyk_image:
|
|
assert cmyk_image.mode == "CMYK"
|
|
assert cmyk_image.size == (100, 100)
|
|
assert cmyk_image.format == "EPS"
|
|
|
|
cmyk_image.load()
|
|
assert cmyk_image.mode == "RGB"
|
|
|
|
if features.check("jpg"):
|
|
assert_image_similar_tofile(
|
|
cmyk_image, "Tests/images/pil_sample_rgb.jpg", 10
|
|
)
|
|
|
|
|
|
@pytest.mark.skipif(not HAS_GHOSTSCRIPT, reason="Ghostscript not available")
|
|
def test_showpage() -> None:
|
|
# See https://github.com/python-pillow/Pillow/issues/2615
|
|
with Image.open("Tests/images/eps/reqd_showpage.eps") as plot_image:
|
|
with Image.open("Tests/images/eps/reqd_showpage.png") as target:
|
|
# should not crash/hang
|
|
plot_image.load()
|
|
# fonts could be slightly different
|
|
assert_image_similar(plot_image, target, 6)
|
|
|
|
|
|
@pytest.mark.skipif(not HAS_GHOSTSCRIPT, reason="Ghostscript not available")
|
|
def test_transparency() -> None:
|
|
with Image.open("Tests/images/eps/reqd_showpage.eps") as plot_image:
|
|
assert isinstance(plot_image, EpsImagePlugin.EpsImageFile)
|
|
|
|
plot_image.load(transparency=True)
|
|
assert plot_image.mode == "RGBA"
|
|
|
|
with Image.open("Tests/images/eps/reqd_showpage_transparency.png") as target:
|
|
# fonts could be slightly different
|
|
assert_image_similar(plot_image, target, 6)
|
|
|
|
|
|
@pytest.mark.skipif(not HAS_GHOSTSCRIPT, reason="Ghostscript not available")
|
|
def test_file_object(tmp_path: Path) -> None:
|
|
# issue 479
|
|
with Image.open(FILE1) as image1:
|
|
with open(tmp_path / "temp.eps", "wb") as fh:
|
|
image1.save(fh, "EPS")
|
|
|
|
|
|
@pytest.mark.skipif(not HAS_GHOSTSCRIPT, reason="Ghostscript not available")
|
|
def test_bytesio_object() -> None:
|
|
with open(FILE1, "rb") as f:
|
|
img_bytes = io.BytesIO(f.read())
|
|
|
|
with Image.open(img_bytes) as img:
|
|
img.load()
|
|
|
|
with Image.open(FILE1_COMPARE) as image1_scale1_compare:
|
|
image1_scale1_compare = image1_scale1_compare.convert("RGB")
|
|
image1_scale1_compare.load()
|
|
assert_image_similar(img, image1_scale1_compare, 5)
|
|
|
|
|
|
@pytest.mark.skipif(not HAS_GHOSTSCRIPT, reason="Ghostscript not available")
|
|
@pytest.mark.parametrize(
|
|
# These images have an "ImageData" descriptor.
|
|
"filename",
|
|
(
|
|
"Tests/images/eps/1.eps",
|
|
"Tests/images/eps/1_boundingbox_after_imagedata.eps",
|
|
"Tests/images/eps/1_second_imagedata.eps",
|
|
),
|
|
)
|
|
def test_1(filename: str) -> None:
|
|
with Image.open(filename) as im:
|
|
assert_image_equal_tofile(im, "Tests/images/eps/1.bmp")
|
|
|
|
|
|
def test_image_mode_not_supported(tmp_path: Path) -> None:
|
|
im = hopper("RGBA")
|
|
tmpfile = tmp_path / "temp.eps"
|
|
with pytest.raises(ValueError):
|
|
im.save(tmpfile)
|
|
|
|
|
|
@pytest.mark.skipif(not HAS_GHOSTSCRIPT, reason="Ghostscript not available")
|
|
@skip_unless_feature("zlib")
|
|
def test_render_scale1() -> None:
|
|
# We need png support for these render test
|
|
|
|
# Zero bounding box
|
|
with Image.open(FILE1) as image1_scale1:
|
|
image1_scale1.load()
|
|
with Image.open(FILE1_COMPARE) as image1_scale1_compare:
|
|
image1_scale1_compare = image1_scale1_compare.convert("RGB")
|
|
image1_scale1_compare.load()
|
|
assert_image_similar(image1_scale1, image1_scale1_compare, 5)
|
|
|
|
# Non-zero bounding box
|
|
with Image.open(FILE2) as image2_scale1:
|
|
image2_scale1.load()
|
|
with Image.open(FILE2_COMPARE) as image2_scale1_compare:
|
|
image2_scale1_compare = image2_scale1_compare.convert("RGB")
|
|
image2_scale1_compare.load()
|
|
assert_image_similar(image2_scale1, image2_scale1_compare, 10)
|
|
|
|
|
|
@pytest.mark.skipif(not HAS_GHOSTSCRIPT, reason="Ghostscript not available")
|
|
@skip_unless_feature("zlib")
|
|
def test_render_scale2() -> None:
|
|
# We need png support for these render test
|
|
|
|
# Zero bounding box
|
|
with Image.open(FILE1) as image1_scale2:
|
|
assert isinstance(image1_scale2, EpsImagePlugin.EpsImageFile)
|
|
image1_scale2.load(scale=2)
|
|
with Image.open(FILE1_COMPARE_SCALE2) as image1_scale2_compare:
|
|
image1_scale2_compare = image1_scale2_compare.convert("RGB")
|
|
image1_scale2_compare.load()
|
|
assert_image_similar(image1_scale2, image1_scale2_compare, 5)
|
|
|
|
# Non-zero bounding box
|
|
with Image.open(FILE2) as image2_scale2:
|
|
assert isinstance(image2_scale2, EpsImagePlugin.EpsImageFile)
|
|
image2_scale2.load(scale=2)
|
|
with Image.open(FILE2_COMPARE_SCALE2) as image2_scale2_compare:
|
|
image2_scale2_compare = image2_scale2_compare.convert("RGB")
|
|
image2_scale2_compare.load()
|
|
assert_image_similar(image2_scale2, image2_scale2_compare, 10)
|
|
|
|
|
|
@pytest.mark.skipif(not HAS_GHOSTSCRIPT, reason="Ghostscript not available")
|
|
@pytest.mark.parametrize(
|
|
"filename", (FILE1, FILE2, "Tests/images/eps/illu10_preview.eps")
|
|
)
|
|
def test_resize(filename: str) -> None:
|
|
with Image.open(filename) as im:
|
|
new_size = (100, 100)
|
|
im = im.resize(new_size)
|
|
assert im.size == new_size
|
|
|
|
|
|
@pytest.mark.skipif(not HAS_GHOSTSCRIPT, reason="Ghostscript not available")
|
|
@pytest.mark.parametrize("filename", (FILE1, FILE2))
|
|
def test_thumbnail(filename: str) -> None:
|
|
# Issue #619
|
|
with Image.open(filename) as im:
|
|
new_size = (100, 100)
|
|
im.thumbnail(new_size)
|
|
assert max(im.size) == max(new_size)
|
|
|
|
|
|
def test_read_binary_preview() -> None:
|
|
# Issue 302
|
|
# open image with binary preview
|
|
with Image.open(FILE3):
|
|
pass
|
|
|
|
|
|
@pytest.mark.parametrize("prefix", (b"", simple_binary_header))
|
|
@pytest.mark.parametrize(
|
|
"line_ending",
|
|
(b"\r\n", b"\n", b"\n\r", b"\r"),
|
|
)
|
|
def test_readline(prefix: bytes, line_ending: bytes) -> None:
|
|
simple_file = prefix + line_ending.join(simple_eps_file_with_comments)
|
|
data = io.BytesIO(simple_file)
|
|
test_file = EpsImagePlugin.EpsImageFile(data)
|
|
assert test_file.info["Comment1"] == "Some Value"
|
|
assert test_file.info["SecondComment"] == "Another Value"
|
|
assert test_file.size == (100, 100)
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"filename",
|
|
(
|
|
"Tests/images/eps/illu10_no_preview.eps",
|
|
"Tests/images/eps/illu10_preview.eps",
|
|
"Tests/images/eps/illuCS6_no_preview.eps",
|
|
"Tests/images/eps/illuCS6_preview.eps",
|
|
),
|
|
)
|
|
def test_open_eps(filename: str) -> None:
|
|
# https://github.com/python-pillow/Pillow/issues/1104
|
|
with Image.open(filename) as img:
|
|
assert img.mode == "RGB"
|
|
|
|
|
|
@pytest.mark.skipif(not HAS_GHOSTSCRIPT, reason="Ghostscript not available")
|
|
def test_emptyline() -> None:
|
|
# Test file includes an empty line in the header data
|
|
emptyline_file = "Tests/images/eps/zero_bb_emptyline.eps"
|
|
|
|
with Image.open(emptyline_file) as image:
|
|
image.load()
|
|
assert image.mode == "RGB"
|
|
assert image.size == (460, 352)
|
|
assert image.format == "EPS"
|
|
|
|
|
|
@timeout_unless_slower_valgrind(5)
|
|
@pytest.mark.parametrize(
|
|
"test_file",
|
|
["Tests/images/eps/timeout-d675703545fee17acab56e5fec644c19979175de.eps"],
|
|
)
|
|
def test_timeout(test_file: str) -> None:
|
|
with open(test_file, "rb") as f:
|
|
with pytest.raises(UnidentifiedImageError):
|
|
with Image.open(f):
|
|
pass
|
|
|
|
|
|
def test_bounding_box_in_trailer() -> None:
|
|
# Check bounding boxes are parsed in the same way
|
|
# when specified in the header and the trailer
|
|
with (
|
|
Image.open("Tests/images/eps/zero_bb_trailer.eps") as trailer_image,
|
|
Image.open(FILE1) as header_image,
|
|
):
|
|
assert trailer_image.size == header_image.size
|
|
|
|
|
|
def test_eof_before_bounding_box() -> None:
|
|
with pytest.raises(OSError):
|
|
with Image.open("Tests/images/eps/zero_bb_eof_before_boundingbox.eps"):
|
|
pass
|
|
|
|
|
|
def test_invalid_data_after_eof() -> None:
|
|
with open("Tests/images/eps/illuCS6_preview.eps", "rb") as f:
|
|
img_bytes = io.BytesIO(f.read() + b"\r\n%" + (b" " * 255))
|
|
|
|
with Image.open(img_bytes) as img:
|
|
assert img.mode == "RGB"
|