diff --git a/tests/artifacts/README.MD b/tests/artifacts/README.MD new file mode 100644 index 0000000..c330c6f --- /dev/null +++ b/tests/artifacts/README.MD @@ -0,0 +1,37 @@ +# Test artifacts + +This directory contains test artifacts for the `readlif` package. These artifacts consist of example LIF files from various sources. In some cases, they are paired with expected outputs in the form of TIFF files containing the image data for selected 2D (X-Y) planes in the corresponding LIF file. In these cases, there is one TIFF file for each plane, and the TIFF files are named according to the plane they represent. + +## About the LIF files + +These are brief informal notes about the contents of the LIF files in this directory. They were determined by manually opening the LIF files in Fiji using the BioFormats plugin and examining the metadata. + +### `xyzt-example/xyzt-example.lif` + +Contains a single image. 1024 x 1024. 8-bit. Shape: 2C, 3T, 3Z. + +### `xz-example/xz-example.lif` + +Contains three images. All three are XZ in the first two dimensions. + +BioFormats labels: + +- xzt: 128 x 128; 2C x 20T +- xzy: 128 x 128; 2C x 23Z (note: the 'y' dimension in 'xzy' is loaded as Z by BioFormats) +- xz: 512 x 512; 2C. + +### `misc/LeicaLASX_wavelength-sweep_example.lif` + +Contains three images. All 64 x 64. The time dimension appears to be a Lambda scan but the metadata implies it was a time dimension (it contains `Plane` elements with `"deltaT"` attributes). BioFormats loads the images as XYZT. + +The BioFormats labels of the three images are (in pseudo-regex): + +``` +x y (lambdaEmi|lambdaExc): 64x64; (20T|15T|(11Z x 20T)) +``` + +### `misc/new_lasx.lif` + +Contains a single image. 1024 x 1024. 8-bit. Shape: 2C, 39Z. + +This file was added in #36; there is no other documentation of what it contains. diff --git a/tests/artifacts/lif-files/LeicaLASX_wavelength-sweep_example.lif b/tests/artifacts/misc/LeicaLASX_wavelength-sweep_example.lif similarity index 100% rename from tests/artifacts/lif-files/LeicaLASX_wavelength-sweep_example.lif rename to tests/artifacts/misc/LeicaLASX_wavelength-sweep_example.lif diff --git a/tests/artifacts/lif-files/new_lasx.lif b/tests/artifacts/misc/new_lasx.lif similarity index 100% rename from tests/artifacts/lif-files/new_lasx.lif rename to tests/artifacts/misc/new_lasx.lif diff --git a/tests/artifacts/misc/valid-tiff.tif b/tests/artifacts/misc/valid-tiff.tif new file mode 100644 index 0000000..df72bca Binary files /dev/null and b/tests/artifacts/misc/valid-tiff.tif differ diff --git a/tests/artifacts/tiff-files/xem_y32.tif b/tests/artifacts/misc/xem_y32.tif similarity index 100% rename from tests/artifacts/tiff-files/xem_y32.tif rename to tests/artifacts/misc/xem_y32.tif diff --git a/tests/artifacts/tiff-files/c0z0t0.tif b/tests/artifacts/xyzt-example/expected-outputs/c0z0t0.tif similarity index 100% rename from tests/artifacts/tiff-files/c0z0t0.tif rename to tests/artifacts/xyzt-example/expected-outputs/c0z0t0.tif diff --git a/tests/artifacts/tiff-files/c0z2t0.tif b/tests/artifacts/xyzt-example/expected-outputs/c0z2t0.tif similarity index 100% rename from tests/artifacts/tiff-files/c0z2t0.tif rename to tests/artifacts/xyzt-example/expected-outputs/c0z2t0.tif diff --git a/tests/artifacts/tiff-files/c0z2t2.tif b/tests/artifacts/xyzt-example/expected-outputs/c0z2t2.tif similarity index 100% rename from tests/artifacts/tiff-files/c0z2t2.tif rename to tests/artifacts/xyzt-example/expected-outputs/c0z2t2.tif diff --git a/tests/artifacts/tiff-files/c1z0t0.tif b/tests/artifacts/xyzt-example/expected-outputs/c1z0t0.tif similarity index 100% rename from tests/artifacts/tiff-files/c1z0t0.tif rename to tests/artifacts/xyzt-example/expected-outputs/c1z0t0.tif diff --git a/tests/artifacts/lif-files/xyzt_test.lif b/tests/artifacts/xyzt-example/xyzt-example.lif similarity index 100% rename from tests/artifacts/lif-files/xyzt_test.lif rename to tests/artifacts/xyzt-example/xyzt-example.lif diff --git a/tests/artifacts/tiff-files/xz_c0_t0.tif b/tests/artifacts/xz-example/expected-outputs/c0t0.tif similarity index 100% rename from tests/artifacts/tiff-files/xz_c0_t0.tif rename to tests/artifacts/xz-example/expected-outputs/c0t0.tif diff --git a/tests/artifacts/tiff-files/xz_c1_t8.tif b/tests/artifacts/xz-example/expected-outputs/c1t8.tif similarity index 100% rename from tests/artifacts/tiff-files/xz_c1_t8.tif rename to tests/artifacts/xz-example/expected-outputs/c1t8.tif diff --git a/tests/artifacts/lif-files/testdata_2channel_xz.lif b/tests/artifacts/xz-example/xz-example.lif similarity index 100% rename from tests/artifacts/lif-files/testdata_2channel_xz.lif rename to tests/artifacts/xz-example/xz-example.lif diff --git a/tests/conftest.py b/tests/conftest.py new file mode 100644 index 0000000..4e91e62 --- /dev/null +++ b/tests/conftest.py @@ -0,0 +1,57 @@ +import pathlib + +import pytest + + +@pytest.fixture +def artifacts_dirpath(): + return pathlib.Path(__file__).parent / "artifacts" + + +@pytest.fixture +def valid_lif_filepath(artifacts_dirpath): + """ + The filepath to a LIF file that is valid and can be read by the reader. + """ + return artifacts_dirpath / "misc" / "new_lasx.lif" + + +@pytest.fixture +def valid_single_image_lif_filepath(artifacts_dirpath): + """ + The filepath to a LIF file that contains only one image. + """ + return artifacts_dirpath / "xyzt-example" / "xyzt-example.lif" + + +@pytest.fixture +def valid_multi_image_lif_filepath(artifacts_dirpath): + """ + The filepath to a LIF file that contains multiple images. + """ + return artifacts_dirpath / "xz-example" / "xz-example.lif" + + +@pytest.fixture +def valid_tiff_filepath(artifacts_dirpath): + """ + The filepath to a TIFF file that is valid. + """ + return artifacts_dirpath / "misc" / "valid-tiff.tif" + + +@pytest.fixture +def xyzt_example_lif_filepath(artifacts_dirpath): + """ + The filepath to a LIF file that contains XYZT data. + """ + return artifacts_dirpath / "xyzt-example" / "xyzt-example.lif" + + +@pytest.fixture +def xz_example_lif_filepath(artifacts_dirpath): + """ + The filepath to a LIF file that contains an image in which the second dimension + is Z rather than Y. + """ + return artifacts_dirpath / "xz-example" / "xz-example.lif" diff --git a/tests/test_reader.py b/tests/test_reader.py new file mode 100644 index 0000000..5bd6bc7 --- /dev/null +++ b/tests/test_reader.py @@ -0,0 +1,146 @@ +import pytest +from PIL import Image + +from readlif.reader import LifFile + + +def test_lif_file_with_file_path(valid_lif_filepath): + lif_image = LifFile(valid_lif_filepath).get_image(0) + # This should not raise an error. + lif_image.get_frame(z=0, t=0, c=0) + + +def test_lif_file_with_file_buffer(valid_lif_filepath): + with open(valid_lif_filepath, "rb") as file: + lif_image = LifFile(file).get_image(0) + # This should not raise an error. + lif_image.get_frame(z=0, t=0, c=0) + + +def test_lif_file_with_non_lif_files(tmp_path, artifacts_dirpath): + with open(tmp_path / "not-a-lif-file.txt", "w") as file: + with pytest.raises(ValueError): + LifFile(file) + + with pytest.raises(ValueError): + LifFile(artifacts_dirpath / "misc" / "valid-tiff.tif") + + +def test_lif_file_with_single_image_lif(valid_single_image_lif_filepath): + lif_file = LifFile(valid_single_image_lif_filepath) + assert repr(lif_file) == "'LifFile object with 1 image'" + assert len(lif_file.image_list) == 1 + with pytest.raises(ValueError): + lif_file.get_image(1) + + +def test_lif_file_get_iter_image(valid_single_image_lif_filepath, valid_multi_image_lif_filepath): + images = [image for image in LifFile(valid_single_image_lif_filepath).get_iter_image()] + assert len(images) == 1 + + images = [image for image in LifFile(valid_multi_image_lif_filepath).get_iter_image()] + assert len(images) > 1 + + +def test_lif_file_with_new_lasx(artifacts_dirpath): + """ + TODO(KC): figure out what is special or "new" about the "new_lasx.lif" file. + """ + lif_file = LifFile(artifacts_dirpath / "misc" / "new_lasx.lif") + assert len(lif_file.image_list) == 1 + + +def test_lif_image_get_frame_out_of_range_args(xyzt_example_lif_filepath): + lif_image = LifFile(xyzt_example_lif_filepath).get_image(0) + + for dimension_kwarg, lif_image_attr in [ + ("z", "nz"), + ("t", "nt"), + ("c", "channels"), + ("m", "n_mosaic"), + ]: + max_index_for_dimension = getattr(lif_image, lif_image_attr) - 1 + lif_image.get_frame(**{dimension_kwarg: max_index_for_dimension}) + with pytest.raises(ValueError): + lif_image.get_frame(**{dimension_kwarg: max_index_for_dimension + 1}) + + # TODO: use a better-justified out of range index for this test. + with pytest.raises(ValueError): + lif_image._get_item(100) + + +def test_lif_image_settings(xz_example_lif_filepath): + """ + TODO: expand this test to cover more attributes and use more than one LIF file. + """ + lif_image = LifFile(xz_example_lif_filepath).get_image(0) + assert lif_image.settings["ObjectiveNumber"] == "11506353" + + +def test_lif_image_attributes(xyzt_example_lif_filepath): + """ + TODO: expand this test to cover more attributes and use more than one LIF file. + """ + lif_image = LifFile(xyzt_example_lif_filepath).get_image(0) + assert lif_image.scale[0] == pytest.approx(9.8709062997224) + assert lif_image.bit_depth[0] == 8 + + +def test_lif_image_get_frame(xyzt_example_lif_filepath): + lif_image = LifFile(xyzt_example_lif_filepath).get_image(0) + czt_pairs = [[0, 0, 0], [0, 2, 0], [0, 2, 2], [1, 0, 0]] + for c, z, t in czt_pairs: + reference_image = Image.open( + xyzt_example_lif_filepath.parent / "expected-outputs" / f"c{c}z{z}t{t}.tif" + ) + lif_image_frame = lif_image.get_frame(z=z, t=t, c=c) + assert lif_image_frame.tobytes() == reference_image.tobytes() + + +def test_lif_image_get_plane(xyzt_example_lif_filepath): + """ + TODO: eliminate duplication with `test_lif_image_get_frame`. + """ + lif_image = LifFile(xyzt_example_lif_filepath).get_image(0) + czt_pairs = [[0, 0, 0], [0, 2, 0], [0, 2, 2], [1, 0, 0]] + for c, z, t in czt_pairs: + reference_image = Image.open( + xyzt_example_lif_filepath.parent / "expected-outputs" / f"c{c}z{z}t{t}.tif" + ) + lif_image_plane = lif_image.get_plane(c=c, requested_dims={3: z, 4: t}) + assert lif_image_plane.tobytes() == reference_image.tobytes() + + +def test_lif_image_get_plane_on_xz_image(xz_example_lif_filepath): + lif_image = LifFile(xz_example_lif_filepath).get_image(0) + for c, t in [(0, 0), (1, 8)]: + reference_image = Image.open( + xz_example_lif_filepath.parent / "expected-outputs" / f"c{c}t{t}.tif" + ) + lif_image_plane = lif_image.get_plane(c=c, requested_dims={4: t}) + assert lif_image_plane.tobytes() == reference_image.tobytes() + + +def test_lif_image_get_plane_nonexistent_plane(artifacts_dirpath): + """ + TODO: determine if another, less mysterious LIF file can be used for this test. + """ + lif_image = LifFile( + artifacts_dirpath / "misc" / "LeicaLASX_wavelength-sweep_example.lif" + ).get_image(0) + with pytest.raises(NotImplementedError): + lif_image.get_plane(display_dims=(1, 5), c=0, requested_dims={2: 31}) + + +def test_lif_image_repr(xyzt_example_lif_filepath): + lif_image = LifFile(xyzt_example_lif_filepath).get_image(0) + assert ( + repr(lif_image) == "'LifImage object with dimensions: Dims(x=1024, y=1024, z=3, t=3, m=1)'" + ) + + +def test_lif_image_get_iters(xyzt_example_lif_filepath): + lif_image = LifFile(xyzt_example_lif_filepath).get_image(0) + assert len([image for image in lif_image.get_iter_c()]) == 2 + assert len([image for image in lif_image.get_iter_t()]) == 3 + assert len([image for image in lif_image.get_iter_z()]) == 3 diff --git a/tests/test_readlif.py b/tests/test_readlif.py deleted file mode 100644 index b217f4d..0000000 --- a/tests/test_readlif.py +++ /dev/null @@ -1,143 +0,0 @@ -import pathlib - -import pytest -from PIL import Image - -from readlif.reader import LifFile -from readlif.utilities import get_xml - -ARTIFACTS_DIRPATH = pathlib.Path(__file__).parent / "artifacts" -TIFF_FILES_DIRPATH = ARTIFACTS_DIRPATH / "tiff-files" -LIF_FILES_DIRPATH = ARTIFACTS_DIRPATH / "lif-files" - - -def test_image_loading(): - test_array = [[0, 0, 0], [0, 2, 0], [0, 2, 2], [1, 0, 0]] - for i in test_array: - c = str(i[0]) - z = str(i[1]) - t = str(i[2]) - ref = Image.open(TIFF_FILES_DIRPATH / f"c{c}z{z}t{t}.tif") - obj = LifFile(LIF_FILES_DIRPATH / "xyzt_test.lif").get_image(0) - test = obj.get_frame(z=z, t=t, c=c) - assert test.tobytes() == ref.tobytes() - - -def test_image_loading_from_buffer(): - test_array = [[0, 0, 0], [0, 2, 0], [0, 2, 2], [1, 0, 0]] - for i in test_array: - c = str(i[0]) - z = str(i[1]) - t = str(i[2]) - ref = Image.open(TIFF_FILES_DIRPATH / f"c{c}z{z}t{t}.tif") - - with open(LIF_FILES_DIRPATH / "xyzt_test.lif", "rb") as open_f: - obj = LifFile(open_f).get_image(0) - test = obj.get_frame(z=z, t=t, c=c) - assert test.tobytes() == ref.tobytes() - - -def test_xml_header(): - _, test = get_xml(LIF_FILES_DIRPATH / "xyzt_test.lif") - assert test[:50] == '')