mirror of
				https://github.com/python-pillow/Pillow.git
				synced 2025-11-04 09:57:43 +03:00 
			
		
		
		
	
		
			
				
	
	
		
			436 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			436 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"
 | 
						|
 | 
						|
 | 
						|
@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"
 |