Skip to content

Commit

Permalink
Merge branch 'main' into ndinterp_allow_dask_array_subclasses
Browse files Browse the repository at this point in the history
  • Loading branch information
GenevieveBuckley committed May 14, 2024
2 parents e4d3c6e + 54a3925 commit a4fd9a7
Show file tree
Hide file tree
Showing 11 changed files with 148 additions and 142 deletions.
2 changes: 1 addition & 1 deletion continuous_integration/environment-3.10.yml
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ dependencies:
- pytest==7.2.2
- pytest-cov==4.0.0
- pytest-flake8==1.1.1
- dask==2023.3.1
- dask==2024.4.1
- numpy==1.24.2
- scipy==1.10.1
- scikit-image==0.19.3
Expand Down
2 changes: 1 addition & 1 deletion continuous_integration/environment-3.11.yml
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ dependencies:
- pytest==7.2.2
- pytest-cov==4.0.0
- pytest-flake8==1.1.1
- dask==2023.3.1
- dask==2024.4.1
- numpy==1.24.2
- scipy==1.10.1
- scikit-image==0.19.3
Expand Down
28 changes: 14 additions & 14 deletions continuous_integration/environment-3.9.yml
Original file line number Diff line number Diff line change
Expand Up @@ -5,17 +5,17 @@ channels:

dependencies:
- python=3.9.*
- pip==22.0.2
- wheel==0.37.1
- coverage==6.3
- flake8==4.0.1
- pytest==6.2.5
- pytest-cov==4.0.0
- pytest-flake8==1.0.7
- dask==2023.2.1
- numpy==1.22.1
- scipy==1.7.3
- scikit-image==0.19.1
- pims==0.5
- slicerator==1.0.0
- pandas==2.0.0
- pip==24.0
- wheel==0.43.0
- coverage==7.5.1
- flake8==7.0.0
- pytest==8.2.0
- pytest-cov==5.0.0
- pytest-flake8==1.1.1
- dask==2024.4.1
- numpy==1.26.4
- scipy==1.13.0
- scikit-image==0.22.0
- pims==0.6.1
- slicerator==1.1.0
- pandas==2.2.2
2 changes: 1 addition & 1 deletion continuous_integration/environment-doc.yml
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ dependencies:
- pip==22.3
- wheel==0.37.1
- jinja2<3.1
- dask==2023.2.0
- dask==2024.4.1
- numpy==1.23.4
- scipy==1.9.2
- scikit-image==0.19.3
Expand Down
6 changes: 3 additions & 3 deletions dask_image/ndmorph/_utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ def _get_structure(image, structure):
raise RuntimeError(
"`structure` must have the same rank as `image`."
)
if not issubclass(structure.dtype.type, np.bool8):
if not issubclass(structure.dtype.type, np.bool_):
structure = (structure != 0)
else:
raise TypeError("`structure` must be an array.")
Expand Down Expand Up @@ -59,9 +59,9 @@ def _get_mask(image, mask):
if isinstance(mask, (np.ndarray, da.Array)):
if mask.shape != image.shape:
raise RuntimeError("`mask` must have the same shape as `image`.")
if not issubclass(mask_type, np.bool8):
if not issubclass(mask_type, np.bool_):
mask = (mask != 0)
elif issubclass(mask_type, np.bool8):
elif issubclass(mask_type, np.bool_):
mask = bool(mask)
else:
raise TypeError("`mask` must be a Boolean or an array.")
Expand Down
4 changes: 2 additions & 2 deletions setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -29,8 +29,8 @@ def run_tests(self):
history = history_file.read()

requirements = [
"dask[array] >=2023.2.0",
"dask[dataframe] >=2023.2.0",
"dask[array] >=2024.4.1",
"dask[dataframe] >=2024.4.1",
"numpy >=1.18",
"scipy >=1.7.0",
"pims >=0.4.1",
Expand Down
31 changes: 14 additions & 17 deletions tests/test_dask_image/test_imread/test_core.py
Original file line number Diff line number Diff line change
Expand Up @@ -33,17 +33,17 @@ def test_errs_imread(err_type, nframes):
]
)
@pytest.mark.parametrize(
"nframes, shape, runtime_warning",
"nframes, shape",
[
(1, (1, 4, 3), None),
(-1, (1, 4, 3), None),
(3, (1, 4, 3), "`nframes` larger than"),
(1, (5, 4, 3), None),
(2, (5, 4, 3), "`nframes` does not nicely divide"),
(1, (10, 5, 4, 3), None),
(5, (10, 5, 4, 3), None),
(10, (10, 5, 4, 3), None),
(-1, (10, 5, 4, 3), None),
(1, (1, 4, 3)),
(-1, (1, 4, 3)),
(3, (1, 4, 3)),
(1, (5, 4, 3)),
(2, (5, 4, 3)),
(1, (10, 5, 4, 3)),
(5, (10, 5, 4, 3)),
(10, (10, 5, 4, 3)),
(-1, (10, 5, 4, 3)),
]
)
@pytest.mark.parametrize(
Expand All @@ -61,7 +61,7 @@ def test_errs_imread(err_type, nframes):
False,
]
)
def test_tiff_imread(tmpdir, seed, nframes, shape, runtime_warning, dtype, is_pathlib_Path): # noqa: E501
def test_tiff_imread(tmpdir, seed, nframes, shape, dtype, is_pathlib_Path): # noqa: E501
np.random.seed(seed)

dirpth = tmpdir.mkdir("test_imread")
Expand All @@ -77,12 +77,9 @@ def test_tiff_imread(tmpdir, seed, nframes, shape, runtime_warning, dtype, is_pa
with tifffile.TiffWriter(fn) as fh:
for i in range(len(a)):
fh.save(a[i], contiguous=True)

with pytest.warns(None if runtime_warning is None
else RuntimeWarning, match=runtime_warning):
if is_pathlib_Path:
fn = pathlib.Path(fn)
d = dask_image.imread.imread(fn, nframes=nframes)
if is_pathlib_Path:
fn = pathlib.Path(fn)
d = dask_image.imread.imread(fn, nframes=nframes)

if nframes == -1:
nframes = shape[0]
Expand Down
203 changes: 107 additions & 96 deletions tests/test_dask_image/test_ndfilters/test__threshold.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,62 @@

from dask_image.ndfilters import threshold_local

@pytest.fixture
def simple_test_image():
image = da.from_array(np.array(
[[0, 0, 1, 3, 5],
[0, 1, 4, 3, 4],
[1, 2, 5, 4, 1],
[2, 4, 5, 2, 1],
[4, 5, 1, 0, 0]], dtype=int), chunks=(5, 5))
return image


# ==================================================
# Test Threshold Filters
# ==================================================

@pytest.mark.parametrize('block_size', [
3,
[3, 3],
np.array([3, 3]),
da.from_array(np.array([3, 3]), chunks=1),
da.from_array(np.array([3, 3]), chunks=2),
])
def test_threshold_local_gaussian(simple_test_image, block_size):
ref = np.array(
[[False, False, False, False, True],
[False, False, True, False, True],
[False, False, True, True, False],
[False, True, True, False, False],
[True, True, False, False, False]]
)
out = threshold_local(simple_test_image, block_size, method='gaussian')
assert_equal(ref, (simple_test_image > out).compute())

out = threshold_local(simple_test_image, block_size, method='gaussian',
param=1./3.)
assert_equal(ref, (simple_test_image > out).compute())


@pytest.mark.parametrize('block_size', [
3,
[3, 3],
np.array([3, 3]),
da.from_array(np.array([3, 3]), chunks=1),
da.from_array(np.array([3, 3]), chunks=2),
])
def test_threshold_local_mean(simple_test_image, block_size):
ref = np.array(
[[False, False, False, False, True],
[False, False, True, False, True],
[False, False, True, True, False],
[False, True, True, False, False],
[True, True, False, False, False]]
)
out = threshold_local(simple_test_image, block_size, method='mean')
assert_equal(ref, (simple_test_image > out).compute())


@pytest.mark.parametrize('block_size', [
3,
Expand All @@ -13,99 +69,54 @@
da.from_array(np.array([3, 3]), chunks=1),
da.from_array(np.array([3, 3]), chunks=2),
])
class TestSimpleImage:
def setup(self):
self.image = da.from_array(np.array(
[[0, 0, 1, 3, 5],
[0, 1, 4, 3, 4],
[1, 2, 5, 4, 1],
[2, 4, 5, 2, 1],
[4, 5, 1, 0, 0]], dtype=int), chunks=(5, 5))

def test_threshold_local_gaussian(self, block_size):
ref = np.array(
[[False, False, False, False, True],
[False, False, True, False, True],
[False, False, True, True, False],
[False, True, True, False, False],
[True, True, False, False, False]]
)
out = threshold_local(self.image, block_size, method='gaussian')
assert_equal(ref, (self.image > out).compute())

out = threshold_local(self.image, block_size, method='gaussian',
param=1./3.)
assert_equal(ref, (self.image > out).compute())

def test_threshold_local_mean(self, block_size):
ref = np.array(
[[False, False, False, False, True],
[False, False, True, False, True],
[False, False, True, True, False],
[False, True, True, False, False],
[True, True, False, False, False]]
)
out = threshold_local(self.image, block_size, method='mean')
assert_equal(ref, (self.image > out).compute())

def test_threshold_local_median(self, block_size):
ref = np.array(
[[False, False, False, False, True],
[False, False, True, False, False],
[False, False, True, False, False],
[False, False, True, True, False],
[False, True, False, False, False]]
)
out = threshold_local(self.image, block_size, method='median')
assert_equal(ref, (self.image > out).compute())


class TestGenericFilter:
def setup(self):
self.image = da.from_array(np.array(
[[0, 0, 1, 3, 5],
[0, 1, 4, 3, 4],
[1, 2, 5, 4, 1],
[2, 4, 5, 2, 1],
[4, 5, 1, 0, 0]], dtype=int), chunks=(5, 5))

def test_threshold_local_generic(self):
ref = np.array(
[[1., 7., 16., 29., 37.],
[5., 14., 23., 30., 30.],
[13., 24., 30., 29., 21.],
[25., 29., 28., 19., 10.],
[34., 31., 23., 10., 4.]]
)
unchanged = threshold_local(self.image, 1, method='generic', param=sum)
out = threshold_local(self.image, 3, method='generic', param=sum)
assert np.allclose(unchanged.compute(), self.image.compute())
assert np.allclose(out.compute(), ref)

def test_threshold_local_generic_invalid(self):
expected_error_message = "Must include a valid function to use as "
"the 'param' keyword argument."
with pytest.raises(ValueError) as e:
threshold_local(self.image, 3, method='generic', param='sum')
assert e == expected_error_message


class TestInvalidArguments:
def setup(self):
self.image = da.from_array(np.array(
[[0, 0, 1, 3, 5],
[0, 1, 4, 3, 4],
[1, 2, 5, 4, 1],
[2, 4, 5, 2, 1],
[4, 5, 1, 0, 0]], dtype=int), chunks=(5, 5))

@pytest.mark.parametrize("method, block_size, error_type", [
('median', np.nan, TypeError),
])
def test_nan_blocksize(self, method, block_size, error_type):
with pytest.raises(error_type):
threshold_local(self.image, block_size, method=method)

def test_invalid_threshold_method(self):
with pytest.raises(ValueError):
threshold_local(self.image, 3, method='invalid')
def test_threshold_local_median(simple_test_image, block_size):
ref = np.array(
[[False, False, False, False, True],
[False, False, True, False, False],
[False, False, True, False, False],
[False, False, True, True, False],
[False, True, False, False, False]]
)
out = threshold_local(simple_test_image, block_size, method='median')
assert_equal(ref, (simple_test_image > out).compute())


# ==================================================
# Test Generic Filters
# ==================================================

def test_threshold_local_generic(simple_test_image):
ref = np.array(
[[1., 7., 16., 29., 37.],
[5., 14., 23., 30., 30.],
[13., 24., 30., 29., 21.],
[25., 29., 28., 19., 10.],
[34., 31., 23., 10., 4.]]
)
unchanged = threshold_local(simple_test_image, 1, method='generic', param=sum)
out = threshold_local(simple_test_image, 3, method='generic', param=sum)
assert np.allclose(unchanged.compute(), simple_test_image.compute())
assert np.allclose(out.compute(), ref)

def test_threshold_local_generic_invalid(simple_test_image):
expected_error_message = "Must include a valid function to use as "
"the 'param' keyword argument."
with pytest.raises(ValueError) as e:
threshold_local(simple_test_image, 3, method='generic', param='sum')
assert e == expected_error_message


# ==================================================
# Test Invalid Arguments
# ==================================================

@pytest.mark.parametrize("method, block_size, error_type", [
('median', np.nan, TypeError),
])
def test_nan_blocksize(simple_test_image, method, block_size, error_type):
with pytest.raises(error_type):
threshold_local(simple_test_image, block_size, method=method)

def test_invalid_threshold_method(simple_test_image):
with pytest.raises(ValueError):
threshold_local(simple_test_image, 3, method='invalid')
2 changes: 1 addition & 1 deletion tests/test_dask_image/test_ndinterp/test_spline_filter.py
Original file line number Diff line number Diff line change
Expand Up @@ -172,7 +172,7 @@ def test_spline_filter1d_general(
axis,
):
if axis == 1 and n < 2:
pytest.skip(msg="skip axis=1 for 1d signals")
pytest.skip("skip axis=1 for 1d signals")

validate_spline_filter(
n=n,
Expand Down
4 changes: 1 addition & 3 deletions tests/test_dask_image/test_ndmeasure/test__utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -55,8 +55,6 @@ def test__norm_input_labels_index():

@pytest.mark.parametrize(
"shape, chunks, ind", [
((15, 16), (4, 5), 1),
((15, 16), (4, 5), [1]),
((15, 16), (4, 5), [[1, 2, 3, 4]]),
((15, 16), (4, 5), [[1, 2], [3, 4]]),
((15, 16), (4, 5), [[[1], [2], [3], [4]]]),
Expand All @@ -78,7 +76,7 @@ def test__norm_input_labels_index_warn(shape, chunks, ind):
ind = np.array(ind)
d_ind = da.from_array(ind, chunks=1)

with pytest.warns(None) as w:
with pytest.warns(FutureWarning) as w:
dask_image.ndmeasure._utils._norm_input_labels_index(
d, d_lbls, d_ind
)
Expand Down
Loading

0 comments on commit a4fd9a7

Please sign in to comment.