From 3f252162f6b58fdf04586b71455612f74ac963eb Mon Sep 17 00:00:00 2001 From: Gabriel Selzer Date: Tue, 7 Jan 2025 14:59:03 -0600 Subject: [PATCH] Make try/except granular --- src/imagej/images.py | 39 ++++++++++++++++++++++++---------- tests/test_image_conversion.py | 10 +++++++-- 2 files changed, 36 insertions(+), 13 deletions(-) diff --git a/src/imagej/images.py b/src/imagej/images.py index 5073e5c7..f3709cdf 100644 --- a/src/imagej/images.py +++ b/src/imagej/images.py @@ -140,31 +140,48 @@ def copy_rai_into_ndarray( if not is_arraylike(narr): raise TypeError("narr is not arraylike") - try: - # Check imglib2 version for fast copy availability. - imglib2_version = sj.get_version(jc.RandomAccessibleInterval) - if sj.is_version_at_least(imglib2_version, "5.9.0"): + # Suppose all mechanisms fail. Any one of these might be the one that was + # "supposed" to work. + failure_exceptions = [] + + # Check imglib2 version for fast copy availability. + imglib2_version = sj.get_version(jc.RandomAccessibleInterval) + if sj.is_version_at_least(imglib2_version, "5.9.0"): + try: # ImgLib2 is new enough to use net.imglib2.util.ImgUtil.copy. ImgUtil = sj.jimport("net.imglib2.util.ImgUtil") ImgUtil.copy(rai, sj.to_java(narr)) return narr - - # Check imagej-common version for fast copy availability. - imagej_common_version = sj.get_version(jc.Dataset) - if sj.is_version_at_least(imagej_common_version, "0.30.0"): + except JException as exc: + # Try another method + failure_exceptions.append(exc) + + # Check imagej-common version for fast copy availability. + imagej_common_version = sj.get_version(jc.Dataset) + if sj.is_version_at_least(imagej_common_version, "0.30.0"): + try: # ImageJ Common is new enough to use (deprecated) # net.imagej.util.Images.copy. Images = sj.jimport("net.imagej.util.Images") Images.copy(rai, sj.to_java(narr)) return narr - except JException: - pass + except JException as exc: + # Try another method + failure_exceptions.append(exc) # Fall back to copying with ImageJ Ops's copy.rai op. In theory, Ops # should always be faster. But in practice, the copy.rai operation is # slower than the hardcoded ones above. If we were to fix Ops to be # fast always, we could eliminate the above special casing. - ij.op().run("copy.rai", sj.to_java(narr), rai) + try: + ij.op().run("copy.rai", sj.to_java(narr), rai) + return + except JException as exc: + # Try another method + failure_exceptions.append(exc) + + # Failed + raise Exception("Could not copy rai into ndarray", *failure_exceptions) def dtype(image_or_type) -> np.dtype: diff --git a/tests/test_image_conversion.py b/tests/test_image_conversion.py index 188b4691..93f97069 100644 --- a/tests/test_image_conversion.py +++ b/tests/test_image_conversion.py @@ -5,7 +5,6 @@ import pytest import scyjava as sj import xarray as xr -from jpype import JLong import imagej.convert as convert import imagej.dims as dims @@ -378,9 +377,16 @@ def test_dataset_converts_to_xarray(ij): def test_bittype_img_to_ndarray(ij): + ops_version = sj.get_version(sj.jimport("net.imagej.ops.OpService")) + if not sj.is_version_at_least(ops_version, "2.1.0"): + pytest.skip("Fails without ImageJ Ops >= 2.1.0") + ArrayImgs = sj.jimport("net.imglib2.img.array.ArrayImgs") - dims = JLong[:] @ [10, 10, 10] + # NB ArrayImgs requires a long[] - construct long[] {10, 10, 10} + dims = sj.jarray("j", 3) + dims[:] = [10] * 3 j_img = ArrayImgs.bits(dims) + p_img = ij.py.from_java(j_img) assert p_img.dtype == np.bool_