Merge pull request #4391 from radarhere/pytest

Convert various tests to pytest style
This commit is contained in:
Hugo van Kemenade 2020-01-28 18:22:11 +02:00 committed by GitHub
commit f6e067b3b4
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
25 changed files with 1092 additions and 1064 deletions

View File

@ -1,10 +1,8 @@
import io import io
import unittest
import pytest
from PIL import features from PIL import features
from .helper import PillowTestCase
try: try:
from PIL import _webp from PIL import _webp
@ -13,78 +11,85 @@ except ImportError:
HAVE_WEBP = False HAVE_WEBP = False
class TestFeatures(PillowTestCase): def test_check():
def test_check(self): # Check the correctness of the convenience function
# Check the correctness of the convenience function for module in features.modules:
for module in features.modules: assert features.check_module(module) == features.check(module)
self.assertEqual(features.check_module(module), features.check(module)) for codec in features.codecs:
for codec in features.codecs: assert features.check_codec(codec) == features.check(codec)
self.assertEqual(features.check_codec(codec), features.check(codec)) for feature in features.features:
for feature in features.features: assert features.check_feature(feature) == features.check(feature)
self.assertEqual(features.check_feature(feature), features.check(feature))
@unittest.skipUnless(HAVE_WEBP, "WebP not available")
def test_webp_transparency(self):
self.assertEqual(
features.check("transp_webp"), not _webp.WebPDecoderBuggyAlpha()
)
self.assertEqual(features.check("transp_webp"), _webp.HAVE_TRANSPARENCY)
@unittest.skipUnless(HAVE_WEBP, "WebP not available") @pytest.mark.skipif(not HAVE_WEBP, reason="WebP not available")
def test_webp_mux(self): def test_webp_transparency():
self.assertEqual(features.check("webp_mux"), _webp.HAVE_WEBPMUX) assert features.check("transp_webp") != _webp.WebPDecoderBuggyAlpha()
assert features.check("transp_webp") == _webp.HAVE_TRANSPARENCY
@unittest.skipUnless(HAVE_WEBP, "WebP not available")
def test_webp_anim(self):
self.assertEqual(features.check("webp_anim"), _webp.HAVE_WEBPANIM)
def test_check_modules(self): @pytest.mark.skipif(not HAVE_WEBP, reason="WebP not available")
for feature in features.modules: def test_webp_mux():
self.assertIn(features.check_module(feature), [True, False]) assert features.check("webp_mux") == _webp.HAVE_WEBPMUX
for feature in features.codecs:
self.assertIn(features.check_codec(feature), [True, False])
def test_supported_modules(self):
self.assertIsInstance(features.get_supported_modules(), list)
self.assertIsInstance(features.get_supported_codecs(), list)
self.assertIsInstance(features.get_supported_features(), list)
self.assertIsInstance(features.get_supported(), list)
def test_unsupported_codec(self): @pytest.mark.skipif(not HAVE_WEBP, reason="WebP not available")
# Arrange def test_webp_anim():
codec = "unsupported_codec" assert features.check("webp_anim") == _webp.HAVE_WEBPANIM
# Act / Assert
self.assertRaises(ValueError, features.check_codec, codec)
def test_unsupported_module(self):
# Arrange
module = "unsupported_module"
# Act / Assert
self.assertRaises(ValueError, features.check_module, module)
def test_pilinfo(self): def test_check_modules():
buf = io.StringIO() for feature in features.modules:
features.pilinfo(buf) assert features.check_module(feature) in [True, False]
out = buf.getvalue() for feature in features.codecs:
lines = out.splitlines() assert features.check_codec(feature) in [True, False]
self.assertEqual(lines[0], "-" * 68)
self.assertTrue(lines[1].startswith("Pillow "))
self.assertTrue(lines[2].startswith("Python ")) def test_supported_modules():
lines = lines[3:] assert isinstance(features.get_supported_modules(), list)
while lines[0].startswith(" "): assert isinstance(features.get_supported_codecs(), list)
lines = lines[1:] assert isinstance(features.get_supported_features(), list)
self.assertEqual(lines[0], "-" * 68) assert isinstance(features.get_supported(), list)
self.assertTrue(lines[1].startswith("Python modules loaded from "))
self.assertTrue(lines[2].startswith("Binary modules loaded from "))
self.assertEqual(lines[3], "-" * 68) def test_unsupported_codec():
jpeg = ( # Arrange
"\n" codec = "unsupported_codec"
+ "-" * 68 # Act / Assert
+ "\n" with pytest.raises(ValueError):
+ "JPEG image/jpeg\n" features.check_codec(codec)
+ "Extensions: .jfif, .jpe, .jpeg, .jpg\n"
+ "Features: open, save\n"
+ "-" * 68 def test_unsupported_module():
+ "\n" # Arrange
) module = "unsupported_module"
self.assertIn(jpeg, out) # Act / Assert
with pytest.raises(ValueError):
features.check_module(module)
def test_pilinfo():
buf = io.StringIO()
features.pilinfo(buf)
out = buf.getvalue()
lines = out.splitlines()
assert lines[0] == "-" * 68
assert lines[1].startswith("Pillow ")
assert lines[2].startswith("Python ")
lines = lines[3:]
while lines[0].startswith(" "):
lines = lines[1:]
assert lines[0] == "-" * 68
assert lines[1].startswith("Python modules loaded from ")
assert lines[2].startswith("Binary modules loaded from ")
assert lines[3] == "-" * 68
jpeg = (
"\n"
+ "-" * 68
+ "\n"
+ "JPEG image/jpeg\n"
+ "Extensions: .jfif, .jpe, .jpeg, .jpg\n"
+ "Features: open, save\n"
+ "-" * 68
+ "\n"
)
assert jpeg in out

View File

@ -1,28 +1,29 @@
import pytest
from PIL import CurImagePlugin, Image from PIL import CurImagePlugin, Image
from .helper import PillowTestCase
TEST_FILE = "Tests/images/deerstalker.cur" TEST_FILE = "Tests/images/deerstalker.cur"
class TestFileCur(PillowTestCase): def test_sanity():
def test_sanity(self): with Image.open(TEST_FILE) as im:
with Image.open(TEST_FILE) as im: assert im.size == (32, 32)
self.assertEqual(im.size, (32, 32)) assert isinstance(im, CurImagePlugin.CurImageFile)
self.assertIsInstance(im, CurImagePlugin.CurImageFile) # Check some pixel colors to ensure image is loaded properly
# Check some pixel colors to ensure image is loaded properly assert im.getpixel((10, 1)) == (0, 0, 0, 0)
self.assertEqual(im.getpixel((10, 1)), (0, 0, 0, 0)) assert im.getpixel((11, 1)) == (253, 254, 254, 1)
self.assertEqual(im.getpixel((11, 1)), (253, 254, 254, 1)) assert im.getpixel((16, 16)) == (84, 87, 86, 255)
self.assertEqual(im.getpixel((16, 16)), (84, 87, 86, 255))
def test_invalid_file(self):
invalid_file = "Tests/images/flower.jpg"
self.assertRaises(SyntaxError, CurImagePlugin.CurImageFile, invalid_file) def test_invalid_file():
invalid_file = "Tests/images/flower.jpg"
no_cursors_file = "Tests/images/no_cursors.cur" with pytest.raises(SyntaxError):
CurImagePlugin.CurImageFile(invalid_file)
cur = CurImagePlugin.CurImageFile(TEST_FILE) no_cursors_file = "Tests/images/no_cursors.cur"
cur.fp.close()
with open(no_cursors_file, "rb") as cur.fp: cur = CurImagePlugin.CurImageFile(TEST_FILE)
self.assertRaises(TypeError, cur._open) cur.fp.close()
with open(no_cursors_file, "rb") as cur.fp:
with pytest.raises(TypeError):
cur._open()

View File

@ -1,46 +1,47 @@
import pytest
from PIL import FitsStubImagePlugin, Image from PIL import FitsStubImagePlugin, Image
from .helper import PillowTestCase
TEST_FILE = "Tests/images/hopper.fits" TEST_FILE = "Tests/images/hopper.fits"
class TestFileFitsStub(PillowTestCase): def test_open():
def test_open(self): # Act
# Act with Image.open(TEST_FILE) as im:
with Image.open(TEST_FILE) as im:
# Assert # Assert
self.assertEqual(im.format, "FITS") assert im.format == "FITS"
# Dummy data from the stub # Dummy data from the stub
self.assertEqual(im.mode, "F") assert im.mode == "F"
self.assertEqual(im.size, (1, 1)) assert im.size == (1, 1)
def test_invalid_file(self):
# Arrange
invalid_file = "Tests/images/flower.jpg"
# Act / Assert def test_invalid_file():
self.assertRaises( # Arrange
SyntaxError, FitsStubImagePlugin.FITSStubImageFile, invalid_file invalid_file = "Tests/images/flower.jpg"
)
def test_load(self): # Act / Assert
# Arrange with pytest.raises(SyntaxError):
with Image.open(TEST_FILE) as im: FitsStubImagePlugin.FITSStubImageFile(invalid_file)
# Act / Assert: stub cannot load without an implemented handler
self.assertRaises(IOError, im.load)
def test_save(self): def test_load():
# Arrange # Arrange
with Image.open(TEST_FILE) as im: with Image.open(TEST_FILE) as im:
dummy_fp = None
dummy_filename = "dummy.filename"
# Act / Assert: stub cannot save without an implemented handler # Act / Assert: stub cannot load without an implemented handler
self.assertRaises(IOError, im.save, dummy_filename) with pytest.raises(IOError):
self.assertRaises( im.load()
IOError, FitsStubImagePlugin._save, im, dummy_fp, dummy_filename
)
def test_save():
# Arrange
with Image.open(TEST_FILE) as im:
dummy_fp = None
dummy_filename = "dummy.filename"
# Act / Assert: stub cannot save without an implemented handler
with pytest.raises(IOError):
im.save(dummy_filename)
with pytest.raises(IOError):
FitsStubImagePlugin._save(im, dummy_fp, dummy_filename)

View File

@ -1,42 +1,46 @@
import pytest
from PIL import GribStubImagePlugin, Image from PIL import GribStubImagePlugin, Image
from .helper import PillowTestCase, hopper from .helper import hopper
TEST_FILE = "Tests/images/WAlaska.wind.7days.grb" TEST_FILE = "Tests/images/WAlaska.wind.7days.grb"
class TestFileGribStub(PillowTestCase): def test_open():
def test_open(self): # Act
# Act with Image.open(TEST_FILE) as im:
with Image.open(TEST_FILE) as im:
# Assert # Assert
self.assertEqual(im.format, "GRIB") assert im.format == "GRIB"
# Dummy data from the stub # Dummy data from the stub
self.assertEqual(im.mode, "F") assert im.mode == "F"
self.assertEqual(im.size, (1, 1)) assert im.size == (1, 1)
def test_invalid_file(self):
# Arrange
invalid_file = "Tests/images/flower.jpg"
# Act / Assert def test_invalid_file():
self.assertRaises( # Arrange
SyntaxError, GribStubImagePlugin.GribStubImageFile, invalid_file invalid_file = "Tests/images/flower.jpg"
)
def test_load(self): # Act / Assert
# Arrange with pytest.raises(SyntaxError):
with Image.open(TEST_FILE) as im: GribStubImagePlugin.GribStubImageFile(invalid_file)
# Act / Assert: stub cannot load without an implemented handler
self.assertRaises(IOError, im.load)
def test_save(self): def test_load():
# Arrange # Arrange
im = hopper() with Image.open(TEST_FILE) as im:
tmpfile = self.tempfile("temp.grib")
# Act / Assert: stub cannot save without an implemented handler # Act / Assert: stub cannot load without an implemented handler
self.assertRaises(IOError, im.save, tmpfile) with pytest.raises(IOError):
im.load()
def test_save(tmp_path):
# Arrange
im = hopper()
tmpfile = str(tmp_path / "temp.grib")
# Act / Assert: stub cannot save without an implemented handler
with pytest.raises(IOError):
im.save(tmpfile)

View File

@ -1,46 +1,47 @@
import pytest
from PIL import Hdf5StubImagePlugin, Image from PIL import Hdf5StubImagePlugin, Image
from .helper import PillowTestCase
TEST_FILE = "Tests/images/hdf5.h5" TEST_FILE = "Tests/images/hdf5.h5"
class TestFileHdf5Stub(PillowTestCase): def test_open():
def test_open(self): # Act
# Act with Image.open(TEST_FILE) as im:
with Image.open(TEST_FILE) as im:
# Assert # Assert
self.assertEqual(im.format, "HDF5") assert im.format == "HDF5"
# Dummy data from the stub # Dummy data from the stub
self.assertEqual(im.mode, "F") assert im.mode == "F"
self.assertEqual(im.size, (1, 1)) assert im.size == (1, 1)
def test_invalid_file(self):
# Arrange
invalid_file = "Tests/images/flower.jpg"
# Act / Assert def test_invalid_file():
self.assertRaises( # Arrange
SyntaxError, Hdf5StubImagePlugin.HDF5StubImageFile, invalid_file invalid_file = "Tests/images/flower.jpg"
)
def test_load(self): # Act / Assert
# Arrange with pytest.raises(SyntaxError):
with Image.open(TEST_FILE) as im: Hdf5StubImagePlugin.HDF5StubImageFile(invalid_file)
# Act / Assert: stub cannot load without an implemented handler
self.assertRaises(IOError, im.load)
def test_save(self): def test_load():
# Arrange # Arrange
with Image.open(TEST_FILE) as im: with Image.open(TEST_FILE) as im:
dummy_fp = None
dummy_filename = "dummy.filename"
# Act / Assert: stub cannot save without an implemented handler # Act / Assert: stub cannot load without an implemented handler
self.assertRaises(IOError, im.save, dummy_filename) with pytest.raises(IOError):
self.assertRaises( im.load()
IOError, Hdf5StubImagePlugin._save, im, dummy_fp, dummy_filename
)
def test_save():
# Arrange
with Image.open(TEST_FILE) as im:
dummy_fp = None
dummy_filename = "dummy.filename"
# Act / Assert: stub cannot save without an implemented handler
with pytest.raises(IOError):
im.save(dummy_filename)
with pytest.raises(IOError):
Hdf5StubImagePlugin._save(im, dummy_fp, dummy_filename)

View File

@ -3,66 +3,69 @@ from io import StringIO
from PIL import Image, IptcImagePlugin from PIL import Image, IptcImagePlugin
from .helper import PillowTestCase, hopper from .helper import hopper
TEST_FILE = "Tests/images/iptc.jpg" TEST_FILE = "Tests/images/iptc.jpg"
class TestFileIptc(PillowTestCase): def test_getiptcinfo_jpg_none():
def test_getiptcinfo_jpg_none(self): # Arrange
# Arrange with hopper() as im:
with hopper() as im:
# Act
iptc = IptcImagePlugin.getiptcinfo(im)
# Assert
self.assertIsNone(iptc)
def test_getiptcinfo_jpg_found(self):
# Arrange
with Image.open(TEST_FILE) as im:
# Act
iptc = IptcImagePlugin.getiptcinfo(im)
# Assert
self.assertIsInstance(iptc, dict)
self.assertEqual(iptc[(2, 90)], b"Budapest")
self.assertEqual(iptc[(2, 101)], b"Hungary")
def test_getiptcinfo_tiff_none(self):
# Arrange
with Image.open("Tests/images/hopper.tif") as im:
# Act
iptc = IptcImagePlugin.getiptcinfo(im)
# Assert
self.assertIsNone(iptc)
def test_i(self):
# Arrange
c = b"a"
# Act # Act
ret = IptcImagePlugin.i(c) iptc = IptcImagePlugin.getiptcinfo(im)
# Assert # Assert
self.assertEqual(ret, 97) assert iptc is None
def test_dump(self):
# Arrange def test_getiptcinfo_jpg_found():
c = b"abc" # Arrange
# Temporarily redirect stdout with Image.open(TEST_FILE) as im:
old_stdout = sys.stdout
sys.stdout = mystdout = StringIO()
# Act # Act
IptcImagePlugin.dump(c) iptc = IptcImagePlugin.getiptcinfo(im)
# Reset stdout # Assert
sys.stdout = old_stdout assert isinstance(iptc, dict)
assert iptc[(2, 90)] == b"Budapest"
assert iptc[(2, 101)] == b"Hungary"
# Assert
self.assertEqual(mystdout.getvalue(), "61 62 63 \n") def test_getiptcinfo_tiff_none():
# Arrange
with Image.open("Tests/images/hopper.tif") as im:
# Act
iptc = IptcImagePlugin.getiptcinfo(im)
# Assert
assert iptc is None
def test_i():
# Arrange
c = b"a"
# Act
ret = IptcImagePlugin.i(c)
# Assert
assert ret == 97
def test_dump():
# Arrange
c = b"abc"
# Temporarily redirect stdout
old_stdout = sys.stdout
sys.stdout = mystdout = StringIO()
# Act
IptcImagePlugin.dump(c)
# Reset stdout
sys.stdout = old_stdout
# Assert
assert mystdout.getvalue() == "61 62 63 \n"

View File

@ -1,8 +1,9 @@
from io import BytesIO from io import BytesIO
import pytest
from PIL import Image from PIL import Image
from .helper import PillowTestCase, hopper from .helper import hopper
PIL151 = b""" PIL151 = b"""
#define basic_width 32 #define basic_width 32
@ -28,50 +29,53 @@ static char basic_bits[] = {
""" """
class TestFileXbm(PillowTestCase): def test_pil151():
def test_pil151(self): with Image.open(BytesIO(PIL151)) as im:
with Image.open(BytesIO(PIL151)) as im: im.load()
im.load() assert im.mode == "1"
self.assertEqual(im.mode, "1") assert im.size == (32, 32)
self.assertEqual(im.size, (32, 32))
def test_open(self):
# Arrange
# Created with `convert hopper.png hopper.xbm`
filename = "Tests/images/hopper.xbm"
# Act def test_open():
with Image.open(filename) as im: # Arrange
# Created with `convert hopper.png hopper.xbm`
filename = "Tests/images/hopper.xbm"
# Assert # Act
self.assertEqual(im.mode, "1") with Image.open(filename) as im:
self.assertEqual(im.size, (128, 128))
def test_open_filename_with_underscore(self): # Assert
# Arrange assert im.mode == "1"
# Created with `convert hopper.png hopper_underscore.xbm` assert im.size == (128, 128)
filename = "Tests/images/hopper_underscore.xbm"
# Act
with Image.open(filename) as im:
# Assert def test_open_filename_with_underscore():
self.assertEqual(im.mode, "1") # Arrange
self.assertEqual(im.size, (128, 128)) # Created with `convert hopper.png hopper_underscore.xbm`
filename = "Tests/images/hopper_underscore.xbm"
def test_save_wrong_mode(self): # Act
im = hopper() with Image.open(filename) as im:
out = self.tempfile("temp.xbm")
with self.assertRaises(OSError): # Assert
im.save(out) assert im.mode == "1"
assert im.size == (128, 128)
def test_hotspot(self):
im = hopper("1")
out = self.tempfile("temp.xbm")
hotspot = (0, 7) def test_save_wrong_mode(tmp_path):
im.save(out, hotspot=hotspot) im = hopper()
out = str(tmp_path / "temp.xbm")
with Image.open(out) as reloaded: with pytest.raises(OSError):
self.assertEqual(reloaded.info["hotspot"], hotspot) im.save(out)
def test_hotspot(tmp_path):
im = hopper("1")
out = str(tmp_path / "temp.xbm")
hotspot = (0, 7)
im.save(out, hotspot=hotspot)
with Image.open(out) as reloaded:
assert reloaded.info["hotspot"] == hotspot

View File

@ -1,19 +1,18 @@
import pytest
from PIL import BdfFontFile, FontFile from PIL import BdfFontFile, FontFile
from .helper import PillowTestCase
filename = "Tests/images/courB08.bdf" filename = "Tests/images/courB08.bdf"
class TestFontBdf(PillowTestCase): def test_sanity():
def test_sanity(self): with open(filename, "rb") as test_file:
font = BdfFontFile.BdfFontFile(test_file)
with open(filename, "rb") as test_file: assert isinstance(font, FontFile.FontFile)
font = BdfFontFile.BdfFontFile(test_file) assert len([_f for _f in font.glyph if _f]) == 190
self.assertIsInstance(font, FontFile.FontFile)
self.assertEqual(len([_f for _f in font.glyph if _f]), 190)
def test_invalid_file(self): def test_invalid_file():
with open("Tests/images/flower.jpg", "rb") as fp: with open("Tests/images/flower.jpg", "rb") as fp:
self.assertRaises(SyntaxError, BdfFontFile.BdfFontFile, fp) with pytest.raises(SyntaxError):
BdfFontFile.BdfFontFile(fp)

View File

@ -1,59 +1,60 @@
import pytest
from PIL import Image from PIL import Image
from .helper import PillowTestCase, hopper from .helper import hopper
im = hopper().resize((128, 100)) im = hopper().resize((128, 100))
class TestImageArray(PillowTestCase): def test_toarray():
def test_toarray(self): def test(mode):
def test(mode): ai = im.convert(mode).__array_interface__
ai = im.convert(mode).__array_interface__ return ai["version"], ai["shape"], ai["typestr"], len(ai["data"])
return ai["version"], ai["shape"], ai["typestr"], len(ai["data"])
# self.assertEqual(test("1"), (3, (100, 128), '|b1', 1600)) # assert test("1") == (3, (100, 128), '|b1', 1600))
self.assertEqual(test("L"), (3, (100, 128), "|u1", 12800)) assert test("L") == (3, (100, 128), "|u1", 12800)
# FIXME: wrong? # FIXME: wrong?
self.assertEqual(test("I"), (3, (100, 128), Image._ENDIAN + "i4", 51200)) assert test("I") == (3, (100, 128), Image._ENDIAN + "i4", 51200)
# FIXME: wrong? # FIXME: wrong?
self.assertEqual(test("F"), (3, (100, 128), Image._ENDIAN + "f4", 51200)) assert test("F") == (3, (100, 128), Image._ENDIAN + "f4", 51200)
self.assertEqual(test("LA"), (3, (100, 128, 2), "|u1", 25600)) assert test("LA") == (3, (100, 128, 2), "|u1", 25600)
self.assertEqual(test("RGB"), (3, (100, 128, 3), "|u1", 38400)) assert test("RGB") == (3, (100, 128, 3), "|u1", 38400)
self.assertEqual(test("RGBA"), (3, (100, 128, 4), "|u1", 51200)) assert test("RGBA") == (3, (100, 128, 4), "|u1", 51200)
self.assertEqual(test("RGBX"), (3, (100, 128, 4), "|u1", 51200)) assert test("RGBX") == (3, (100, 128, 4), "|u1", 51200)
def test_fromarray(self):
class Wrapper:
""" Class with API matching Image.fromarray """
def __init__(self, img, arr_params): def test_fromarray():
self.img = img class Wrapper:
self.__array_interface__ = arr_params """ Class with API matching Image.fromarray """
def tobytes(self): def __init__(self, img, arr_params):
return self.img.tobytes() self.img = img
self.__array_interface__ = arr_params
def test(mode): def tobytes(self):
i = im.convert(mode) return self.img.tobytes()
a = i.__array_interface__
a["strides"] = 1 # pretend it's non-contiguous
# Make wrapper instance for image, new array interface
wrapped = Wrapper(i, a)
out = Image.fromarray(wrapped)
return out.mode, out.size, list(i.getdata()) == list(out.getdata())
# self.assertEqual(test("1"), ("1", (128, 100), True)) def test(mode):
self.assertEqual(test("L"), ("L", (128, 100), True)) i = im.convert(mode)
self.assertEqual(test("I"), ("I", (128, 100), True)) a = i.__array_interface__
self.assertEqual(test("F"), ("F", (128, 100), True)) a["strides"] = 1 # pretend it's non-contiguous
self.assertEqual(test("LA"), ("LA", (128, 100), True)) # Make wrapper instance for image, new array interface
self.assertEqual(test("RGB"), ("RGB", (128, 100), True)) wrapped = Wrapper(i, a)
self.assertEqual(test("RGBA"), ("RGBA", (128, 100), True)) out = Image.fromarray(wrapped)
self.assertEqual(test("RGBX"), ("RGBA", (128, 100), True)) return out.mode, out.size, list(i.getdata()) == list(out.getdata())
# Test mode is None with no "typestr" in the array interface # assert test("1") == ("1", (128, 100), True)
with self.assertRaises(TypeError): assert test("L") == ("L", (128, 100), True)
wrapped = Wrapper(test("L"), {"shape": (100, 128)}) assert test("I") == ("I", (128, 100), True)
Image.fromarray(wrapped) assert test("F") == ("F", (128, 100), True)
assert test("LA") == ("LA", (128, 100), True)
assert test("RGB") == ("RGB", (128, 100), True)
assert test("RGBA") == ("RGBA", (128, 100), True)
assert test("RGBX") == ("RGBA", (128, 100), True)
# Test mode is None with no "typestr" in the array interface
with pytest.raises(TypeError):
wrapped = Wrapper(test("L"), {"shape": (100, 128)})
Image.fromarray(wrapped)

View File

@ -2,40 +2,40 @@ import copy
from PIL import Image from PIL import Image
from .helper import PillowTestCase, hopper from .helper import hopper
class TestImageCopy(PillowTestCase): def test_copy():
def test_copy(self): croppedCoordinates = (10, 10, 20, 20)
croppedCoordinates = (10, 10, 20, 20) croppedSize = (10, 10)
croppedSize = (10, 10) for mode in "1", "P", "L", "RGB", "I", "F":
for mode in "1", "P", "L", "RGB", "I", "F": # Internal copy method
# Internal copy method im = hopper(mode)
im = hopper(mode)
out = im.copy()
self.assertEqual(out.mode, im.mode)
self.assertEqual(out.size, im.size)
# Python's copy method
im = hopper(mode)
out = copy.copy(im)
self.assertEqual(out.mode, im.mode)
self.assertEqual(out.size, im.size)
# Internal copy method on a cropped image
im = hopper(mode)
out = im.crop(croppedCoordinates).copy()
self.assertEqual(out.mode, im.mode)
self.assertEqual(out.size, croppedSize)
# Python's copy method on a cropped image
im = hopper(mode)
out = copy.copy(im.crop(croppedCoordinates))
self.assertEqual(out.mode, im.mode)
self.assertEqual(out.size, croppedSize)
def test_copy_zero(self):
im = Image.new("RGB", (0, 0))
out = im.copy() out = im.copy()
self.assertEqual(out.mode, im.mode) assert out.mode == im.mode
self.assertEqual(out.size, im.size) assert out.size == im.size
# Python's copy method
im = hopper(mode)
out = copy.copy(im)
assert out.mode == im.mode
assert out.size == im.size
# Internal copy method on a cropped image
im = hopper(mode)
out = im.crop(croppedCoordinates).copy()
assert out.mode == im.mode
assert out.size == croppedSize
# Python's copy method on a cropped image
im = hopper(mode)
out = copy.copy(im.crop(croppedCoordinates))
assert out.mode == im.mode
assert out.size == croppedSize
def test_copy_zero():
im = Image.new("RGB", (0, 0))
out = im.copy()
assert out.mode == im.mode
assert out.size == im.size

View File

@ -1,67 +1,68 @@
from .helper import PillowTestCase, hopper from .helper import hopper
class TestImageGetColors(PillowTestCase): def test_getcolors():
def test_getcolors(self): def getcolors(mode, limit=None):
def getcolors(mode, limit=None): im = hopper(mode)
im = hopper(mode) if limit:
if limit: colors = im.getcolors(limit)
colors = im.getcolors(limit) else:
else: colors = im.getcolors()
colors = im.getcolors() if colors:
if colors: return len(colors)
return len(colors) return None
return None
self.assertEqual(getcolors("1"), 2) assert getcolors("1") == 2
self.assertEqual(getcolors("L"), 255) assert getcolors("L") == 255
self.assertEqual(getcolors("I"), 255) assert getcolors("I") == 255
self.assertEqual(getcolors("F"), 255) assert getcolors("F") == 255
self.assertEqual(getcolors("P"), 90) # fixed palette assert getcolors("P") == 90 # fixed palette
self.assertIsNone(getcolors("RGB")) assert getcolors("RGB") is None
self.assertIsNone(getcolors("RGBA")) assert getcolors("RGBA") is None
self.assertIsNone(getcolors("CMYK")) assert getcolors("CMYK") is None
self.assertIsNone(getcolors("YCbCr")) assert getcolors("YCbCr") is None
self.assertIsNone(getcolors("L", 128)) assert getcolors("L", 128) is None
self.assertEqual(getcolors("L", 1024), 255) assert getcolors("L", 1024) == 255
self.assertIsNone(getcolors("RGB", 8192)) assert getcolors("RGB", 8192) is None
self.assertEqual(getcolors("RGB", 16384), 10100) assert getcolors("RGB", 16384) == 10100
self.assertEqual(getcolors("RGB", 100000), 10100) assert getcolors("RGB", 100000) == 10100
self.assertEqual(getcolors("RGBA", 16384), 10100) assert getcolors("RGBA", 16384) == 10100
self.assertEqual(getcolors("CMYK", 16384), 10100) assert getcolors("CMYK", 16384) == 10100
self.assertEqual(getcolors("YCbCr", 16384), 9329) assert getcolors("YCbCr", 16384) == 9329
# --------------------------------------------------------------------
def test_pack(self): # --------------------------------------------------------------------
# Pack problems for small tables (@PIL209)
im = hopper().quantize(3).convert("RGB")
expected = [ def test_pack():
(4039, (172, 166, 181)), # Pack problems for small tables (@PIL209)
(4385, (124, 113, 134)),
(7960, (31, 20, 33)),
]
A = im.getcolors(maxcolors=2) im = hopper().quantize(3).convert("RGB")
self.assertIsNone(A)
A = im.getcolors(maxcolors=3) expected = [
A.sort() (4039, (172, 166, 181)),
self.assertEqual(A, expected) (4385, (124, 113, 134)),
(7960, (31, 20, 33)),
]
A = im.getcolors(maxcolors=4) A = im.getcolors(maxcolors=2)
A.sort() assert A is None
self.assertEqual(A, expected)
A = im.getcolors(maxcolors=8) A = im.getcolors(maxcolors=3)
A.sort() A.sort()
self.assertEqual(A, expected) assert A == expected
A = im.getcolors(maxcolors=16) A = im.getcolors(maxcolors=4)
A.sort() A.sort()
self.assertEqual(A, expected) assert A == expected
A = im.getcolors(maxcolors=8)
A.sort()
assert A == expected
A = im.getcolors(maxcolors=16)
A.sort()
assert A == expected

View File

@ -1,29 +1,28 @@
from PIL import Image from PIL import Image
from .helper import PillowTestCase, hopper from .helper import hopper
class TestImageGetData(PillowTestCase): def test_sanity():
def test_sanity(self): data = hopper().getdata()
data = hopper().getdata() len(data)
list(data)
len(data) assert data[0] == (20, 20, 70)
list(data)
self.assertEqual(data[0], (20, 20, 70))
def test_roundtrip(self): def test_roundtrip():
def getdata(mode): def getdata(mode):
im = hopper(mode).resize((32, 30), Image.NEAREST) im = hopper(mode).resize((32, 30), Image.NEAREST)
data = im.getdata() data = im.getdata()
return data[0], len(data), len(list(data)) return data[0], len(data), len(list(data))
self.assertEqual(getdata("1"), (0, 960, 960)) assert getdata("1") == (0, 960, 960)
self.assertEqual(getdata("L"), (17, 960, 960)) assert getdata("L") == (17, 960, 960)
self.assertEqual(getdata("I"), (17, 960, 960)) assert getdata("I") == (17, 960, 960)
self.assertEqual(getdata("F"), (17.0, 960, 960)) assert getdata("F") == (17.0, 960, 960)
self.assertEqual(getdata("RGB"), ((11, 13, 52), 960, 960)) assert getdata("RGB") == ((11, 13, 52), 960, 960)
self.assertEqual(getdata("RGBA"), ((11, 13, 52, 255), 960, 960)) assert getdata("RGBA") == ((11, 13, 52, 255), 960, 960)
self.assertEqual(getdata("CMYK"), ((244, 242, 203, 0), 960, 960)) assert getdata("CMYK") == ((244, 242, 203, 0), 960, 960)
self.assertEqual(getdata("YCbCr"), ((16, 147, 123), 960, 960)) assert getdata("YCbCr") == ((16, 147, 123), 960, 960)

View File

@ -1,28 +1,28 @@
import pytest import pytest
from PIL import Image from PIL import Image
from .helper import PillowTestCase, hopper, is_big_endian, on_ci from .helper import hopper, is_big_endian, on_ci
class TestImageGetExtrema(PillowTestCase): @pytest.mark.xfail(is_big_endian() and on_ci(), reason="Fails on big-endian")
@pytest.mark.xfail(is_big_endian() and on_ci(), reason="Fails on big-endian") def test_extrema():
def test_extrema(self): def extrema(mode):
def extrema(mode): return hopper(mode).getextrema()
return hopper(mode).getextrema()
self.assertEqual(extrema("1"), (0, 255)) assert extrema("1") == (0, 255)
self.assertEqual(extrema("L"), (1, 255)) assert extrema("L") == (1, 255)
self.assertEqual(extrema("I"), (1, 255)) assert extrema("I") == (1, 255)
self.assertEqual(extrema("F"), (1, 255)) assert extrema("F") == (1, 255)
self.assertEqual(extrema("P"), (0, 225)) # fixed palette assert extrema("P") == (0, 225) # fixed palette
self.assertEqual(extrema("RGB"), ((0, 255), (0, 255), (0, 255))) assert extrema("RGB") == ((0, 255), (0, 255), (0, 255))
self.assertEqual(extrema("RGBA"), ((0, 255), (0, 255), (0, 255), (255, 255))) assert extrema("RGBA") == ((0, 255), (0, 255), (0, 255), (255, 255))
self.assertEqual(extrema("CMYK"), ((0, 255), (0, 255), (0, 255), (0, 0))) assert extrema("CMYK") == ((0, 255), (0, 255), (0, 255), (0, 0))
self.assertEqual(extrema("I;16"), (1, 255)) assert extrema("I;16") == (1, 255)
@pytest.mark.xfail(is_big_endian() and on_ci(), reason="Fails on big-endian")
def test_true_16(self): @pytest.mark.xfail(is_big_endian() and on_ci(), reason="Fails on big-endian")
with Image.open("Tests/images/16_bit_noise.tif") as im: def test_true_16():
self.assertEqual(im.mode, "I;16") with Image.open("Tests/images/16_bit_noise.tif") as im:
extrema = im.getextrema() assert im.mode == "I;16"
self.assertEqual(extrema, (106, 285)) extrema = im.getextrema()
assert extrema == (106, 285)

View File

@ -1,31 +1,29 @@
from PIL import Image from PIL import Image
from .helper import PillowTestCase, hopper from .helper import hopper
class TestImageGetProjection(PillowTestCase): def test_sanity():
def test_sanity(self): im = hopper()
im = hopper() projection = im.getprojection()
projection = im.getprojection() assert len(projection) == 2
assert len(projection[0]) == im.size[0]
assert len(projection[1]) == im.size[1]
self.assertEqual(len(projection), 2) # 8-bit image
self.assertEqual(len(projection[0]), im.size[0]) im = Image.new("L", (10, 10))
self.assertEqual(len(projection[1]), im.size[1]) assert im.getprojection()[0] == [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
assert im.getprojection()[1] == [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
im.paste(255, (2, 4, 8, 6))
assert im.getprojection()[0] == [0, 0, 1, 1, 1, 1, 1, 1, 0, 0]
assert im.getprojection()[1] == [0, 0, 0, 0, 1, 1, 0, 0, 0, 0]
# 8-bit image # 32-bit image
im = Image.new("L", (10, 10)) im = Image.new("RGB", (10, 10))
self.assertEqual(im.getprojection()[0], [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]) assert im.getprojection()[0] == [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
self.assertEqual(im.getprojection()[1], [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]) assert im.getprojection()[1] == [0, 0, 0, 0, 0, 0, 0, 0, 0, 0]
im.paste(255, (2, 4, 8, 6)) im.paste(255, (2, 4, 8, 6))
self.assertEqual(im.getprojection()[0], [0, 0, 1, 1, 1, 1, 1, 1, 0, 0]) assert im.getprojection()[0] == [0, 0, 1, 1, 1, 1, 1, 1, 0, 0]
self.assertEqual(im.getprojection()[1], [0, 0, 0, 0, 1, 1, 0, 0, 0, 0]) assert im.getprojection()[1] == [0, 0, 0, 0, 1, 1, 0, 0, 0, 0]
# 32-bit image
im = Image.new("RGB", (10, 10))
self.assertEqual(im.getprojection()[0], [0, 0, 0, 0, 0, 0, 0, 0, 0, 0])
self.assertEqual(im.getprojection()[1], [0, 0, 0, 0, 0, 0, 0, 0, 0, 0])
im.paste(255, (2, 4, 8, 6))
self.assertEqual(im.getprojection()[0], [0, 0, 1, 1, 1, 1, 1, 1, 0, 0])
self.assertEqual(im.getprojection()[1], [0, 0, 0, 0, 1, 1, 0, 0, 0, 0])

View File

@ -1,18 +1,17 @@
from .helper import PillowTestCase, hopper from .helper import hopper
class TestImageHistogram(PillowTestCase): def test_histogram():
def test_histogram(self): def histogram(mode):
def histogram(mode): h = hopper(mode).histogram()
h = hopper(mode).histogram() return len(h), min(h), max(h)
return len(h), min(h), max(h)
self.assertEqual(histogram("1"), (256, 0, 10994)) assert histogram("1") == (256, 0, 10994)
self.assertEqual(histogram("L"), (256, 0, 662)) assert histogram("L") == (256, 0, 662)
self.assertEqual(histogram("I"), (256, 0, 662)) assert histogram("I") == (256, 0, 662)
self.assertEqual(histogram("F"), (256, 0, 662)) assert histogram("F") == (256, 0, 662)
self.assertEqual(histogram("P"), (256, 0, 1871)) assert histogram("P") == (256, 0, 1871)
self.assertEqual(histogram("RGB"), (768, 4, 675)) assert histogram("RGB") == (768, 4, 675)
self.assertEqual(histogram("RGBA"), (1024, 0, 16384)) assert histogram("RGBA") == (1024, 0, 16384)
self.assertEqual(histogram("CMYK"), (1024, 0, 16384)) assert histogram("CMYK") == (1024, 0, 16384)
self.assertEqual(histogram("YCbCr"), (768, 0, 1908)) assert histogram("YCbCr") == (768, 0, 1908)

View File

@ -1,36 +1,40 @@
import os import os
import pytest
from PIL import Image from PIL import Image
from .helper import PillowTestCase, hopper from .helper import hopper
class TestImageLoad(PillowTestCase): def test_sanity():
def test_sanity(self): im = hopper()
pix = im.load()
im = hopper() assert pix[0, 0] == (20, 20, 70)
pix = im.load()
self.assertEqual(pix[0, 0], (20, 20, 70)) def test_close():
im = Image.open("Tests/images/hopper.gif")
im.close()
with pytest.raises(ValueError):
im.load()
with pytest.raises(ValueError):
im.getpixel((0, 0))
def test_close(self):
im = Image.open("Tests/images/hopper.gif")
im.close()
self.assertRaises(ValueError, im.load)
self.assertRaises(ValueError, im.getpixel, (0, 0))
def test_contextmanager(self): def test_contextmanager():
fn = None fn = None
with Image.open("Tests/images/hopper.gif") as im: with Image.open("Tests/images/hopper.gif") as im:
fn = im.fp.fileno() fn = im.fp.fileno()
os.fstat(fn) os.fstat(fn)
self.assertRaises(OSError, os.fstat, fn) with pytest.raises(OSError):
os.fstat(fn)
def test_contextmanager_non_exclusive_fp(self):
with open("Tests/images/hopper.gif", "rb") as fp:
with Image.open(fp):
pass
self.assertFalse(fp.closed) def test_contextmanager_non_exclusive_fp():
with open("Tests/images/hopper.gif", "rb") as fp:
with Image.open(fp):
pass
assert not fp.closed

View File

@ -1,72 +1,72 @@
from PIL import Image from PIL import Image
from .helper import PillowTestCase, hopper from .helper import hopper
class TestImageMode(PillowTestCase): def test_sanity():
def test_sanity(self):
with hopper() as im: with hopper() as im:
im.mode im.mode
from PIL import ImageMode from PIL import ImageMode
ImageMode.getmode("1") ImageMode.getmode("1")
ImageMode.getmode("L") ImageMode.getmode("L")
ImageMode.getmode("P") ImageMode.getmode("P")
ImageMode.getmode("RGB") ImageMode.getmode("RGB")
ImageMode.getmode("I") ImageMode.getmode("I")
ImageMode.getmode("F") ImageMode.getmode("F")
m = ImageMode.getmode("1") m = ImageMode.getmode("1")
self.assertEqual(m.mode, "1") assert m.mode == "1"
self.assertEqual(str(m), "1") assert str(m) == "1"
self.assertEqual(m.bands, ("1",)) assert m.bands == ("1",)
self.assertEqual(m.basemode, "L") assert m.basemode == "L"
self.assertEqual(m.basetype, "L") assert m.basetype == "L"
for mode in ( for mode in (
"I;16", "I;16",
"I;16S", "I;16S",
"I;16L", "I;16L",
"I;16LS", "I;16LS",
"I;16B", "I;16B",
"I;16BS", "I;16BS",
"I;16N", "I;16N",
"I;16NS", "I;16NS",
): ):
m = ImageMode.getmode(mode) m = ImageMode.getmode(mode)
self.assertEqual(m.mode, mode) assert m.mode == mode
self.assertEqual(str(m), mode) assert str(m) == mode
self.assertEqual(m.bands, ("I",)) assert m.bands == ("I",)
self.assertEqual(m.basemode, "L") assert m.basemode == "L"
self.assertEqual(m.basetype, "L") assert m.basetype == "L"
m = ImageMode.getmode("RGB") m = ImageMode.getmode("RGB")
self.assertEqual(m.mode, "RGB") assert m.mode == "RGB"
self.assertEqual(str(m), "RGB") assert str(m) == "RGB"
self.assertEqual(m.bands, ("R", "G", "B")) assert m.bands == ("R", "G", "B")
self.assertEqual(m.basemode, "RGB") assert m.basemode == "RGB"
self.assertEqual(m.basetype, "L") assert m.basetype == "L"
def test_properties(self):
def check(mode, *result):
signature = (
Image.getmodebase(mode),
Image.getmodetype(mode),
Image.getmodebands(mode),
Image.getmodebandnames(mode),
)
self.assertEqual(signature, result)
check("1", "L", "L", 1, ("1",)) def test_properties():
check("L", "L", "L", 1, ("L",)) def check(mode, *result):
check("P", "P", "L", 1, ("P",)) signature = (
check("I", "L", "I", 1, ("I",)) Image.getmodebase(mode),
check("F", "L", "F", 1, ("F",)) Image.getmodetype(mode),
check("RGB", "RGB", "L", 3, ("R", "G", "B")) Image.getmodebands(mode),
check("RGBA", "RGB", "L", 4, ("R", "G", "B", "A")) Image.getmodebandnames(mode),
check("RGBX", "RGB", "L", 4, ("R", "G", "B", "X")) )
check("RGBX", "RGB", "L", 4, ("R", "G", "B", "X")) assert signature == result
check("CMYK", "RGB", "L", 4, ("C", "M", "Y", "K"))
check("YCbCr", "RGB", "L", 3, ("Y", "Cb", "Cr")) check("1", "L", "L", 1, ("1",))
check("L", "L", "L", 1, ("L",))
check("P", "P", "L", 1, ("P",))
check("I", "L", "I", 1, ("I",))
check("F", "L", "F", 1, ("F",))
check("RGB", "RGB", "L", 3, ("R", "G", "B"))
check("RGBA", "RGB", "L", 4, ("R", "G", "B", "A"))
check("RGBX", "RGB", "L", 4, ("R", "G", "B", "X"))
check("RGBX", "RGB", "L", 4, ("R", "G", "B", "X"))
check("CMYK", "RGB", "L", 4, ("C", "M", "Y", "K"))
check("YCbCr", "RGB", "L", 3, ("Y", "Cb", "Cr"))

View File

@ -1,52 +1,48 @@
from PIL import Image from PIL import Image
from .helper import PillowTestCase
def test_interface():
im = Image.new("RGBA", (1, 1), (1, 2, 3, 0))
assert im.getpixel((0, 0)) == (1, 2, 3, 0)
im = Image.new("RGBA", (1, 1), (1, 2, 3))
assert im.getpixel((0, 0)) == (1, 2, 3, 255)
im.putalpha(Image.new("L", im.size, 4))
assert im.getpixel((0, 0)) == (1, 2, 3, 4)
im.putalpha(5)
assert im.getpixel((0, 0)) == (1, 2, 3, 5)
class TestImagePutAlpha(PillowTestCase): def test_promote():
def test_interface(self): im = Image.new("L", (1, 1), 1)
assert im.getpixel((0, 0)) == 1
im = Image.new("RGBA", (1, 1), (1, 2, 3, 0)) im.putalpha(2)
self.assertEqual(im.getpixel((0, 0)), (1, 2, 3, 0)) assert im.mode == "LA"
assert im.getpixel((0, 0)) == (1, 2)
im = Image.new("RGBA", (1, 1), (1, 2, 3)) im = Image.new("P", (1, 1), 1)
self.assertEqual(im.getpixel((0, 0)), (1, 2, 3, 255)) assert im.getpixel((0, 0)) == 1
im.putalpha(Image.new("L", im.size, 4)) im.putalpha(2)
self.assertEqual(im.getpixel((0, 0)), (1, 2, 3, 4)) assert im.mode == "PA"
assert im.getpixel((0, 0)) == (1, 2)
im.putalpha(5) im = Image.new("RGB", (1, 1), (1, 2, 3))
self.assertEqual(im.getpixel((0, 0)), (1, 2, 3, 5)) assert im.getpixel((0, 0)) == (1, 2, 3)
def test_promote(self): im.putalpha(4)
assert im.mode == "RGBA"
assert im.getpixel((0, 0)) == (1, 2, 3, 4)
im = Image.new("L", (1, 1), 1)
self.assertEqual(im.getpixel((0, 0)), 1)
im.putalpha(2) def test_readonly():
self.assertEqual(im.mode, "LA") im = Image.new("RGB", (1, 1), (1, 2, 3))
self.assertEqual(im.getpixel((0, 0)), (1, 2)) im.readonly = 1
im = Image.new("P", (1, 1), 1) im.putalpha(4)
self.assertEqual(im.getpixel((0, 0)), 1) assert not im.readonly
assert im.mode == "RGBA"
im.putalpha(2) assert im.getpixel((0, 0)) == (1, 2, 3, 4)
self.assertEqual(im.mode, "PA")
self.assertEqual(im.getpixel((0, 0)), (1, 2))
im = Image.new("RGB", (1, 1), (1, 2, 3))
self.assertEqual(im.getpixel((0, 0)), (1, 2, 3))
im.putalpha(4)
self.assertEqual(im.mode, "RGBA")
self.assertEqual(im.getpixel((0, 0)), (1, 2, 3, 4))
def test_readonly(self):
im = Image.new("RGB", (1, 1), (1, 2, 3))
im.readonly = 1
im.putalpha(4)
self.assertFalse(im.readonly)
self.assertEqual(im.mode, "RGBA")
self.assertEqual(im.getpixel((0, 0)), (1, 2, 3, 4))

View File

@ -1,184 +1,192 @@
import pytest
from PIL import Image, ImageColor from PIL import Image, ImageColor
from .helper import PillowTestCase
def test_hash():
# short 3 components
assert (255, 0, 0) == ImageColor.getrgb("#f00")
assert (0, 255, 0) == ImageColor.getrgb("#0f0")
assert (0, 0, 255) == ImageColor.getrgb("#00f")
# short 4 components
assert (255, 0, 0, 0) == ImageColor.getrgb("#f000")
assert (0, 255, 0, 0) == ImageColor.getrgb("#0f00")
assert (0, 0, 255, 0) == ImageColor.getrgb("#00f0")
assert (0, 0, 0, 255) == ImageColor.getrgb("#000f")
# long 3 components
assert (222, 0, 0) == ImageColor.getrgb("#de0000")
assert (0, 222, 0) == ImageColor.getrgb("#00de00")
assert (0, 0, 222) == ImageColor.getrgb("#0000de")
# long 4 components
assert (222, 0, 0, 0) == ImageColor.getrgb("#de000000")
assert (0, 222, 0, 0) == ImageColor.getrgb("#00de0000")
assert (0, 0, 222, 0) == ImageColor.getrgb("#0000de00")
assert (0, 0, 0, 222) == ImageColor.getrgb("#000000de")
# case insensitivity
assert ImageColor.getrgb("#DEF") == ImageColor.getrgb("#def")
assert ImageColor.getrgb("#CDEF") == ImageColor.getrgb("#cdef")
assert ImageColor.getrgb("#DEFDEF") == ImageColor.getrgb("#defdef")
assert ImageColor.getrgb("#CDEFCDEF") == ImageColor.getrgb("#cdefcdef")
# not a number
with pytest.raises(ValueError):
ImageColor.getrgb("#fo0")
with pytest.raises(ValueError):
ImageColor.getrgb("#fo00")
with pytest.raises(ValueError):
ImageColor.getrgb("#fo0000")
with pytest.raises(ValueError):
ImageColor.getrgb("#fo000000")
# wrong number of components
with pytest.raises(ValueError):
ImageColor.getrgb("#f0000")
with pytest.raises(ValueError):
ImageColor.getrgb("#f000000")
with pytest.raises(ValueError):
ImageColor.getrgb("#f00000000")
with pytest.raises(ValueError):
ImageColor.getrgb("#f000000000")
with pytest.raises(ValueError):
ImageColor.getrgb("#f00000 ")
class TestImageColor(PillowTestCase): def test_colormap():
def test_hash(self): assert (0, 0, 0) == ImageColor.getrgb("black")
# short 3 components assert (255, 255, 255) == ImageColor.getrgb("white")
self.assertEqual((255, 0, 0), ImageColor.getrgb("#f00")) assert (255, 255, 255) == ImageColor.getrgb("WHITE")
self.assertEqual((0, 255, 0), ImageColor.getrgb("#0f0"))
self.assertEqual((0, 0, 255), ImageColor.getrgb("#00f"))
# short 4 components with pytest.raises(ValueError):
self.assertEqual((255, 0, 0, 0), ImageColor.getrgb("#f000")) ImageColor.getrgb("black ")
self.assertEqual((0, 255, 0, 0), ImageColor.getrgb("#0f00"))
self.assertEqual((0, 0, 255, 0), ImageColor.getrgb("#00f0"))
self.assertEqual((0, 0, 0, 255), ImageColor.getrgb("#000f"))
# long 3 components
self.assertEqual((222, 0, 0), ImageColor.getrgb("#de0000"))
self.assertEqual((0, 222, 0), ImageColor.getrgb("#00de00"))
self.assertEqual((0, 0, 222), ImageColor.getrgb("#0000de"))
# long 4 components def test_functions():
self.assertEqual((222, 0, 0, 0), ImageColor.getrgb("#de000000")) # rgb numbers
self.assertEqual((0, 222, 0, 0), ImageColor.getrgb("#00de0000")) assert (255, 0, 0) == ImageColor.getrgb("rgb(255,0,0)")
self.assertEqual((0, 0, 222, 0), ImageColor.getrgb("#0000de00")) assert (0, 255, 0) == ImageColor.getrgb("rgb(0,255,0)")
self.assertEqual((0, 0, 0, 222), ImageColor.getrgb("#000000de")) assert (0, 0, 255) == ImageColor.getrgb("rgb(0,0,255)")
# case insensitivity # percents
self.assertEqual(ImageColor.getrgb("#DEF"), ImageColor.getrgb("#def")) assert (255, 0, 0) == ImageColor.getrgb("rgb(100%,0%,0%)")
self.assertEqual(ImageColor.getrgb("#CDEF"), ImageColor.getrgb("#cdef")) assert (0, 255, 0) == ImageColor.getrgb("rgb(0%,100%,0%)")
self.assertEqual(ImageColor.getrgb("#DEFDEF"), ImageColor.getrgb("#defdef")) assert (0, 0, 255) == ImageColor.getrgb("rgb(0%,0%,100%)")
self.assertEqual(ImageColor.getrgb("#CDEFCDEF"), ImageColor.getrgb("#cdefcdef"))
# not a number # rgba numbers
self.assertRaises(ValueError, ImageColor.getrgb, "#fo0") assert (255, 0, 0, 0) == ImageColor.getrgb("rgba(255,0,0,0)")
self.assertRaises(ValueError, ImageColor.getrgb, "#fo00") assert (0, 255, 0, 0) == ImageColor.getrgb("rgba(0,255,0,0)")
self.assertRaises(ValueError, ImageColor.getrgb, "#fo0000") assert (0, 0, 255, 0) == ImageColor.getrgb("rgba(0,0,255,0)")
self.assertRaises(ValueError, ImageColor.getrgb, "#fo000000") assert (0, 0, 0, 255) == ImageColor.getrgb("rgba(0,0,0,255)")
# wrong number of components assert (255, 0, 0) == ImageColor.getrgb("hsl(0,100%,50%)")
self.assertRaises(ValueError, ImageColor.getrgb, "#f0000") assert (255, 0, 0) == ImageColor.getrgb("hsl(360,100%,50%)")
self.assertRaises(ValueError, ImageColor.getrgb, "#f000000") assert (0, 255, 255) == ImageColor.getrgb("hsl(180,100%,50%)")
self.assertRaises(ValueError, ImageColor.getrgb, "#f00000000")
self.assertRaises(ValueError, ImageColor.getrgb, "#f000000000")
self.assertRaises(ValueError, ImageColor.getrgb, "#f00000 ")
def test_colormap(self): assert (255, 0, 0) == ImageColor.getrgb("hsv(0,100%,100%)")
self.assertEqual((0, 0, 0), ImageColor.getrgb("black")) assert (255, 0, 0) == ImageColor.getrgb("hsv(360,100%,100%)")
self.assertEqual((255, 255, 255), ImageColor.getrgb("white")) assert (0, 255, 255) == ImageColor.getrgb("hsv(180,100%,100%)")
self.assertEqual((255, 255, 255), ImageColor.getrgb("WHITE"))
self.assertRaises(ValueError, ImageColor.getrgb, "black ") # alternate format
assert ImageColor.getrgb("hsb(0,100%,50%)") == ImageColor.getrgb("hsv(0,100%,50%)")
def test_functions(self): # floats
# rgb numbers assert (254, 3, 3) == ImageColor.getrgb("hsl(0.1,99.2%,50.3%)")
self.assertEqual((255, 0, 0), ImageColor.getrgb("rgb(255,0,0)")) assert (255, 0, 0) == ImageColor.getrgb("hsl(360.,100.0%,50%)")
self.assertEqual((0, 255, 0), ImageColor.getrgb("rgb(0,255,0)"))
self.assertEqual((0, 0, 255), ImageColor.getrgb("rgb(0,0,255)"))
# percents assert (253, 2, 2) == ImageColor.getrgb("hsv(0.1,99.2%,99.3%)")
self.assertEqual((255, 0, 0), ImageColor.getrgb("rgb(100%,0%,0%)")) assert (255, 0, 0) == ImageColor.getrgb("hsv(360.,100.0%,100%)")
self.assertEqual((0, 255, 0), ImageColor.getrgb("rgb(0%,100%,0%)"))
self.assertEqual((0, 0, 255), ImageColor.getrgb("rgb(0%,0%,100%)"))
# rgba numbers # case insensitivity
self.assertEqual((255, 0, 0, 0), ImageColor.getrgb("rgba(255,0,0,0)")) assert ImageColor.getrgb("RGB(255,0,0)") == ImageColor.getrgb("rgb(255,0,0)")
self.assertEqual((0, 255, 0, 0), ImageColor.getrgb("rgba(0,255,0,0)")) assert ImageColor.getrgb("RGB(100%,0%,0%)") == ImageColor.getrgb("rgb(100%,0%,0%)")
self.assertEqual((0, 0, 255, 0), ImageColor.getrgb("rgba(0,0,255,0)")) assert ImageColor.getrgb("RGBA(255,0,0,0)") == ImageColor.getrgb("rgba(255,0,0,0)")
self.assertEqual((0, 0, 0, 255), ImageColor.getrgb("rgba(0,0,0,255)")) assert ImageColor.getrgb("HSL(0,100%,50%)") == ImageColor.getrgb("hsl(0,100%,50%)")
assert ImageColor.getrgb("HSV(0,100%,50%)") == ImageColor.getrgb("hsv(0,100%,50%)")
assert ImageColor.getrgb("HSB(0,100%,50%)") == ImageColor.getrgb("hsb(0,100%,50%)")
self.assertEqual((255, 0, 0), ImageColor.getrgb("hsl(0,100%,50%)")) # space agnosticism
self.assertEqual((255, 0, 0), ImageColor.getrgb("hsl(360,100%,50%)")) assert (255, 0, 0) == ImageColor.getrgb("rgb( 255 , 0 , 0 )")
self.assertEqual((0, 255, 255), ImageColor.getrgb("hsl(180,100%,50%)")) assert (255, 0, 0) == ImageColor.getrgb("rgb( 100% , 0% , 0% )")
assert (255, 0, 0, 0) == ImageColor.getrgb("rgba( 255 , 0 , 0 , 0 )")
assert (255, 0, 0) == ImageColor.getrgb("hsl( 0 , 100% , 50% )")
assert (255, 0, 0) == ImageColor.getrgb("hsv( 0 , 100% , 100% )")
self.assertEqual((255, 0, 0), ImageColor.getrgb("hsv(0,100%,100%)")) # wrong number of components
self.assertEqual((255, 0, 0), ImageColor.getrgb("hsv(360,100%,100%)")) with pytest.raises(ValueError):
self.assertEqual((0, 255, 255), ImageColor.getrgb("hsv(180,100%,100%)")) ImageColor.getrgb("rgb(255,0)")
with pytest.raises(ValueError):
ImageColor.getrgb("rgb(255,0,0,0)")
# alternate format with pytest.raises(ValueError):
self.assertEqual( ImageColor.getrgb("rgb(100%,0%)")
ImageColor.getrgb("hsb(0,100%,50%)"), ImageColor.getrgb("hsv(0,100%,50%)") with pytest.raises(ValueError):
) ImageColor.getrgb("rgb(100%,0%,0)")
with pytest.raises(ValueError):
ImageColor.getrgb("rgb(100%,0%,0 %)")
with pytest.raises(ValueError):
ImageColor.getrgb("rgb(100%,0%,0%,0%)")
# floats with pytest.raises(ValueError):
self.assertEqual((254, 3, 3), ImageColor.getrgb("hsl(0.1,99.2%,50.3%)")) ImageColor.getrgb("rgba(255,0,0)")
self.assertEqual((255, 0, 0), ImageColor.getrgb("hsl(360.,100.0%,50%)")) with pytest.raises(ValueError):
ImageColor.getrgb("rgba(255,0,0,0,0)")
self.assertEqual((253, 2, 2), ImageColor.getrgb("hsv(0.1,99.2%,99.3%)")) with pytest.raises(ValueError):
self.assertEqual((255, 0, 0), ImageColor.getrgb("hsv(360.,100.0%,100%)")) ImageColor.getrgb("hsl(0,100%)")
with pytest.raises(ValueError):
ImageColor.getrgb("hsl(0,100%,0%,0%)")
with pytest.raises(ValueError):
ImageColor.getrgb("hsl(0%,100%,50%)")
with pytest.raises(ValueError):
ImageColor.getrgb("hsl(0,100,50%)")
with pytest.raises(ValueError):
ImageColor.getrgb("hsl(0,100%,50)")
# case insensitivity with pytest.raises(ValueError):
self.assertEqual( ImageColor.getrgb("hsv(0,100%)")
ImageColor.getrgb("RGB(255,0,0)"), ImageColor.getrgb("rgb(255,0,0)") with pytest.raises(ValueError):
) ImageColor.getrgb("hsv(0,100%,0%,0%)")
self.assertEqual( with pytest.raises(ValueError):
ImageColor.getrgb("RGB(100%,0%,0%)"), ImageColor.getrgb("rgb(100%,0%,0%)") ImageColor.getrgb("hsv(0%,100%,50%)")
) with pytest.raises(ValueError):
self.assertEqual( ImageColor.getrgb("hsv(0,100,50%)")
ImageColor.getrgb("RGBA(255,0,0,0)"), ImageColor.getrgb("rgba(255,0,0,0)") with pytest.raises(ValueError):
) ImageColor.getrgb("hsv(0,100%,50)")
self.assertEqual(
ImageColor.getrgb("HSL(0,100%,50%)"), ImageColor.getrgb("hsl(0,100%,50%)")
)
self.assertEqual(
ImageColor.getrgb("HSV(0,100%,50%)"), ImageColor.getrgb("hsv(0,100%,50%)")
)
self.assertEqual(
ImageColor.getrgb("HSB(0,100%,50%)"), ImageColor.getrgb("hsb(0,100%,50%)")
)
# space agnosticism
self.assertEqual((255, 0, 0), ImageColor.getrgb("rgb( 255 , 0 , 0 )"))
self.assertEqual((255, 0, 0), ImageColor.getrgb("rgb( 100% , 0% , 0% )"))
self.assertEqual(
(255, 0, 0, 0), ImageColor.getrgb("rgba( 255 , 0 , 0 , 0 )")
)
self.assertEqual((255, 0, 0), ImageColor.getrgb("hsl( 0 , 100% , 50% )"))
self.assertEqual((255, 0, 0), ImageColor.getrgb("hsv( 0 , 100% , 100% )"))
# wrong number of components # look for rounding errors (based on code by Tim Hatch)
self.assertRaises(ValueError, ImageColor.getrgb, "rgb(255,0)") def test_rounding_errors():
self.assertRaises(ValueError, ImageColor.getrgb, "rgb(255,0,0,0)") for color in ImageColor.colormap:
expected = Image.new("RGB", (1, 1), color).convert("L").getpixel((0, 0))
actual = ImageColor.getcolor(color, "L")
assert expected == actual
self.assertRaises(ValueError, ImageColor.getrgb, "rgb(100%,0%)") assert (0, 255, 115) == ImageColor.getcolor("rgba(0, 255, 115, 33)", "RGB")
self.assertRaises(ValueError, ImageColor.getrgb, "rgb(100%,0%,0)") Image.new("RGB", (1, 1), "white")
self.assertRaises(ValueError, ImageColor.getrgb, "rgb(100%,0%,0 %)")
self.assertRaises(ValueError, ImageColor.getrgb, "rgb(100%,0%,0%,0%)")
self.assertRaises(ValueError, ImageColor.getrgb, "rgba(255,0,0)") assert (0, 0, 0, 255) == ImageColor.getcolor("black", "RGBA")
self.assertRaises(ValueError, ImageColor.getrgb, "rgba(255,0,0,0,0)") assert (255, 255, 255, 255) == ImageColor.getcolor("white", "RGBA")
assert (0, 255, 115, 33) == ImageColor.getcolor("rgba(0, 255, 115, 33)", "RGBA")
Image.new("RGBA", (1, 1), "white")
self.assertRaises(ValueError, ImageColor.getrgb, "hsl(0,100%)") assert 0 == ImageColor.getcolor("black", "L")
self.assertRaises(ValueError, ImageColor.getrgb, "hsl(0,100%,0%,0%)") assert 255 == ImageColor.getcolor("white", "L")
self.assertRaises(ValueError, ImageColor.getrgb, "hsl(0%,100%,50%)") assert 163 == ImageColor.getcolor("rgba(0, 255, 115, 33)", "L")
self.assertRaises(ValueError, ImageColor.getrgb, "hsl(0,100,50%)") Image.new("L", (1, 1), "white")
self.assertRaises(ValueError, ImageColor.getrgb, "hsl(0,100%,50)")
self.assertRaises(ValueError, ImageColor.getrgb, "hsv(0,100%)") assert 0 == ImageColor.getcolor("black", "1")
self.assertRaises(ValueError, ImageColor.getrgb, "hsv(0,100%,0%,0%)") assert 255 == ImageColor.getcolor("white", "1")
self.assertRaises(ValueError, ImageColor.getrgb, "hsv(0%,100%,50%)") # The following test is wrong, but is current behavior
self.assertRaises(ValueError, ImageColor.getrgb, "hsv(0,100,50%)") # The correct result should be 255 due to the mode 1
self.assertRaises(ValueError, ImageColor.getrgb, "hsv(0,100%,50)") assert 163 == ImageColor.getcolor("rgba(0, 255, 115, 33)", "1")
# Correct behavior
# assert
# 255, ImageColor.getcolor("rgba(0, 255, 115, 33)", "1"))
Image.new("1", (1, 1), "white")
# look for rounding errors (based on code by Tim Hatch) assert (0, 255) == ImageColor.getcolor("black", "LA")
def test_rounding_errors(self): assert (255, 255) == ImageColor.getcolor("white", "LA")
assert (163, 33) == ImageColor.getcolor("rgba(0, 255, 115, 33)", "LA")
for color in ImageColor.colormap: Image.new("LA", (1, 1), "white")
expected = Image.new("RGB", (1, 1), color).convert("L").getpixel((0, 0))
actual = ImageColor.getcolor(color, "L")
self.assertEqual(expected, actual)
self.assertEqual(
(0, 255, 115), ImageColor.getcolor("rgba(0, 255, 115, 33)", "RGB")
)
Image.new("RGB", (1, 1), "white")
self.assertEqual((0, 0, 0, 255), ImageColor.getcolor("black", "RGBA"))
self.assertEqual((255, 255, 255, 255), ImageColor.getcolor("white", "RGBA"))
self.assertEqual(
(0, 255, 115, 33), ImageColor.getcolor("rgba(0, 255, 115, 33)", "RGBA")
)
Image.new("RGBA", (1, 1), "white")
self.assertEqual(0, ImageColor.getcolor("black", "L"))
self.assertEqual(255, ImageColor.getcolor("white", "L"))
self.assertEqual(163, ImageColor.getcolor("rgba(0, 255, 115, 33)", "L"))
Image.new("L", (1, 1), "white")
self.assertEqual(0, ImageColor.getcolor("black", "1"))
self.assertEqual(255, ImageColor.getcolor("white", "1"))
# The following test is wrong, but is current behavior
# The correct result should be 255 due to the mode 1
self.assertEqual(163, ImageColor.getcolor("rgba(0, 255, 115, 33)", "1"))
# Correct behavior
# self.assertEqual(
# 255, ImageColor.getcolor("rgba(0, 255, 115, 33)", "1"))
Image.new("1", (1, 1), "white")
self.assertEqual((0, 255), ImageColor.getcolor("black", "LA"))
self.assertEqual((255, 255), ImageColor.getcolor("white", "LA"))
self.assertEqual((163, 33), ImageColor.getcolor("rgba(0, 255, 115, 33)", "LA"))
Image.new("LA", (1, 1), "white")

View File

@ -1,7 +1,5 @@
from PIL import Image, ImageMath from PIL import Image, ImageMath
from .helper import PillowTestCase
def pixel(im): def pixel(im):
if hasattr(im, "im"): if hasattr(im, "im"):
@ -24,153 +22,168 @@ B2 = B.resize((2, 2))
images = {"A": A, "B": B, "F": F, "I": I} images = {"A": A, "B": B, "F": F, "I": I}
class TestImageMath(PillowTestCase): def test_sanity():
def test_sanity(self): assert ImageMath.eval("1") == 1
self.assertEqual(ImageMath.eval("1"), 1) assert ImageMath.eval("1+A", A=2) == 3
self.assertEqual(ImageMath.eval("1+A", A=2), 3) assert pixel(ImageMath.eval("A+B", A=A, B=B)) == "I 3"
self.assertEqual(pixel(ImageMath.eval("A+B", A=A, B=B)), "I 3") assert pixel(ImageMath.eval("A+B", images)) == "I 3"
self.assertEqual(pixel(ImageMath.eval("A+B", images)), "I 3") assert pixel(ImageMath.eval("float(A)+B", images)) == "F 3.0"
self.assertEqual(pixel(ImageMath.eval("float(A)+B", images)), "F 3.0") assert pixel(ImageMath.eval("int(float(A)+B)", images)) == "I 3"
self.assertEqual(pixel(ImageMath.eval("int(float(A)+B)", images)), "I 3")
def test_ops(self):
self.assertEqual(pixel(ImageMath.eval("-A", images)), "I -1") def test_ops():
self.assertEqual(pixel(ImageMath.eval("+B", images)), "L 2") assert pixel(ImageMath.eval("-A", images)) == "I -1"
assert pixel(ImageMath.eval("+B", images)) == "L 2"
self.assertEqual(pixel(ImageMath.eval("A+B", images)), "I 3") assert pixel(ImageMath.eval("A+B", images)) == "I 3"
self.assertEqual(pixel(ImageMath.eval("A-B", images)), "I -1") assert pixel(ImageMath.eval("A-B", images)) == "I -1"
self.assertEqual(pixel(ImageMath.eval("A*B", images)), "I 2") assert pixel(ImageMath.eval("A*B", images)) == "I 2"
self.assertEqual(pixel(ImageMath.eval("A/B", images)), "I 0") assert pixel(ImageMath.eval("A/B", images)) == "I 0"
self.assertEqual(pixel(ImageMath.eval("B**2", images)), "I 4") assert pixel(ImageMath.eval("B**2", images)) == "I 4"
self.assertEqual(pixel(ImageMath.eval("B**33", images)), "I 2147483647") assert pixel(ImageMath.eval("B**33", images)) == "I 2147483647"
self.assertEqual(pixel(ImageMath.eval("float(A)+B", images)), "F 3.0") assert pixel(ImageMath.eval("float(A)+B", images)) == "F 3.0"
self.assertEqual(pixel(ImageMath.eval("float(A)-B", images)), "F -1.0") assert pixel(ImageMath.eval("float(A)-B", images)) == "F -1.0"
self.assertEqual(pixel(ImageMath.eval("float(A)*B", images)), "F 2.0") assert pixel(ImageMath.eval("float(A)*B", images)) == "F 2.0"
self.assertEqual(pixel(ImageMath.eval("float(A)/B", images)), "F 0.5") assert pixel(ImageMath.eval("float(A)/B", images)) == "F 0.5"
self.assertEqual(pixel(ImageMath.eval("float(B)**2", images)), "F 4.0") assert pixel(ImageMath.eval("float(B)**2", images)) == "F 4.0"
self.assertEqual( assert pixel(ImageMath.eval("float(B)**33", images)) == "F 8589934592.0"
pixel(ImageMath.eval("float(B)**33", images)), "F 8589934592.0"
)
def test_logical(self):
self.assertEqual(pixel(ImageMath.eval("not A", images)), 0)
self.assertEqual(pixel(ImageMath.eval("A and B", images)), "L 2")
self.assertEqual(pixel(ImageMath.eval("A or B", images)), "L 1")
def test_convert(self): def test_logical():
self.assertEqual(pixel(ImageMath.eval("convert(A+B, 'L')", images)), "L 3") assert pixel(ImageMath.eval("not A", images)) == 0
self.assertEqual(pixel(ImageMath.eval("convert(A+B, '1')", images)), "1 0") assert pixel(ImageMath.eval("A and B", images)) == "L 2"
self.assertEqual( assert pixel(ImageMath.eval("A or B", images)) == "L 1"
pixel(ImageMath.eval("convert(A+B, 'RGB')", images)), "RGB (3, 3, 3)"
)
def test_compare(self):
self.assertEqual(pixel(ImageMath.eval("min(A, B)", images)), "I 1")
self.assertEqual(pixel(ImageMath.eval("max(A, B)", images)), "I 2")
self.assertEqual(pixel(ImageMath.eval("A == 1", images)), "I 1")
self.assertEqual(pixel(ImageMath.eval("A == 2", images)), "I 0")
def test_one_image_larger(self): def test_convert():
self.assertEqual(pixel(ImageMath.eval("A+B", A=A2, B=B)), "I 3") assert pixel(ImageMath.eval("convert(A+B, 'L')", images)) == "L 3"
self.assertEqual(pixel(ImageMath.eval("A+B", A=A, B=B2)), "I 3") assert pixel(ImageMath.eval("convert(A+B, '1')", images)) == "1 0"
assert pixel(ImageMath.eval("convert(A+B, 'RGB')", images)) == "RGB (3, 3, 3)"
def test_abs(self):
self.assertEqual(pixel(ImageMath.eval("abs(A)", A=A)), "I 1")
self.assertEqual(pixel(ImageMath.eval("abs(B)", B=B)), "I 2")
def test_binary_mod(self): def test_compare():
self.assertEqual(pixel(ImageMath.eval("A%A", A=A)), "I 0") assert pixel(ImageMath.eval("min(A, B)", images)) == "I 1"
self.assertEqual(pixel(ImageMath.eval("B%B", B=B)), "I 0") assert pixel(ImageMath.eval("max(A, B)", images)) == "I 2"
self.assertEqual(pixel(ImageMath.eval("A%B", A=A, B=B)), "I 1") assert pixel(ImageMath.eval("A == 1", images)) == "I 1"
self.assertEqual(pixel(ImageMath.eval("B%A", A=A, B=B)), "I 0") assert pixel(ImageMath.eval("A == 2", images)) == "I 0"
self.assertEqual(pixel(ImageMath.eval("Z%A", A=A, Z=Z)), "I 0")
self.assertEqual(pixel(ImageMath.eval("Z%B", B=B, Z=Z)), "I 0")
def test_bitwise_invert(self):
self.assertEqual(pixel(ImageMath.eval("~Z", Z=Z)), "I -1")
self.assertEqual(pixel(ImageMath.eval("~A", A=A)), "I -2")
self.assertEqual(pixel(ImageMath.eval("~B", B=B)), "I -3")
def test_bitwise_and(self): def test_one_image_larger():
self.assertEqual(pixel(ImageMath.eval("Z&Z", A=A, Z=Z)), "I 0") assert pixel(ImageMath.eval("A+B", A=A2, B=B)) == "I 3"
self.assertEqual(pixel(ImageMath.eval("Z&A", A=A, Z=Z)), "I 0") assert pixel(ImageMath.eval("A+B", A=A, B=B2)) == "I 3"
self.assertEqual(pixel(ImageMath.eval("A&Z", A=A, Z=Z)), "I 0")
self.assertEqual(pixel(ImageMath.eval("A&A", A=A, Z=Z)), "I 1")
def test_bitwise_or(self):
self.assertEqual(pixel(ImageMath.eval("Z|Z", A=A, Z=Z)), "I 0")
self.assertEqual(pixel(ImageMath.eval("Z|A", A=A, Z=Z)), "I 1")
self.assertEqual(pixel(ImageMath.eval("A|Z", A=A, Z=Z)), "I 1")
self.assertEqual(pixel(ImageMath.eval("A|A", A=A, Z=Z)), "I 1")
def test_bitwise_xor(self): def test_abs():
self.assertEqual(pixel(ImageMath.eval("Z^Z", A=A, Z=Z)), "I 0") assert pixel(ImageMath.eval("abs(A)", A=A)) == "I 1"
self.assertEqual(pixel(ImageMath.eval("Z^A", A=A, Z=Z)), "I 1") assert pixel(ImageMath.eval("abs(B)", B=B)) == "I 2"
self.assertEqual(pixel(ImageMath.eval("A^Z", A=A, Z=Z)), "I 1")
self.assertEqual(pixel(ImageMath.eval("A^A", A=A, Z=Z)), "I 0")
def test_bitwise_leftshift(self):
self.assertEqual(pixel(ImageMath.eval("Z<<0", Z=Z)), "I 0")
self.assertEqual(pixel(ImageMath.eval("Z<<1", Z=Z)), "I 0")
self.assertEqual(pixel(ImageMath.eval("A<<0", A=A)), "I 1")
self.assertEqual(pixel(ImageMath.eval("A<<1", A=A)), "I 2")
def test_bitwise_rightshift(self): def test_binary_mod():
self.assertEqual(pixel(ImageMath.eval("Z>>0", Z=Z)), "I 0") assert pixel(ImageMath.eval("A%A", A=A)) == "I 0"
self.assertEqual(pixel(ImageMath.eval("Z>>1", Z=Z)), "I 0") assert pixel(ImageMath.eval("B%B", B=B)) == "I 0"
self.assertEqual(pixel(ImageMath.eval("A>>0", A=A)), "I 1") assert pixel(ImageMath.eval("A%B", A=A, B=B)) == "I 1"
self.assertEqual(pixel(ImageMath.eval("A>>1", A=A)), "I 0") assert pixel(ImageMath.eval("B%A", A=A, B=B)) == "I 0"
assert pixel(ImageMath.eval("Z%A", A=A, Z=Z)) == "I 0"
assert pixel(ImageMath.eval("Z%B", B=B, Z=Z)) == "I 0"
def test_logical_eq(self):
self.assertEqual(pixel(ImageMath.eval("A==A", A=A)), "I 1")
self.assertEqual(pixel(ImageMath.eval("B==B", B=B)), "I 1")
self.assertEqual(pixel(ImageMath.eval("A==B", A=A, B=B)), "I 0")
self.assertEqual(pixel(ImageMath.eval("B==A", A=A, B=B)), "I 0")
def test_logical_ne(self): def test_bitwise_invert():
self.assertEqual(pixel(ImageMath.eval("A!=A", A=A)), "I 0") assert pixel(ImageMath.eval("~Z", Z=Z)) == "I -1"
self.assertEqual(pixel(ImageMath.eval("B!=B", B=B)), "I 0") assert pixel(ImageMath.eval("~A", A=A)) == "I -2"
self.assertEqual(pixel(ImageMath.eval("A!=B", A=A, B=B)), "I 1") assert pixel(ImageMath.eval("~B", B=B)) == "I -3"
self.assertEqual(pixel(ImageMath.eval("B!=A", A=A, B=B)), "I 1")
def test_logical_lt(self):
self.assertEqual(pixel(ImageMath.eval("A<A", A=A)), "I 0")
self.assertEqual(pixel(ImageMath.eval("B<B", B=B)), "I 0")
self.assertEqual(pixel(ImageMath.eval("A<B", A=A, B=B)), "I 1")
self.assertEqual(pixel(ImageMath.eval("B<A", A=A, B=B)), "I 0")
def test_logical_le(self): def test_bitwise_and():
self.assertEqual(pixel(ImageMath.eval("A<=A", A=A)), "I 1") assert pixel(ImageMath.eval("Z&Z", A=A, Z=Z)) == "I 0"
self.assertEqual(pixel(ImageMath.eval("B<=B", B=B)), "I 1") assert pixel(ImageMath.eval("Z&A", A=A, Z=Z)) == "I 0"
self.assertEqual(pixel(ImageMath.eval("A<=B", A=A, B=B)), "I 1") assert pixel(ImageMath.eval("A&Z", A=A, Z=Z)) == "I 0"
self.assertEqual(pixel(ImageMath.eval("B<=A", A=A, B=B)), "I 0") assert pixel(ImageMath.eval("A&A", A=A, Z=Z)) == "I 1"
def test_logical_gt(self):
self.assertEqual(pixel(ImageMath.eval("A>A", A=A)), "I 0")
self.assertEqual(pixel(ImageMath.eval("B>B", B=B)), "I 0")
self.assertEqual(pixel(ImageMath.eval("A>B", A=A, B=B)), "I 0")
self.assertEqual(pixel(ImageMath.eval("B>A", A=A, B=B)), "I 1")
def test_logical_ge(self): def test_bitwise_or():
self.assertEqual(pixel(ImageMath.eval("A>=A", A=A)), "I 1") assert pixel(ImageMath.eval("Z|Z", A=A, Z=Z)) == "I 0"
self.assertEqual(pixel(ImageMath.eval("B>=B", B=B)), "I 1") assert pixel(ImageMath.eval("Z|A", A=A, Z=Z)) == "I 1"
self.assertEqual(pixel(ImageMath.eval("A>=B", A=A, B=B)), "I 0") assert pixel(ImageMath.eval("A|Z", A=A, Z=Z)) == "I 1"
self.assertEqual(pixel(ImageMath.eval("B>=A", A=A, B=B)), "I 1") assert pixel(ImageMath.eval("A|A", A=A, Z=Z)) == "I 1"
def test_logical_equal(self):
self.assertEqual(pixel(ImageMath.eval("equal(A, A)", A=A)), "I 1")
self.assertEqual(pixel(ImageMath.eval("equal(B, B)", B=B)), "I 1")
self.assertEqual(pixel(ImageMath.eval("equal(Z, Z)", Z=Z)), "I 1")
self.assertEqual(pixel(ImageMath.eval("equal(A, B)", A=A, B=B)), "I 0")
self.assertEqual(pixel(ImageMath.eval("equal(B, A)", A=A, B=B)), "I 0")
self.assertEqual(pixel(ImageMath.eval("equal(A, Z)", A=A, Z=Z)), "I 0")
def test_logical_not_equal(self): def test_bitwise_xor():
self.assertEqual(pixel(ImageMath.eval("notequal(A, A)", A=A)), "I 0") assert pixel(ImageMath.eval("Z^Z", A=A, Z=Z)) == "I 0"
self.assertEqual(pixel(ImageMath.eval("notequal(B, B)", B=B)), "I 0") assert pixel(ImageMath.eval("Z^A", A=A, Z=Z)) == "I 1"
self.assertEqual(pixel(ImageMath.eval("notequal(Z, Z)", Z=Z)), "I 0") assert pixel(ImageMath.eval("A^Z", A=A, Z=Z)) == "I 1"
self.assertEqual(pixel(ImageMath.eval("notequal(A, B)", A=A, B=B)), "I 1") assert pixel(ImageMath.eval("A^A", A=A, Z=Z)) == "I 0"
self.assertEqual(pixel(ImageMath.eval("notequal(B, A)", A=A, B=B)), "I 1")
self.assertEqual(pixel(ImageMath.eval("notequal(A, Z)", A=A, Z=Z)), "I 1")
def test_bitwise_leftshift():
assert pixel(ImageMath.eval("Z<<0", Z=Z)) == "I 0"
assert pixel(ImageMath.eval("Z<<1", Z=Z)) == "I 0"
assert pixel(ImageMath.eval("A<<0", A=A)) == "I 1"
assert pixel(ImageMath.eval("A<<1", A=A)) == "I 2"
def test_bitwise_rightshift():
assert pixel(ImageMath.eval("Z>>0", Z=Z)) == "I 0"
assert pixel(ImageMath.eval("Z>>1", Z=Z)) == "I 0"
assert pixel(ImageMath.eval("A>>0", A=A)) == "I 1"
assert pixel(ImageMath.eval("A>>1", A=A)) == "I 0"
def test_logical_eq():
assert pixel(ImageMath.eval("A==A", A=A)) == "I 1"
assert pixel(ImageMath.eval("B==B", B=B)) == "I 1"
assert pixel(ImageMath.eval("A==B", A=A, B=B)) == "I 0"
assert pixel(ImageMath.eval("B==A", A=A, B=B)) == "I 0"
def test_logical_ne():
assert pixel(ImageMath.eval("A!=A", A=A)) == "I 0"
assert pixel(ImageMath.eval("B!=B", B=B)) == "I 0"
assert pixel(ImageMath.eval("A!=B", A=A, B=B)) == "I 1"
assert pixel(ImageMath.eval("B!=A", A=A, B=B)) == "I 1"
def test_logical_lt():
assert pixel(ImageMath.eval("A<A", A=A)) == "I 0"
assert pixel(ImageMath.eval("B<B", B=B)) == "I 0"
assert pixel(ImageMath.eval("A<B", A=A, B=B)) == "I 1"
assert pixel(ImageMath.eval("B<A", A=A, B=B)) == "I 0"
def test_logical_le():
assert pixel(ImageMath.eval("A<=A", A=A)) == "I 1"
assert pixel(ImageMath.eval("B<=B", B=B)) == "I 1"
assert pixel(ImageMath.eval("A<=B", A=A, B=B)) == "I 1"
assert pixel(ImageMath.eval("B<=A", A=A, B=B)) == "I 0"
def test_logical_gt():
assert pixel(ImageMath.eval("A>A", A=A)) == "I 0"
assert pixel(ImageMath.eval("B>B", B=B)) == "I 0"
assert pixel(ImageMath.eval("A>B", A=A, B=B)) == "I 0"
assert pixel(ImageMath.eval("B>A", A=A, B=B)) == "I 1"
def test_logical_ge():
assert pixel(ImageMath.eval("A>=A", A=A)) == "I 1"
assert pixel(ImageMath.eval("B>=B", B=B)) == "I 1"
assert pixel(ImageMath.eval("A>=B", A=A, B=B)) == "I 0"
assert pixel(ImageMath.eval("B>=A", A=A, B=B)) == "I 1"
def test_logical_equal():
assert pixel(ImageMath.eval("equal(A, A)", A=A)) == "I 1"
assert pixel(ImageMath.eval("equal(B, B)", B=B)) == "I 1"
assert pixel(ImageMath.eval("equal(Z, Z)", Z=Z)) == "I 1"
assert pixel(ImageMath.eval("equal(A, B)", A=A, B=B)) == "I 0"
assert pixel(ImageMath.eval("equal(B, A)", A=A, B=B)) == "I 0"
assert pixel(ImageMath.eval("equal(A, Z)", A=A, Z=Z)) == "I 0"
def test_logical_not_equal():
assert pixel(ImageMath.eval("notequal(A, A)", A=A)) == "I 0"
assert pixel(ImageMath.eval("notequal(B, B)", B=B)) == "I 0"
assert pixel(ImageMath.eval("notequal(Z, Z)", Z=Z)) == "I 0"
assert pixel(ImageMath.eval("notequal(A, B)", A=A, B=B)) == "I 1"
assert pixel(ImageMath.eval("notequal(B, A)", A=A, B=B)) == "I 1"
assert pixel(ImageMath.eval("notequal(A, Z)", A=A, Z=Z)) == "I 1"

View File

@ -1,57 +1,61 @@
import unittest import pytest
from PIL import Image, ImageShow from PIL import Image, ImageShow
from .helper import PillowTestCase, hopper, is_win32, on_ci, on_github_actions from .helper import hopper, is_win32, on_ci, on_github_actions
class TestImageShow(PillowTestCase): def test_sanity():
def test_sanity(self): dir(Image)
dir(Image) dir(ImageShow)
dir(ImageShow)
def test_register(self):
# Test registering a viewer that is not a class
ImageShow.register("not a class")
# Restore original state def test_register():
ImageShow._viewers.pop() # Test registering a viewer that is not a class
ImageShow.register("not a class")
def test_viewer_show(self): # Restore original state
class TestViewer(ImageShow.Viewer): ImageShow._viewers.pop()
methodCalled = False
def show_image(self, image, **options):
self.methodCalled = True
return True
viewer = TestViewer() def test_viewer_show():
ImageShow.register(viewer, -1) class TestViewer(ImageShow.Viewer):
methodCalled = False
for mode in ("1", "I;16", "LA", "RGB", "RGBA"): def show_image(self, image, **options):
with hopper() as im: self.methodCalled = True
self.assertTrue(ImageShow.show(im)) return True
self.assertTrue(viewer.methodCalled)
# Restore original state viewer = TestViewer()
ImageShow._viewers.pop(0) ImageShow.register(viewer, -1)
@unittest.skipUnless( for mode in ("1", "I;16", "LA", "RGB", "RGBA"):
on_ci() and not (is_win32() and on_github_actions()), with hopper() as im:
"Only run on CIs; hangs on Windows on GitHub Actions", assert ImageShow.show(im)
) assert viewer.methodCalled
def test_show(self):
for mode in ("1", "I;16", "LA", "RGB", "RGBA"):
im = hopper(mode)
self.assertTrue(ImageShow.show(im))
def test_viewer(self): # Restore original state
viewer = ImageShow.Viewer() ImageShow._viewers.pop(0)
self.assertIsNone(viewer.get_format(None))
self.assertRaises(NotImplementedError, viewer.get_command, None) @pytest.mark.skipif(
not on_ci() or (is_win32() and on_github_actions()),
reason="Only run on CIs; hangs on Windows on GitHub Actions",
)
def test_show():
for mode in ("1", "I;16", "LA", "RGB", "RGBA"):
im = hopper(mode)
assert ImageShow.show(im)
def test_viewers(self):
for viewer in ImageShow._viewers: def test_viewer():
viewer.get_command("test.jpg") viewer = ImageShow.Viewer()
assert viewer.get_format(None) is None
with pytest.raises(NotImplementedError):
viewer.get_command(None)
def test_viewers():
for viewer in ImageShow._viewers:
viewer.get_command("test.jpg")

View File

@ -3,7 +3,7 @@ from io import BytesIO
from PIL import Image, ImageWin from PIL import Image, ImageWin
from .helper import PillowTestCase, hopper, is_win32 from .helper import hopper, is_win32
# see https://github.com/python-pillow/Pillow/pull/1431#issuecomment-144692652 # see https://github.com/python-pillow/Pillow/pull/1431#issuecomment-144692652
@ -81,34 +81,33 @@ if is_win32():
memcpy(bp + bf.bfOffBits, pixels, bi.biSizeImage) memcpy(bp + bf.bfOffBits, pixels, bi.biSizeImage)
return bytearray(buf) return bytearray(buf)
class TestImageWinPointers(PillowTestCase): def test_pointer(tmp_path):
def test_pointer(self): im = hopper()
im = hopper() (width, height) = im.size
(width, height) = im.size opath = str(tmp_path / "temp.png")
opath = self.tempfile("temp.png") imdib = ImageWin.Dib(im)
imdib = ImageWin.Dib(im)
hdr = BITMAPINFOHEADER() hdr = BITMAPINFOHEADER()
hdr.biSize = ctypes.sizeof(hdr) hdr.biSize = ctypes.sizeof(hdr)
hdr.biWidth = width hdr.biWidth = width
hdr.biHeight = height hdr.biHeight = height
hdr.biPlanes = 1 hdr.biPlanes = 1
hdr.biBitCount = 32 hdr.biBitCount = 32
hdr.biCompression = BI_RGB hdr.biCompression = BI_RGB
hdr.biSizeImage = width * height * 4 hdr.biSizeImage = width * height * 4
hdr.biClrUsed = 0 hdr.biClrUsed = 0
hdr.biClrImportant = 0 hdr.biClrImportant = 0
hdc = CreateCompatibleDC(None) hdc = CreateCompatibleDC(None)
pixels = ctypes.c_void_p() pixels = ctypes.c_void_p()
dib = CreateDIBSection( dib = CreateDIBSection(
hdc, ctypes.byref(hdr), DIB_RGB_COLORS, ctypes.byref(pixels), None, 0 hdc, ctypes.byref(hdr), DIB_RGB_COLORS, ctypes.byref(pixels), None, 0
) )
SelectObject(hdc, dib) SelectObject(hdc, dib)
imdib.expose(hdc) imdib.expose(hdc)
bitmap = serialize_dib(hdr, pixels) bitmap = serialize_dib(hdr, pixels)
DeleteObject(dib) DeleteObject(dib)
DeleteDC(hdc) DeleteDC(hdc)
Image.open(BytesIO(bitmap)).save(opath) Image.open(BytesIO(bitmap)).save(opath)

View File

@ -1,38 +1,32 @@
import unittest import pytest
from PIL import Image from PIL import Image
from .helper import PillowTestCase
def test_setmode():
class TestLibImage(PillowTestCase): im = Image.new("L", (1, 1), 255)
def test_setmode(self): im.im.setmode("1")
assert im.im.getpixel((0, 0)) == 255
im.im.setmode("L")
assert im.im.getpixel((0, 0)) == 255
im = Image.new("L", (1, 1), 255) im = Image.new("1", (1, 1), 1)
im.im.setmode("1") im.im.setmode("L")
self.assertEqual(im.im.getpixel((0, 0)), 255) assert im.im.getpixel((0, 0)) == 255
im.im.setmode("1")
assert im.im.getpixel((0, 0)) == 255
im = Image.new("RGB", (1, 1), (1, 2, 3))
im.im.setmode("RGB")
assert im.im.getpixel((0, 0)) == (1, 2, 3)
im.im.setmode("RGBA")
assert im.im.getpixel((0, 0)) == (1, 2, 3, 255)
im.im.setmode("RGBX")
assert im.im.getpixel((0, 0)) == (1, 2, 3, 255)
im.im.setmode("RGB")
assert im.im.getpixel((0, 0)) == (1, 2, 3)
with pytest.raises(ValueError):
im.im.setmode("L") im.im.setmode("L")
self.assertEqual(im.im.getpixel((0, 0)), 255) with pytest.raises(ValueError):
im.im.setmode("RGBABCDE")
im = Image.new("1", (1, 1), 1)
im.im.setmode("L")
self.assertEqual(im.im.getpixel((0, 0)), 255)
im.im.setmode("1")
self.assertEqual(im.im.getpixel((0, 0)), 255)
im = Image.new("RGB", (1, 1), (1, 2, 3))
im.im.setmode("RGB")
self.assertEqual(im.im.getpixel((0, 0)), (1, 2, 3))
im.im.setmode("RGBA")
self.assertEqual(im.im.getpixel((0, 0)), (1, 2, 3, 255))
im.im.setmode("RGBX")
self.assertEqual(im.im.getpixel((0, 0)), (1, 2, 3, 255))
im.im.setmode("RGB")
self.assertEqual(im.im.getpixel((0, 0)), (1, 2, 3))
self.assertRaises(ValueError, im.im.setmode, "L")
self.assertRaises(ValueError, im.im.setmode, "RGBABCDE")
if __name__ == "__main__":
unittest.main()

View File

@ -4,57 +4,55 @@ from io import StringIO
from PIL import Image, PSDraw from PIL import Image, PSDraw
from .helper import PillowTestCase
def _create_document(ps):
title = "hopper"
box = (1 * 72, 2 * 72, 7 * 72, 10 * 72) # in points
ps.begin_document(title)
# draw diagonal lines in a cross
ps.line((1 * 72, 2 * 72), (7 * 72, 10 * 72))
ps.line((7 * 72, 2 * 72), (1 * 72, 10 * 72))
# draw the image (75 dpi)
with Image.open("Tests/images/hopper.ppm") as im:
ps.image(box, im, 75)
ps.rectangle(box)
# draw title
ps.setfont("Courier", 36)
ps.text((3 * 72, 4 * 72), title)
ps.end_document()
class TestPsDraw(PillowTestCase): def test_draw_postscript(tmp_path):
def _create_document(self, ps): # Based on Pillow tutorial, but there is no textsize:
title = "hopper" # https://pillow.readthedocs.io/en/latest/handbook/tutorial.html#drawing-postscript
box = (1 * 72, 2 * 72, 7 * 72, 10 * 72) # in points
ps.begin_document(title) # Arrange
tempfile = str(tmp_path / "temp.ps")
with open(tempfile, "wb") as fp:
# Act
ps = PSDraw.PSDraw(fp)
_create_document(ps)
# draw diagonal lines in a cross # Assert
ps.line((1 * 72, 2 * 72), (7 * 72, 10 * 72)) # Check non-zero file was created
ps.line((7 * 72, 2 * 72), (1 * 72, 10 * 72)) assert os.path.isfile(tempfile)
assert os.path.getsize(tempfile) > 0
# draw the image (75 dpi)
with Image.open("Tests/images/hopper.ppm") as im:
ps.image(box, im, 75)
ps.rectangle(box)
# draw title def test_stdout():
ps.setfont("Courier", 36) # Temporarily redirect stdout
ps.text((3 * 72, 4 * 72), title) old_stdout = sys.stdout
sys.stdout = mystdout = StringIO()
ps.end_document() ps = PSDraw.PSDraw()
_create_document(ps)
def test_draw_postscript(self): # Reset stdout
sys.stdout = old_stdout
# Based on Pillow tutorial, but there is no textsize: assert mystdout.getvalue() != ""
# https://pillow.readthedocs.io/en/latest/handbook/tutorial.html#drawing-postscript
# Arrange
tempfile = self.tempfile("temp.ps")
with open(tempfile, "wb") as fp:
# Act
ps = PSDraw.PSDraw(fp)
self._create_document(ps)
# Assert
# Check non-zero file was created
self.assertTrue(os.path.isfile(tempfile))
self.assertGreater(os.path.getsize(tempfile), 0)
def test_stdout(self):
# Temporarily redirect stdout
old_stdout = sys.stdout
sys.stdout = mystdout = StringIO()
ps = PSDraw.PSDraw()
self._create_document(ps)
# Reset stdout
sys.stdout = old_stdout
self.assertNotEqual(mystdout.getvalue(), "")

View File

@ -1,32 +1,28 @@
import unittest import pytest
from PIL import __version__ from PIL import __version__
from .helper import PillowTestCase
try: try:
import pyroma import pyroma
except ImportError: except ImportError:
pyroma = None pyroma = None
@unittest.skipIf(pyroma is None, "Pyroma is not installed") @pytest.mark.skipif(pyroma is None, reason="Pyroma is not installed")
class TestPyroma(PillowTestCase): def test_pyroma():
def test_pyroma(self): # Arrange
# Arrange data = pyroma.projectdata.get_data(".")
data = pyroma.projectdata.get_data(".")
# Act # Act
rating = pyroma.ratings.rate(data) rating = pyroma.ratings.rate(data)
# Assert # Assert
if "rc" in __version__: if "rc" in __version__:
# Pyroma needs to chill about RC versions and not kill all our tests. # Pyroma needs to chill about RC versions and not kill all our tests.
self.assertEqual( assert rating == (
rating, 9,
(9, ["The package's version number does not comply with PEP-386."]), ["The package's version number does not comply with PEP-386."],
) )
else: else:
# Should have a perfect score # Should have a perfect score
self.assertEqual(rating, (10, [])) assert rating == (10, [])