Skip to content

Commit bb5c5ae

Browse files
authoredMar 4, 2025··
feat: aggressive fill holes (#9)
* fix: closing and opening are idempotent, no need to provide iterations parameters * feat(fill_holes): add fix_borders and morphological_close params * fix: rename to morphological_closing and add pixel change counts * fix: label may not be present in tiny cutouts after erosion * test: check fix borders and morphological closing work * ci: add crackle-codec to test run * docs: show how to use new options
1 parent 4656b13 commit bb5c5ae

File tree

6 files changed

+98
-37
lines changed

6 files changed

+98
-37
lines changed
 

‎.github/workflows/run_tests.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ jobs:
2828
- name: Install dependencies
2929
run: |
3030
python -m pip install --upgrade pip
31-
python -m pip install pytest pybind11 setuptools wheel
31+
python -m pip install pytest pybind11 setuptools wheel crackle-codec
3232
3333
- name: Compile
3434
run: python setup.py develop

‎README.md

+12-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@ We provide the following multithreaded (except where noted) operations:
1010
- Grayscale Stenciled Dilation, Erosion, Opening, Closing
1111
- Multi-Label Spherical Erosion
1212
- Binary Spherical Dilation, Opening, and Closing
13-
- Multi-Label Fill Voids (single threaded)
13+
- Multi-Label Fill Voids (mostly single threaded)
1414

1515
Highlights compared to other libraries:
1616

@@ -71,6 +71,17 @@ morphed = fastmorph.spherical_erode(labels, radius=1, parallel=2, anisotropy=(1,
7171
# Note that for multilabel images, by default, if a label is totally enclosed by another,
7272
# a FillError will be raised. If remove_enclosed is True, the label will be overwritten.
7373
filled_labels, ct = fastmorph.fill_holes(labels, return_fill_count=True, remove_enclosed=False)
74+
75+
# If the holes in your segmentation are imperfectly sealed, consider
76+
# using the following options.
77+
filled_labels = fastmorph.fill_holes(
78+
labels,
79+
# runs 2d fill on the sides of the cube for each binary image
80+
fix_borders=True,
81+
# does a dilate and then an erode after filling holes
82+
morphological_closing=True,
83+
)
84+
7485
```
7586

7687
## Performance

‎automated_test.py

+40
Original file line numberDiff line numberDiff line change
@@ -69,6 +69,46 @@ def test_fill_holes():
6969
assert res[1] == 1
7070
assert res[2] == set()
7171

72+
def test_fill_fix_borders():
73+
labels = np.ones([100,100,100], dtype=np.uint8)
74+
labels[40:60,40:60,:] = 0
75+
labels[40:60,:,40:60] = 0
76+
labels[:,40:60,40:60] = 0
77+
78+
res, cts = fastmorph.fill_holes(
79+
labels,
80+
fix_borders=False,
81+
return_fill_count=True,
82+
morphological_closing=False,
83+
)
84+
85+
assert cts[1] == 0
86+
assert np.all(res == labels)
87+
88+
res, cts = fastmorph.fill_holes(
89+
labels,
90+
fix_borders=True,
91+
return_fill_count=True,
92+
morphological_closing=False,
93+
)
94+
95+
assert cts[1] == np.count_nonzero(labels == 0)
96+
assert np.all(res == 1)
97+
98+
def test_complex_fill():
99+
import crackle
100+
labels = crackle.load("test_data/complex_soma.ckl.gz")
101+
ans = crackle.load("test_data/complex_soma_result.ckl.gz")
102+
103+
res = fastmorph.fill_holes(
104+
labels,
105+
remove_enclosed=True,
106+
fix_borders=True,
107+
morphological_closing=True,
108+
)
109+
110+
assert np.all(res == ans)
111+
72112
def test_spherical_open_close_run():
73113
labels = np.zeros((10,10,10), dtype=bool)
74114
res = fastmorph.spherical_open(labels, radius=1)

‎fastmorph/__init__.py

+45-35
Original file line numberDiff line numberDiff line change
@@ -121,74 +121,56 @@ def opening(
121121
background_only:bool = True,
122122
parallel:int = 0,
123123
mode:Mode = Mode.multilabel,
124-
iterations:int = 1,
125124
erode_border:bool = True,
126125
) -> np.ndarray:
127126
"""Performs morphological opening of labels.
128127
128+
This operator is idempotent with the exception
129+
of boundary effects.
130+
129131
background_only is passed through to dilate.
130132
True: Only evaluate background voxels for dilation.
131133
False: Allow labels to erode each other as they grow.
132134
parallel: how many pthreads to use in a threadpool
133135
mode:
134136
Mode.multilabel: are all surrounding pixels the same?
135137
Mode.grey: use grayscale image dilation (min value)
136-
137-
iterations: number of times to iterate the result
138-
139138
erode_border: if True, the border is treated as background,
140139
else it is regarded as a value that would preserve the
141140
current value. Only has an effect for multilabel erosion.
142141
"""
143-
if iterations < 0:
144-
raise ValueError(f"iterations ({iterations}) must be a positive integer.")
145-
elif iterations == 0:
146-
return np.copy(labels, order="F")
147-
148-
output = labels
149-
for i in range(iterations):
150-
output = dilate(
151-
erode(output, parallel, mode, iterations=1, erode_border=erode_border),
152-
background_only, parallel, mode, iterations=1
153-
)
154-
return output
142+
return dilate(
143+
erode(labels, parallel, mode, iterations=1, erode_border=erode_border),
144+
background_only, parallel, mode, iterations=1
145+
)
155146

156147
def closing(
157148
labels:np.ndarray,
158149
background_only:bool = True,
159150
parallel:int = 0,
160151
mode:Mode = Mode.multilabel,
161-
iterations:int = 1,
162152
erode_border:bool = True,
163153
) -> np.ndarray:
164154
"""Performs morphological closing of labels.
165155
156+
This operator is idempotent with the exception
157+
of boundary effects.
158+
166159
background_only is passed through to dilate.
167160
True: Only evaluate background voxels for dilation.
168161
False: Allow labels to erode each other as they grow.
169162
parallel: how many pthreads to use in a threadpool
170163
mode:
171164
Mode.multilabel: are all surrounding pixels the same?
172165
Mode.grey: use grayscale image dilation (min value)
173-
174-
iterations: number of times to iterate the result
175-
176166
erode_border: if True, the border is treated as background,
177167
else it is regarded as a value that would preserve the
178168
current value. Only has an effect for multilabel erosion.
179169
"""
180-
if iterations < 0:
181-
raise ValueError(f"iterations ({iterations}) must be a positive integer.")
182-
elif iterations == 0:
183-
return np.copy(labels, order="F")
184-
185-
output = labels
186-
for i in range(iterations):
187-
output = erode(
188-
dilate(output, background_only, parallel, mode, iterations=1),
189-
parallel, mode, iterations=1, erode_border=erode_border,
190-
)
191-
return output
170+
return erode(
171+
dilate(labels, background_only, parallel, mode, iterations=1),
172+
parallel, mode, iterations=1, erode_border=erode_border,
173+
)
192174

193175
def spherical_dilate(
194176
labels:np.ndarray,
@@ -283,6 +265,8 @@ def fill_holes(
283265
return_fill_count:bool = False,
284266
remove_enclosed:bool = False,
285267
return_removed:bool = False,
268+
fix_borders:bool = False,
269+
morphological_closing:bool = False,
286270
) -> np.ndarray:
287271
"""
288272
For fill holes in toplogically closed objects.
@@ -298,7 +282,7 @@ def fill_holes(
298282
Return value: (filled_labels, fill_count (if specified), removed_set (if specified))
299283
"""
300284
assert np.issubdtype(labels.dtype, np.integer) or np.issubdtype(labels.dtype, bool), "fill_holes is currently only supported for integer or binary images."
301-
if np.issubdtype(labels.dtype, bool):
285+
if np.issubdtype(labels.dtype, bool) and not fix_borders and not morphological_closing:
302286
filled_labels, filled_ct = fill_voids.fill(labels, return_fill_count=True)
303287
ret = [ filled_labels ]
304288
if return_fill_count:
@@ -327,10 +311,36 @@ def fill_holes(
327311
continue
328312

329313
binary_image = (cc_labels[slices] == label)
330-
binary_image, pixels_filled = fill_voids.fill(
314+
315+
pixels_filled = 0
316+
317+
if morphological_closing:
318+
dilated_binary_image = dilate(binary_image)
319+
pixels_filled += np.sum(dilated_binary_image != binary_image)
320+
binary_image = dilated_binary_image
321+
del dilated_binary_image
322+
323+
if fix_borders:
324+
binary_image[:,:,0], pf1 = fill_voids.fill(binary_image[:,:,0], return_fill_count=True)
325+
binary_image[:,:,-1], pf2 = fill_voids.fill(binary_image[:,:,-1], return_fill_count=True)
326+
binary_image[:,0,:], pf3 = fill_voids.fill(binary_image[:,0,:], return_fill_count=True)
327+
binary_image[:,-1,:], pf4 = fill_voids.fill(binary_image[:,-1,:], return_fill_count=True)
328+
binary_image[0,:,:], pf5 = fill_voids.fill(binary_image[0,:,:], return_fill_count=True)
329+
binary_image[-1,:,:], pf6 = fill_voids.fill(binary_image[-1,:,:], return_fill_count=True)
330+
pixels_filled += pf1 + pf2 + pf3 + pf4 + pf5 + pf6
331+
332+
binary_image, pf7 = fill_voids.fill(
331333
binary_image, in_place=True,
332334
return_fill_count=True
333335
)
336+
pixels_filled += pf7
337+
338+
if morphological_closing:
339+
eroded_binary_image = erode(binary_image, erode_border=False)
340+
pixels_filled -= np.sum(eroded_binary_image != binary_image)
341+
binary_image = eroded_binary_image
342+
del eroded_binary_image
343+
334344
fill_counts[label] = pixels_filled
335345
output[slices][binary_image] = mapping[label]
336346

@@ -339,7 +349,7 @@ def fill_holes(
339349

340350
sub_labels = fastremap.unique(cc_labels[slices][binary_image])
341351
sub_labels = set(sub_labels)
342-
sub_labels.remove(label)
352+
sub_labels.discard(label)
343353
sub_labels.discard(0)
344354
if not remove_enclosed and sub_labels:
345355
sub_labels = [ int(l) for l in sub_labels ]

‎test_data/complex_soma.ckl.gz

568 KB
Binary file not shown.

‎test_data/complex_soma_result.ckl.gz

85.7 KB
Binary file not shown.

0 commit comments

Comments
 (0)
Please sign in to comment.