2024-08-25 17:54:00 +03:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2025-01-22 00:41:54 +03:00
|
|
|
from typing import Any # undone
|
2024-08-25 17:54:00 +03:00
|
|
|
|
|
|
|
import pytest
|
|
|
|
|
|
|
|
from PIL import Image
|
|
|
|
|
2025-01-22 00:41:54 +03:00
|
|
|
from .helper import (
|
|
|
|
assert_deep_equal,
|
|
|
|
assert_image_equal,
|
|
|
|
hopper,
|
|
|
|
)
|
2024-08-25 17:54:00 +03:00
|
|
|
|
|
|
|
pyarrow = pytest.importorskip("pyarrow", reason="PyArrow not installed")
|
|
|
|
|
|
|
|
TEST_IMAGE_SIZE = (10, 10)
|
|
|
|
|
|
|
|
|
|
|
|
def _test_img_equals_pyarray(img: Image.Image, arr: Any, mask) -> None:
|
|
|
|
assert img.height * img.width == len(arr)
|
|
|
|
px = img.load()
|
|
|
|
assert px is not None
|
|
|
|
for x in range(0, img.size[0], int(img.size[0] / 10)):
|
|
|
|
for y in range(0, img.size[1], int(img.size[1] / 10)):
|
|
|
|
if mask:
|
|
|
|
for ix, elt in enumerate(mask):
|
2025-01-22 00:41:54 +03:00
|
|
|
assert px[x, y][ix] == arr[y * img.width + x].as_py()[elt]
|
2024-08-25 17:54:00 +03:00
|
|
|
else:
|
|
|
|
assert_deep_equal(px[x, y], arr[y * img.width + x].as_py())
|
|
|
|
|
|
|
|
|
|
|
|
# really hard to get a non-nullable list type
|
2025-01-22 00:41:54 +03:00
|
|
|
fl_uint8_4_type = pyarrow.field(
|
|
|
|
"_", pyarrow.list_(pyarrow.field("_", pyarrow.uint8()).with_nullable(False), 4)
|
|
|
|
).type
|
|
|
|
|
2024-08-25 17:54:00 +03:00
|
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
|
|
"mode, dtype, mask",
|
|
|
|
(
|
|
|
|
("L", pyarrow.uint8(), None),
|
|
|
|
("I", pyarrow.int32(), None),
|
|
|
|
("F", pyarrow.float32(), None),
|
2025-01-22 00:41:54 +03:00
|
|
|
("LA", fl_uint8_4_type, [0, 3]),
|
|
|
|
("RGB", fl_uint8_4_type, [0, 1, 2]),
|
2024-08-25 17:54:00 +03:00
|
|
|
("RGBA", fl_uint8_4_type, None),
|
|
|
|
("RGBX", fl_uint8_4_type, None),
|
|
|
|
("CMYK", fl_uint8_4_type, None),
|
2025-01-22 00:41:54 +03:00
|
|
|
("YCbCr", fl_uint8_4_type, [0, 1, 2]),
|
|
|
|
("HSV", fl_uint8_4_type, [0, 1, 2]),
|
2024-08-25 17:54:00 +03:00
|
|
|
),
|
|
|
|
)
|
2025-01-22 00:41:54 +03:00
|
|
|
def test_to_array(mode: str, dtype: Any, mask: Any) -> None:
|
2024-08-25 17:54:00 +03:00
|
|
|
img = hopper(mode)
|
|
|
|
|
|
|
|
# Resize to non-square
|
|
|
|
img = img.crop((3, 0, 124, 127))
|
|
|
|
assert img.size == (121, 127)
|
|
|
|
|
|
|
|
arr = pyarrow.array(img)
|
|
|
|
_test_img_equals_pyarray(img, arr, mask)
|
|
|
|
assert arr.type == dtype
|
2024-08-25 18:00:08 +03:00
|
|
|
|
2025-01-22 00:27:41 +03:00
|
|
|
reloaded = Image.fromarrow(arr, mode, img.size)
|
|
|
|
|
|
|
|
assert reloaded
|
|
|
|
|
|
|
|
assert_image_equal(img, reloaded)
|
2024-08-25 18:00:08 +03:00
|
|
|
|
2025-01-22 00:41:54 +03:00
|
|
|
|
2024-08-25 18:00:08 +03:00
|
|
|
def test_lifetime():
|
|
|
|
# valgrind shouldn't error out here.
|
|
|
|
# arrays should be accessible after the image is deleted.
|
|
|
|
|
2025-01-22 00:41:54 +03:00
|
|
|
img = hopper("L")
|
2024-08-25 18:00:08 +03:00
|
|
|
|
|
|
|
arr_1 = pyarrow.array(img)
|
|
|
|
arr_2 = pyarrow.array(img)
|
|
|
|
|
2025-01-22 00:41:54 +03:00
|
|
|
del img
|
2024-08-25 18:00:08 +03:00
|
|
|
|
|
|
|
assert arr_1.sum().as_py() > 0
|
2025-01-22 00:41:54 +03:00
|
|
|
del arr_1
|
2024-08-25 18:00:08 +03:00
|
|
|
|
|
|
|
assert arr_2.sum().as_py() > 0
|
2025-01-22 00:41:54 +03:00
|
|
|
del arr_2
|
|
|
|
|
2024-08-25 18:00:08 +03:00
|
|
|
|
|
|
|
def test_lifetime2():
|
|
|
|
# valgrind shouldn't error out here.
|
|
|
|
# img should remain after the arrays are collected.
|
|
|
|
|
2025-01-22 00:41:54 +03:00
|
|
|
img = hopper("L")
|
2024-08-25 18:00:08 +03:00
|
|
|
|
|
|
|
arr_1 = pyarrow.array(img)
|
|
|
|
arr_2 = pyarrow.array(img)
|
|
|
|
|
|
|
|
assert arr_1.sum().as_py() > 0
|
2025-01-22 00:41:54 +03:00
|
|
|
del arr_1
|
2024-08-25 18:00:08 +03:00
|
|
|
|
|
|
|
assert arr_2.sum().as_py() > 0
|
2025-01-22 00:41:54 +03:00
|
|
|
del arr_2
|
2024-08-25 18:00:08 +03:00
|
|
|
|
|
|
|
img2 = img.copy()
|
|
|
|
px = img2.load()
|
2025-01-22 00:41:54 +03:00
|
|
|
assert isinstance(px[0, 0], int)
|