diff --git a/.circleci/config.yml b/.circleci/config.yml index 50ade45..4521be9 100644 --- a/.circleci/config.yml +++ b/.circleci/config.yml @@ -1,23 +1,29 @@ version: 2.0 jobs: - Python_2.7: - docker: - - image: circleci/python:2.7 - steps: &steps - - checkout - - run: sudo pip install -r requirements.txt - - run: sudo pip install pytest - - run: python --version ; pip --version ; pwd ; ls - - run: pytest - Python_3.6: - docker: - - image: circleci/python:3.6 - steps: *steps + Python_3.9: + docker: + - image: circleci/python:3.9 + steps: &steps + - checkout + - run: sudo pip install -r requirements.txt + - run: sudo pip install pytest ruff + - run: python --version ; pip --version ; pwd ; ls + - run: pytest + - run: ruff + Python_3.10: + docker: + - image: circleci/python:3.10 + steps: *steps + Python_3.11: + docker: + - image: circleci/python:3.11 + steps: *steps workflows: version: 2 build: jobs: - - Python_2.7 - - Python_3.6 \ No newline at end of file + - Python_3.9 + - Python_3.10 + - Python_3.11 diff --git a/README.md b/README.md index 86d0d73..8e27cc4 100644 --- a/README.md +++ b/README.md @@ -2,6 +2,7 @@ [![coverage](https://github.com/nipy/nilabels/blob/master/coverage.svg)](https://github.com/SebastianoF/nilabels/blob/master/coverage.svg) [![CircleCI](https://circleci.com/gh/nipy/nilabels.svg?style=svg)](https://circleci.com/gh/nipy/nilabels) [![PyPI version](https://badge.fury.io/py/nilabels.svg)](https://badge.fury.io/py/nilabels) +[![Ruff](https://img.shields.io/endpoint?url=https://raw.githubusercontent.com/astral-sh/ruff/main/assets/badge/v2.json)](https://github.com/astral-sh/ruff)
@@ -13,7 +14,7 @@
NiLabels is a cacophony of tools to automate simple manipulations and measurements of medical image
segmentations in nifti format. It is strongly based on and influenced by the library [NiBabel](http://nipy.org/nibabel/)
-+ Written in [Python 3.6](https://docs.python-guide.org/) back compatible with 2.7
++ Written in [Python 3](https://docs.python-guide.org/)
+ [Motivations](https://github.com/SebastianoF/nilabels/wiki/Motivations)
+ [Features](https://github.com/SebastianoF/nilabels/wiki/What-you-can-do-with-nilabels)
+ [Design pattern](https://github.com/SebastianoF/nilabels/wiki/Design-Pattern)
@@ -94,6 +95,17 @@ More tools are introduced in the [documentation](https://github.com/SebastianoF/
+ [How to run the tests](https://github.com/SebastianoF/nilabels/wiki/Testing)
+### Development
+
+`nilabel` is a python package managed with [poetry](https://python-poetry.org/) and linted with [ruff](https://docs.astral.sh/ruff/), tested with [pytest](https://docs.pytest.org/en/8.0.x/)
+
+### TODO
+
+Other than the many TODOs around the code, there are two more things:
+
+- typechecking with mypy
+- migrate from cicleCI to github workflows
+
### Licencing and Copyright
Copyright (c) 2017, Sebastiano Ferraris. NiLabels (ex. [LABelsToolkit](https://github.com/SebastianoF/LABelsToolkit))
diff --git a/examples/a_generate_phantoms_for_examples.py b/examples/a_generate_phantoms_for_examples.py
index 8a92bc1..fb94e1a 100644
--- a/examples/a_generate_phantoms_for_examples.py
+++ b/examples/a_generate_phantoms_for_examples.py
@@ -8,43 +8,54 @@
from nilabels.definitions import root_dir
from nilabels.tools.detections.get_segmentation import intensity_segmentation
-
# ---------- Simple shapes generators ---------------
-def o_shape(omega=(250, 250), radius=50,
- background_intensity=0, foreground_intensity=20, dtype=np.uint8):
+def o_shape(omega=(250, 250), radius=50, background_intensity=0, foreground_intensity=20, dtype=np.uint8):
m = background_intensity * np.ones(omega, dtype=dtype)
if len(omega) == 2:
c = [omega[j] / 2 for j in range(len(omega))]
for x in range(omega[0]):
for y in range(omega[1]):
- if (x - c[0]) ** 2 + (y - c[1]) ** 2 < radius ** 2:
+ if (x - c[0]) ** 2 + (y - c[1]) ** 2 < radius**2:
m[x, y] = foreground_intensity
elif len(omega) == 3:
c = [omega[j] / 2 for j in range(len(omega))]
for x in range(omega[0]):
for y in range(omega[1]):
for z in range(omega[2]):
- if (x - c[0]) ** 2 + (y - c[1]) ** 2 + (z - c[2]) ** 2 < radius ** 2:
+ if (x - c[0]) ** 2 + (y - c[1]) ** 2 + (z - c[2]) ** 2 < radius**2:
m[x, y, z] = foreground_intensity
return m
-def c_shape(omega=(250, 250), internal_radius=40, external_radius=60, opening_height=50,
- background_intensity=0, foreground_intensity=20, dtype=np.uint8, margin=None):
-
- def get_a_2d_c(omega2, internal_radius2d, external_radius2d, opening_height2d, background_intensity2d,
- foreground_intensity2d, dtype2d):
-
+def c_shape(
+ omega=(250, 250),
+ internal_radius=40,
+ external_radius=60,
+ opening_height=50,
+ background_intensity=0,
+ foreground_intensity=20,
+ dtype=np.uint8,
+ margin=None,
+):
+ def get_a_2d_c(
+ omega2,
+ internal_radius2d,
+ external_radius2d,
+ opening_height2d,
+ background_intensity2d,
+ foreground_intensity2d,
+ dtype2d,
+ ):
m = background_intensity2d * np.ones(omega2[:2], dtype=dtype2d)
c = [omega2[j] / 2 for j in range(len(omega2))]
# create the crown
for x in range(omega2[0]):
for y in range(omega2[1]):
- if internal_radius2d ** 2 < (x - c[0]) ** 2 + (y - c[1]) ** 2 < external_radius2d ** 2:
+ if internal_radius2d**2 < (x - c[0]) ** 2 + (y - c[1]) ** 2 < external_radius2d**2:
m[x, y] = foreground_intensity2d
# open the c
@@ -58,31 +69,42 @@ def get_a_2d_c(omega2, internal_radius2d, external_radius2d, opening_height2d, b
return m
- c_2d = get_a_2d_c(omega2=omega[:2], internal_radius2d=internal_radius, external_radius2d=external_radius,
- opening_height2d=opening_height, background_intensity2d=background_intensity,
- foreground_intensity2d=foreground_intensity, dtype2d=dtype)
+ c_2d = get_a_2d_c(
+ omega2=omega[:2],
+ internal_radius2d=internal_radius,
+ external_radius2d=external_radius,
+ opening_height2d=opening_height,
+ background_intensity2d=background_intensity,
+ foreground_intensity2d=foreground_intensity,
+ dtype2d=dtype,
+ )
if len(omega) == 2:
return c_2d
- elif len(omega) == 3:
+ if len(omega) == 3:
if margin is None:
return np.repeat(c_2d, omega[2]).reshape(omega)
- else:
- res = np.zeros(omega, dtype=dtype)
- for z in range(margin, omega[2] - 2 * margin):
- res[..., z] = c_2d
- return res
+
+ res = np.zeros(omega, dtype=dtype)
+ for z in range(margin, omega[2] - 2 * margin):
+ res[..., z] = c_2d
+ return res
+ raise ValueError("Size of omega can be 2 or 3")
-def ellipsoid_shape(omega, focus_1, focus_2, distance, background_intensity=0, foreground_intensity=100,
- dtype=np.uint8):
+def ellipsoid_shape(
+ omega, focus_1, focus_2, distance, background_intensity=0, foreground_intensity=100, dtype=np.uint8,
+):
sky = background_intensity * np.ones(omega, dtype=dtype)
for xi in range(omega[0]):
for yi in range(omega[1]):
for zi in range(omega[2]):
- if np.sqrt((focus_1[0] - xi) ** 2 + (focus_1[1] - yi) ** 2 + (focus_1[2] - zi) ** 2) + \
- np.sqrt((focus_2[0] - xi) ** 2 + (focus_2[1] - yi) ** 2 + (focus_2[2] - zi) ** 2) <= distance:
+ if (
+ np.sqrt((focus_1[0] - xi) ** 2 + (focus_1[1] - yi) ** 2 + (focus_1[2] - zi) ** 2)
+ + np.sqrt((focus_2[0] - xi) ** 2 + (focus_2[1] - yi) ** 2 + (focus_2[2] - zi) ** 2)
+ <= distance
+ ):
sky[xi, yi, zi] = foreground_intensity
return sky
@@ -109,12 +131,12 @@ def sphere_shape(omega, centre, radius, foreground_intensity=100, dtype=np.uint8
def generate_figures(creation_list, pfo_examples, segmentation_levels=7, sigma_smoothing=6, foreground=10):
- print('\n.\n.\n\nGenerate figures for the examples, may take some seconds, and will take approx 150MB.\n.\n.')
+ print("\n.\n.\n\nGenerate figures for the examples, may take some seconds, and will take approx 150MB.\n.\n.")
- if creation_list['Examples folder']:
- os.system('mkdir -p ' + pfo_examples)
+ if creation_list["Examples folder"]:
+ os.system("mkdir -p " + pfo_examples)
- if creation_list['Punt e mes']:
+ if creation_list["Punt e mes"]:
data_o_punt = o_shape(omega=(256, 256, 256), foreground_intensity=foreground, dtype=np.float64)
data_o_punt = fil.gaussian_filter(data_o_punt, sigma=sigma_smoothing)
@@ -123,12 +145,12 @@ def generate_figures(creation_list, pfo_examples, segmentation_levels=7, sigma_s
nib_o_punt = nib.Nifti1Image(data_o_punt, affine=np.eye(4))
nib_o_punt_seg = nib.Nifti1Image(data_o_punt_segmentation, affine=np.eye(4))
- nib.save(nib_o_punt, filename=jph(pfo_examples, 'punt.nii.gz'))
- nib.save(nib_o_punt_seg, filename=jph(pfo_examples, 'punt_seg.nii.gz'))
+ nib.save(nib_o_punt, filename=jph(pfo_examples, "punt.nii.gz"))
+ nib.save(nib_o_punt_seg, filename=jph(pfo_examples, "punt_seg.nii.gz"))
del data_o_punt_segmentation, nib_o_punt, nib_o_punt_seg
- print('punt generated')
+ print("punt generated")
data_o_mes = np.zeros_like(data_o_punt)
data_o_mes[:, :128, :] = data_o_punt[:, :128, :] # R, A ,S
@@ -138,14 +160,14 @@ def generate_figures(creation_list, pfo_examples, segmentation_levels=7, sigma_s
nib_o_mes = nib.Nifti1Image(data_o_mes, affine=np.eye(4))
nib_o_mes_seg = nib.Nifti1Image(data_o_mes_segmentation, affine=np.eye(4))
- nib.save(nib_o_mes, filename=jph(pfo_examples, 'mes.nii.gz'))
- nib.save(nib_o_mes_seg, filename=jph(pfo_examples, 'mes_seg.nii.gz'))
+ nib.save(nib_o_mes, filename=jph(pfo_examples, "mes.nii.gz"))
+ nib.save(nib_o_mes_seg, filename=jph(pfo_examples, "mes_seg.nii.gz"))
del data_o_punt, data_o_mes, data_o_mes_segmentation, nib_o_mes, nib_o_mes_seg
- print('mes generated')
+ print("mes generated")
- if creation_list['C']:
+ if creation_list["C"]:
data_c_ = c_shape(omega=(256, 256, 256), foreground_intensity=foreground)
data_c_ = fil.gaussian_filter(data_c_, sigma_smoothing)
@@ -157,18 +179,25 @@ def generate_figures(creation_list, pfo_examples, segmentation_levels=7, sigma_s
nib_c = nib.Nifti1Image(data_c, affine=np.eye(4))
nib_c_seg = nib.Nifti1Image(data_c_segmentation, affine=np.eye(4))
- nib.save(nib_c, filename=jph(pfo_examples, 'acee.nii.gz'))
- nib.save(nib_c_seg, filename=jph(pfo_examples, 'acee_seg.nii.gz'))
+ nib.save(nib_c, filename=jph(pfo_examples, "acee.nii.gz"))
+ nib.save(nib_c_seg, filename=jph(pfo_examples, "acee_seg.nii.gz"))
del data_c_, data_c, data_c_segmentation, nib_c, nib_c_seg
- print('C generated')
-
- if creation_list['Planetaruim']:
+ print("C generated")
+ if creation_list["Planetaruim"]:
omega = (256, 256, 256)
- centers = [(32, 36, 40), (100, 61, 94), (130, 140, 99), (220, 110, 210),
- (224, 220, 216), (156, 195, 162), (126, 116, 157), (36, 146, 46)]
+ centers = [
+ (32, 36, 40),
+ (100, 61, 94),
+ (130, 140, 99),
+ (220, 110, 210),
+ (224, 220, 216),
+ (156, 195, 162),
+ (126, 116, 157),
+ (36, 146, 46),
+ ]
radii = [21, 55, 34, 13, 21, 55, 34, 13]
intensities = [100] * 8
@@ -178,17 +207,17 @@ def generate_figures(creation_list, pfo_examples, segmentation_levels=7, sigma_s
for x in range(center[0] - radius - 1, center[0] + radius + 1):
for y in range(center[1] - radius - 1, center[1] + radius + 1):
for z in range(center[2] - radius - 1, center[2] + radius + 1):
- if (center[0] - x) ** 2 + (center[1] - y) ** 2 + (center[2] - z) ** 2 <= radius ** 2:
+ if (center[0] - x) ** 2 + (center[1] - y) ** 2 + (center[2] - z) ** 2 <= radius**2:
sky[x, y, z] = intensity
planetarium = fil.gaussian_filter(sky, np.min(radii) - np.std(radii))
nib_planetarium = nib.Nifti1Image(planetarium, affine=np.eye(4))
- nib.save(nib_planetarium, filename=jph(pfo_examples, 'planetarium.nii.gz'))
+ nib.save(nib_planetarium, filename=jph(pfo_examples, "planetarium.nii.gz"))
- print('Planetarium generated')
+ print("Planetarium generated")
- if creation_list['Buckle ellipsoids']:
+ if creation_list["Buckle ellipsoids"]:
# Toy example for symmetrisation with registration test
omega = (120, 140, 160)
@@ -198,35 +227,37 @@ def generate_figures(creation_list, pfo_examples, segmentation_levels=7, sigma_s
d_left = 95
d_right = 95
- ellipsoid_left = ellipsoid_shape(omega, foci_ellipses_left[0], foci_ellipses_left[1], d_left,
- foreground_intensity=foreground)
- ellipsoid_right = ellipsoid_shape(omega, foci_ellipses_right[0], foci_ellipses_right[1], d_right,
- foreground_intensity=foreground)
+ ellipsoid_left = ellipsoid_shape(
+ omega, foci_ellipses_left[0], foci_ellipses_left[1], d_left, foreground_intensity=foreground,
+ )
+ ellipsoid_right = ellipsoid_shape(
+ omega, foci_ellipses_right[0], foci_ellipses_right[1], d_right, foreground_intensity=foreground,
+ )
- two_ellipsoids = foreground * (ellipsoid_left + ellipsoid_right).astype(np.bool).astype(np.float64)
+ two_ellipsoids = foreground * (ellipsoid_left + ellipsoid_right).astype(bool).astype(np.float64)
two_ellipsoids = fil.gaussian_filter(two_ellipsoids, sigma=sigma_smoothing)
nib_ellipsoids = nib.Nifti1Image(two_ellipsoids, affine=np.eye(4))
- nib.save(nib_ellipsoids, filename=jph(pfo_examples, 'ellipsoids.nii.gz'))
+ nib.save(nib_ellipsoids, filename=jph(pfo_examples, "ellipsoids.nii.gz"))
two_ellipsoids_segmentation = intensity_segmentation(two_ellipsoids, num_levels=segmentation_levels)
nib_ellipsoids_seg = nib.Nifti1Image(two_ellipsoids_segmentation, affine=np.eye(4))
- nib.save(nib_ellipsoids_seg, filename=jph(pfo_examples, 'ellipsoids_seg.nii.gz'))
+ nib.save(nib_ellipsoids_seg, filename=jph(pfo_examples, "ellipsoids_seg.nii.gz"))
two_ellipsoids_half_segmentation = np.zeros_like(two_ellipsoids_segmentation)
two_ellipsoids_half_segmentation[:60, :, :] = two_ellipsoids_segmentation[:60, :, :]
nib_ellipsoids_seg_half = nib.Nifti1Image(two_ellipsoids_half_segmentation, affine=np.eye(4))
- nib.save(nib_ellipsoids_seg_half, filename=jph(pfo_examples, 'ellipsoids_seg_half.nii.gz'))
+ nib.save(nib_ellipsoids_seg_half, filename=jph(pfo_examples, "ellipsoids_seg_half.nii.gz"))
- print('Buckle ellipsoids half segmented and whole segmented generated')
+ print("Buckle ellipsoids half segmented and whole segmented generated")
- if creation_list['Ellipsoids family']:
+ if creation_list["Ellipsoids family"]:
# Toy example for registration propagation tests
- pfo_ellipsoids_family = jph(pfo_examples, 'ellipsoids_family')
- os.system('mkdir -p ' + pfo_ellipsoids_family)
+ pfo_ellipsoids_family = jph(pfo_examples, "ellipsoids_family")
+ os.system("mkdir -p " + pfo_ellipsoids_family)
omega = (100, 100, 100)
num_ellipsoids = 10
@@ -237,10 +268,10 @@ def generate_figures(creation_list, pfo_examples, segmentation_levels=7, sigma_s
target_seg = intensity_segmentation(target_data, num_levels=segmentation_levels)
nib_target = nib.Nifti1Image(target_data, affine=np.eye(4))
- nib.save(nib_target, filename=jph(pfo_ellipsoids_family, 'target.nii.gz'))
+ nib.save(nib_target, filename=jph(pfo_ellipsoids_family, "target.nii.gz"))
nib_target_seg = nib.Nifti1Image(target_seg, affine=np.eye(4))
- nib.save(nib_target_seg, filename=jph(pfo_ellipsoids_family, 'target_seg.nii.gz'))
+ nib.save(nib_target_seg, filename=jph(pfo_ellipsoids_family, "target_seg.nii.gz"))
# --- Generate random ellipsoids in omega: --
for k in range(1, num_ellipsoids + 1):
@@ -253,71 +284,87 @@ def generate_figures(creation_list, pfo_examples, segmentation_levels=7, sigma_s
# sphere point picking (Wolfram)
theta = 2 * np.pi * u
phi = np.arccos(2 * v - 1)
- first_focus = np.array([radius_first_focus * np.cos(theta) * np.sin(phi) + center_first_focus[0],
- radius_first_focus * np.sin(theta) * np.sin(phi) + center_first_focus[1],
- radius_first_focus * np.cos(phi) + center_first_focus[2]])
+ first_focus = np.array(
+ [
+ radius_first_focus * np.cos(theta) * np.sin(phi) + center_first_focus[0],
+ radius_first_focus * np.sin(theta) * np.sin(phi) + center_first_focus[1],
+ radius_first_focus * np.cos(phi) + center_first_focus[2],
+ ],
+ )
# get the second focus as symmetric of the fist focus respect to the midpoint
- second_focus = np.array([2 * mid_point[0] - first_focus[0],
- 2 * mid_point[1] - first_focus[1],
- 2 * mid_point[2] - first_focus[2]])
+ second_focus = np.array(
+ [
+ 2 * mid_point[0] - first_focus[0],
+ 2 * mid_point[1] - first_focus[1],
+ 2 * mid_point[2] - first_focus[2],
+ ],
+ )
np.testing.assert_almost_equal(mid_point, (first_focus + second_focus) / float(2))
# get the twice_aance from the 2 focus as a random value, having as min the twice_aance between
twice_a_2_focus = np.linalg.norm(first_focus - second_focus)
epsilon = 3
- twice_a = np.random.uniform(low=twice_a_2_focus + epsilon,
- high=twice_a_2_focus + epsilon +
- np.linalg.norm(first_focus - np.array([25, 25, 50])))
+ twice_a = np.random.uniform(
+ low=twice_a_2_focus + epsilon,
+ high=twice_a_2_focus + epsilon + np.linalg.norm(first_focus - np.array([25, 25, 50])),
+ )
# get the ellipsoid
- ellips_data = ellipsoid_shape(omega, first_focus, second_focus, twice_a,
- foreground_intensity=foreground, dtype=np.float64)
+ ellips_data = ellipsoid_shape(
+ omega, first_focus, second_focus, twice_a, foreground_intensity=foreground, dtype=np.float64,
+ )
ellips_data = fil.gaussian_filter(ellips_data, sigma=sigma_smoothing)
nib_ellipsoids = nib.Nifti1Image(ellips_data, affine=np.eye(4))
- nib.save(nib_ellipsoids, filename=jph(pfo_ellipsoids_family, 'ellipsoid' + str(k) + '.nii.gz'))
+ nib.save(nib_ellipsoids, filename=jph(pfo_ellipsoids_family, "ellipsoid" + str(k) + ".nii.gz"))
ellips_data_seg = intensity_segmentation(ellips_data, num_levels=segmentation_levels)
nib_ellipsoids_seg = nib.Nifti1Image(ellips_data_seg, affine=np.eye(4))
- nib.save(nib_ellipsoids_seg, filename=jph(pfo_ellipsoids_family, 'ellipsoid' + str(k) + '_seg.nii.gz'))
+ nib.save(nib_ellipsoids_seg, filename=jph(pfo_ellipsoids_family, "ellipsoid" + str(k) + "_seg.nii.gz"))
- print('Data ellipsoids generated')
+ print("Data ellipsoids generated")
# open target and ellipsoids
- str_pfi_ellipsoids = ''
+ str_pfi_ellipsoids = ""
for k in range(1, num_ellipsoids + 1):
- str_pfi_ellipsoids += jph(pfo_ellipsoids_family, 'ellipsoid' + str(k) + '.nii.gz') + ' '
+ str_pfi_ellipsoids += jph(pfo_ellipsoids_family, "ellipsoid" + str(k) + ".nii.gz") + " "
- cmd = 'itksnap -g {0} -s {1} -o {2}'.format(jph(pfo_ellipsoids_family, 'target.nii.gz'),
- jph(pfo_ellipsoids_family, 'target_seg.nii.gz'),
- str_pfi_ellipsoids)
+ cmd = "itksnap -g {0} -s {1} -o {2}".format(
+ jph(pfo_ellipsoids_family, "target.nii.gz"),
+ jph(pfo_ellipsoids_family, "target_seg.nii.gz"),
+ str_pfi_ellipsoids,
+ )
os.system(cmd)
- if creation_list['Cubes in the sky']:
+ if creation_list["Cubes in the sky"]:
omega = [80, 80, 80]
cube_a = [[10, 60, 55], 11, 1]
cube_b = [[50, 55, 42], 17, 2]
cube_c = [[25, 20, 20], 19, 3]
cube_d = [[55, 16, 9], 9, 4]
- sky1 = cube_shape(omega, center=cube_a[0], side_length=cube_a[1], foreground_intensity=1) + \
- cube_shape(omega, center=cube_b[0], side_length=cube_b[1], foreground_intensity=1) + \
- cube_shape(omega, center=cube_c[0], side_length=cube_c[1], foreground_intensity=1) + \
- cube_shape(omega, center=cube_d[0], side_length=cube_d[1], foreground_intensity=1)
+ sky1 = (
+ cube_shape(omega, center=cube_a[0], side_length=cube_a[1], foreground_intensity=1)
+ + cube_shape(omega, center=cube_b[0], side_length=cube_b[1], foreground_intensity=1)
+ + cube_shape(omega, center=cube_c[0], side_length=cube_c[1], foreground_intensity=1)
+ + cube_shape(omega, center=cube_d[0], side_length=cube_d[1], foreground_intensity=1)
+ )
im1 = nib.Nifti1Image(sky1, affine=np.eye(4))
- sky2 = cube_shape(omega, center=cube_a[0], side_length=cube_a[1], foreground_intensity=cube_a[2]) + \
- cube_shape(omega, center=cube_b[0], side_length=cube_b[1], foreground_intensity=cube_b[2]) + \
- cube_shape(omega, center=cube_c[0], side_length=cube_c[1], foreground_intensity=cube_c[2]) + \
- cube_shape(omega, center=cube_d[0], side_length=cube_d[1], foreground_intensity=cube_d[2])
+ sky2 = (
+ cube_shape(omega, center=cube_a[0], side_length=cube_a[1], foreground_intensity=cube_a[2])
+ + cube_shape(omega, center=cube_b[0], side_length=cube_b[1], foreground_intensity=cube_b[2])
+ + cube_shape(omega, center=cube_c[0], side_length=cube_c[1], foreground_intensity=cube_c[2])
+ + cube_shape(omega, center=cube_d[0], side_length=cube_d[1], foreground_intensity=cube_d[2])
+ )
im2 = nib.Nifti1Image(sky2, affine=np.eye(4))
- nib.save(im1, filename=jph(pfo_examples, 'cubes_in_space_bin.nii.gz'))
- nib.save(im2, filename=jph(pfo_examples, 'cubes_in_space.nii.gz'))
+ nib.save(im1, filename=jph(pfo_examples, "cubes_in_space_bin.nii.gz"))
+ nib.save(im2, filename=jph(pfo_examples, "cubes_in_space.nii.gz"))
- if creation_list['Sandwich']:
+ if creation_list["Sandwich"]:
omega = [9, 9, 10]
sandwich = np.zeros(omega)
@@ -326,9 +373,9 @@ def generate_figures(creation_list, pfo_examples, segmentation_levels=7, sigma_s
sandwich[:, 5:, :] = 4 * np.ones([9, 4, 10])
im_sandwich = nib.Nifti1Image(sandwich, affine=np.diag([0.1, 0.2, 0.3, 1]))
- nib.save(im_sandwich, filename=jph(pfo_examples, 'sandwich.nii.gz'))
+ nib.save(im_sandwich, filename=jph(pfo_examples, "sandwich.nii.gz"))
- if creation_list['Four-folds']:
+ if creation_list["Four-folds"]:
# Case A: best case, the two ellipsoids are overlapping - label id = 1
# Case B: ellipsoids are only translated - label id = 2
# Case C: no dispersion, low precision - label id = 3
@@ -343,66 +390,74 @@ def generate_figures(creation_list, pfo_examples, segmentation_levels=7, sigma_s
second_focus_11 = [72, 72, 23]
twice_a_11 = 20
- elli_11 = ellipsoid_shape(omega, first_focus_11, second_focus_11, twice_a_11,
- foreground_intensity=foreground, dtype=np.float64)
+ elli_11 = ellipsoid_shape(
+ omega, first_focus_11, second_focus_11, twice_a_11, foreground_intensity=foreground, dtype=np.float64,
+ )
first_focus_21 = [79, 79, 25]
second_focus_21 = [72, 72, 23]
twice_a_21 = 20
- elli_21 = ellipsoid_shape(omega, first_focus_21, second_focus_21, twice_a_21,
- foreground_intensity=foreground, dtype=np.float64)
+ elli_21 = ellipsoid_shape(
+ omega, first_focus_21, second_focus_21, twice_a_21, foreground_intensity=foreground, dtype=np.float64,
+ )
# B) center 25, 75, 25
first_focus_12 = [29, 79, 25]
second_focus_12 = [24, 72, 23]
twice_a_12 = 20
- elli_12 = ellipsoid_shape(omega, first_focus_12, second_focus_12, twice_a_12,
- foreground_intensity=foreground, dtype=np.float64)
+ elli_12 = ellipsoid_shape(
+ omega, first_focus_12, second_focus_12, twice_a_12, foreground_intensity=foreground, dtype=np.float64,
+ )
first_focus_22 = [24, 74, 20] # - 5 respect to the 1
second_focus_22 = [19, 67, 18]
twice_a_22 = 20
- elli_22 = ellipsoid_shape(omega, first_focus_22, second_focus_22, twice_a_22,
- foreground_intensity=foreground, dtype=np.float64)
+ elli_22 = ellipsoid_shape(
+ omega, first_focus_22, second_focus_22, twice_a_22, foreground_intensity=foreground, dtype=np.float64,
+ )
# C) Center 75, 25, 25
first_focus_13 = [75, 25, 25] # shpere
second_focus_13 = [75, 25, 25]
twice_a_13 = 24
- elli_13 = ellipsoid_shape(omega, first_focus_13, second_focus_13, twice_a_13,
- foreground_intensity=foreground, dtype=np.float64)
+ elli_13 = ellipsoid_shape(
+ omega, first_focus_13, second_focus_13, twice_a_13, foreground_intensity=foreground, dtype=np.float64,
+ )
- r = .5 * twice_a_13
+ r = 0.5 * twice_a_13
# f = 15
# a = np.sqrt( (f ** 2 + np.sqrt(f**4 + 4*r**4)) / 2. )
a = 22
- f = np.sqrt((a ** 4 - r ** 4) / float(a ** 2))
+ f = np.sqrt((a**4 - r**4) / float(a**2))
assert f < a, [f, a]
first_focus_23 = [75 - f, 25, 25] # ellipsoid
second_focus_23 = [75 + f, 25, 25]
twice_a_23 = 2 * a
- elli_23 = ellipsoid_shape(omega, first_focus_23, second_focus_23, twice_a_23,
- foreground_intensity=foreground, dtype=np.float64)
+ elli_23 = ellipsoid_shape(
+ omega, first_focus_23, second_focus_23, twice_a_23, foreground_intensity=foreground, dtype=np.float64,
+ )
# D) Center (25, 25, 25)
first_focus_14 = [30, 35, 25]
second_focus_14 = [30, 30, 25]
twice_a_14 = 15
- elli_14 = ellipsoid_shape(omega, first_focus_14, second_focus_14, twice_a_14,
- foreground_intensity=foreground, dtype=np.float64)
+ elli_14 = ellipsoid_shape(
+ omega, first_focus_14, second_focus_14, twice_a_14, foreground_intensity=foreground, dtype=np.float64,
+ )
first_focus_24 = [10, 12, 25]
second_focus_24 = [9, 8, 25]
twice_a_24 = 15
- elli_24 = ellipsoid_shape(omega, first_focus_24, second_focus_24, twice_a_24,
- foreground_intensity=foreground, dtype=np.float64)
+ elli_24 = ellipsoid_shape(
+ omega, first_focus_24, second_focus_24, twice_a_24, foreground_intensity=foreground, dtype=np.float64,
+ )
# ---------- #
# image one: #
@@ -410,7 +465,7 @@ def generate_figures(creation_list, pfo_examples, segmentation_levels=7, sigma_s
elli_one = elli_11 + 2 * elli_12 + 3 * elli_13 + 4 * elli_14
im_four_folds_one = nib.Nifti1Image(elli_one, np.eye(4))
- nib.save(img=im_four_folds_one, filename=jph(pfo_examples, 'fourfolds_one.nii.gz'))
+ nib.save(img=im_four_folds_one, filename=jph(pfo_examples, "fourfolds_one.nii.gz"))
# ---------- #
# image two: #
@@ -418,20 +473,22 @@ def generate_figures(creation_list, pfo_examples, segmentation_levels=7, sigma_s
elli_two = elli_21 + 2 * elli_22 + 3 * elli_23 + 4 * elli_24
im_four_folds_two = nib.Nifti1Image(elli_two, np.eye(4))
- nib.save(img=im_four_folds_two, filename=jph(pfo_examples, 'fourfolds_two.nii.gz'))
-
-
-if __name__ == '__main__':
- creation_steps = {'Examples folder' : True,
- 'Punt e mes' : True,
- 'C' : True,
- 'Planetaruim' : True,
- 'Buckle ellipsoids' : True,
- 'Ellipsoids family' : True,
- 'Cubes in the sky' : True,
- 'Sandwich' : True,
- 'Four-folds' : True}
-
- path_folder_examples = jph(root_dir, 'data_examples')
+ nib.save(img=im_four_folds_two, filename=jph(pfo_examples, "fourfolds_two.nii.gz"))
+
+
+if __name__ == "__main__":
+ creation_steps = {
+ "Examples folder": True,
+ "Punt e mes": True,
+ "C": True,
+ "Planetaruim": True,
+ "Buckle ellipsoids": True,
+ "Ellipsoids family": True,
+ "Cubes in the sky": True,
+ "Sandwich": True,
+ "Four-folds": True,
+ }
+
+ path_folder_examples = jph(root_dir, "data_examples")
generate_figures(creation_steps, path_folder_examples)
diff --git a/examples/caliber_usage_distances_example.py b/examples/caliber_usage_distances_example.py
index 0e75a3b..8e9d127 100644
--- a/examples/caliber_usage_distances_example.py
+++ b/examples/caliber_usage_distances_example.py
@@ -2,14 +2,14 @@
from nilabels.agents.agents_controller import AgentsController as LT
from nilabels.definitions import root_dir
-from nilabels.tools.caliber.distances import dice_score, covariance_distance, hausdorff_distance
+from nilabels.tools.caliber.distances import covariance_distance, dice_score, hausdorff_distance
-if __name__ == '__main__':
+if __name__ == "__main__":
# Paths to input
- pfo_examples = jph(root_dir, 'data_examples')
- pfi_seg1 = jph(pfo_examples, 'fourfolds_one.nii.gz')
- pfi_seg2 = jph(pfo_examples, 'fourfolds_two.nii.gz')
+ pfo_examples = jph(root_dir, "data_examples")
+ pfi_seg1 = jph(pfo_examples, "fourfolds_one.nii.gz")
+ pfi_seg2 = jph(pfo_examples, "fourfolds_two.nii.gz")
where_to_save = None
# Instantiate a Labels Manager class
diff --git a/examples/caliber_usage_example.py b/examples/caliber_usage_example.py
index c5562ba..5b2c23a 100644
--- a/examples/caliber_usage_example.py
+++ b/examples/caliber_usage_example.py
@@ -6,48 +6,47 @@
from nilabels.agents.agents_controller import AgentsController as NiL
from nilabels.definitions import root_dir
-if __name__ == '__main__':
+if __name__ == "__main__":
- examples_folder = jph(root_dir, 'data_examples')
+ examples_folder = jph(root_dir, "data_examples")
- pfi_im = jph(examples_folder, 'cubes_in_space.nii.gz')
- pfi_im_bin = jph(examples_folder, 'cubes_in_space_bin.nii.gz')
+ pfi_im = jph(examples_folder, "cubes_in_space.nii.gz")
+ pfi_im_bin = jph(examples_folder, "cubes_in_space_bin.nii.gz")
for p in [pfi_im, pfi_im_bin, examples_folder]:
if not os.path.exists(p):
- raise IOError('Run lm.tools.benchmarking.generate_images_examples.py to create the images examples before this, please.')
+ raise OSError("Run lm.tools.benchmarking.generate_images_examples.py to create the images examples before this, please.")
# total volume:
m = NiL()
- print('The image contains 4 cubes of sides 11, 17, 19 and 9:\n')
- print('11**3 + 17**3 + 19**3 + 9**3 = {} '.format(11**3 + 17**3 + 19**3 + 9**3))
- print('sa.get_total_volume() = {} '.format(m.measure.get_total_volume(pfi_im)['Volume'].values))
+ print("The image contains 4 cubes of sides 11, 17, 19 and 9:\n")
+ print(f"11**3 + 17**3 + 19**3 + 9**3 = {11**3 + 17**3 + 19**3 + 9**3} ")
+ print("sa.get_total_volume() = {} ".format(m.measure.get_total_volume(pfi_im)["Volume"].values))
# Get volumes per label:
- print('The 4 cubes of sides 11, 17, 19 and 9 are labelled 1, 2, 3 and 4 resp.:')
- print('Volume measured label 1 = {}'.format(m.measure.volume(pfi_im, labels=1)['Volume'].values))
- print('11**3 = {}'.format(11 ** 3))
- print('Volume measured label 2 = {}'.format(m.measure.volume(pfi_im, labels=2)['Volume'].values))
- print('17**3 = {}'.format(17 ** 3))
- print('Volume measured label 3 = {}'.format(m.measure.volume(pfi_im, labels=3)['Volume'].values))
- print('19**3 = {}'.format(19 ** 3))
- print('Volume measured label 4 = {}'.format(m.measure.volume(pfi_im, labels=4)['Volume'].values))
- print('9**3 = {}'.format(9 ** 3))
- print('Volume measured labels ([1, 3]) = {}'.format(m.measure.volume(pfi_im, labels=[[1, 3]])['Volume'].values))
- print('11**3 + 19**3 = {}'.format(11**3 + 19**3))
- print('\nTo sum up: \n')
- print('Volume measured labels ([1, 2, 3, 4, [1, 3]]) = \n{}\n'.format(
- m.measure.volume(pfi_im, labels=[1, 2, 3, 4, [1, 3], [1, 2, 3, 4]])))
- print('Total volume = {} \n'.format(m.measure.get_total_volume(pfi_im)))
- print('------------')
+ print("The 4 cubes of sides 11, 17, 19 and 9 are labelled 1, 2, 3 and 4 resp.:")
+ print("Volume measured label 1 = {}".format(m.measure.volume(pfi_im, labels=1)["Volume"].values))
+ print(f"11**3 = {11 ** 3}")
+ print("Volume measured label 2 = {}".format(m.measure.volume(pfi_im, labels=2)["Volume"].values))
+ print(f"17**3 = {17 ** 3}")
+ print("Volume measured label 3 = {}".format(m.measure.volume(pfi_im, labels=3)["Volume"].values))
+ print(f"19**3 = {19 ** 3}")
+ print("Volume measured label 4 = {}".format(m.measure.volume(pfi_im, labels=4)["Volume"].values))
+ print(f"9**3 = {9 ** 3}")
+ print("Volume measured labels ([1, 3]) = {}".format(m.measure.volume(pfi_im, labels=[[1, 3]])["Volume"].values))
+ print(f"11**3 + 19**3 = {11**3 + 19**3}")
+ print("\nTo sum up: \n")
+ print(f"Volume measured labels ([1, 2, 3, 4, [1, 3]]) = \n{m.measure.volume(pfi_im, labels=[1, 2, 3, 4, [1, 3], [1, 2, 3, 4]])}\n")
+ print(f"Total volume = {m.measure.get_total_volume(pfi_im)} \n")
+ print("------------")
# Get volumes under each label, given the image weight, corresponding to the label itself:
vals_below_labels = m.measure.values_below_labels(pfi_im, pfi_im, labels=[1, 2, 3, 4, 5, [1, 3]])
- print('average below labels [1, 2, 3, 4, [1, 3]] = \n{}'.format(vals_below_labels))
- print('mu, std below label 1 = {} {}'.format(np.mean(vals_below_labels['1']), np.std(vals_below_labels['1'])))
- print('mu, std below label 2 = {} {}'.format(np.mean(vals_below_labels['2']), np.std(vals_below_labels['2'])))
- print('mu, std below label 3 = {} {}'.format(np.mean(vals_below_labels['3']), np.std(vals_below_labels['3'])))
- print('mu, std below label 4 = {} {}'.format(np.mean(vals_below_labels['4']), np.std(vals_below_labels['4'])))
+ print(f"average below labels [1, 2, 3, 4, [1, 3]] = \n{vals_below_labels}")
+ print("mu, std below label 1 = {} {}".format(np.mean(vals_below_labels["1"]), np.std(vals_below_labels["1"])))
+ print("mu, std below label 2 = {} {}".format(np.mean(vals_below_labels["2"]), np.std(vals_below_labels["2"])))
+ print("mu, std below label 3 = {} {}".format(np.mean(vals_below_labels["3"]), np.std(vals_below_labels["3"])))
+ print("mu, std below label 4 = {} {}".format(np.mean(vals_below_labels["4"]), np.std(vals_below_labels["4"])))
# print('mu, std below label 5 = {} {}'.format(np.mean(vals_below_labels['5']), np.std(vals_below_labels['5'])))
- print('mu, std below label [1, 3] = {} {}'.format(np.mean(vals_below_labels['[1, 3]']), np.std(vals_below_labels['[1, 3]'])))
+ print("mu, std below label [1, 3] = {} {}".format(np.mean(vals_below_labels["[1, 3]"]), np.std(vals_below_labels["[1, 3]"])))
- print('\nValues as they are reported: {}'.format(vals_below_labels))
+ print(f"\nValues as they are reported: {vals_below_labels}")
diff --git a/examples/manipulator_example.py b/examples/manipulator_example.py
index cd77cc8..feb4869 100644
--- a/examples/manipulator_example.py
+++ b/examples/manipulator_example.py
@@ -10,32 +10,33 @@
from nilabels.definitions import root_dir
from nilabels.tools.aux_methods.utils_nib import set_new_data
-if __name__ == '__main__':
-
- if not os.path.exists(jph(root_dir, 'data_examples')):
- print('Run generate_simple_phantoms.py before this, please.')
- raise IOError
+if __name__ == "__main__":
+ if not os.path.exists(jph(root_dir, "data_examples")):
+ print("Run generate_simple_phantoms.py before this, please.")
+ raise OSError
# Create output folder:
- cmd = 'mkdir -p {}'.format(jph(root_dir, 'data_output'))
+ cmd = "mkdir -p {}".format(jph(root_dir, "data_output"))
os.system(cmd)
# Instantiate a manager from the class LabelsManager
- lt = AgentsController(jph(root_dir, 'data_examples'), jph(root_dir, 'data_output'))
- print('Input folder: ' + lt._pfo_in)
- print('Output folder: ' + lt._pfo_out)
-
- run_example = {'Relabel' : False,
- 'Permute' : False,
- 'Erase' : False,
- 'Assign all others a value' : False,
- 'Keep one label' : False,
- 'Extend slice' : False,
- 'Split in 4d' : False,
- 'Split in 4d only some' : False,
- 'Merge in 4d' : True,
- 'Axial symmetrisation' : True,
- 'Symmetrisation with registration' : True}
+ lt = AgentsController(jph(root_dir, "data_examples"), jph(root_dir, "data_output"))
+ print("Input folder: " + lt._pfo_in) # noqa: SLF001
+ print("Output folder: " + lt._pfo_out) # noqa: SLF001
+
+ run_example = {
+ "Relabel": False,
+ "Permute": False,
+ "Erase": False,
+ "Assign all others a value": False,
+ "Keep one label": False,
+ "Extend slice": False,
+ "Split in 4d": False,
+ "Split in 4d only some": False,
+ "Merge in 4d": True,
+ "Axial symmetrisation": True,
+ "Symmetrisation with registration": True,
+ }
open_figures = True
@@ -43,252 +44,275 @@
Example 1:
Relabel the segmentation punt_seg.nii.gz increasing the label values by 1.
"""
- if run_example['Relabel']:
-
+ if run_example["Relabel"]:
# data:
- fin_punt_seg_original = 'punt_seg.nii.gz'
- fin_punt_seg_new = 'punt_seg_relabelled.nii.gz'
+ fin_punt_seg_original = "punt_seg.nii.gz"
+ fin_punt_seg_new = "punt_seg_relabelled.nii.gz"
list_old_labels = [1, 2, 3, 4, 5, 6]
list_new_labels = [2, 3, 4, 5, 6, 7]
# Using the manager:
- lt.manipulate_labels.relabel(fin_punt_seg_original, fin_punt_seg_new,
- list_old_labels, list_new_labels)
+ lt.manipulate_labels.relabel(fin_punt_seg_original, fin_punt_seg_new, list_old_labels, list_new_labels)
# Without the managers: loading the data and applying the relabeller
- im_seg = nib.load(jph(root_dir, 'data_examples', fin_punt_seg_original))
- data_seg = im_seg.get_data()
+ im_seg = nib.load(jph(root_dir, "data_examples", fin_punt_seg_original))
+ data_seg = im_seg.get_fdata()
data_seg_new = rel.relabeller(data_seg, list_old_labels, list_new_labels)
im_relabelled = set_new_data(im_seg, data_seg_new)
# Results comparison:
- nib_seg_new = nib.load(jph(root_dir, 'data_output', fin_punt_seg_new))
- nib_seg_new_data = nib_seg_new.get_data()
+ nib_seg_new = nib.load(jph(root_dir, "data_output", fin_punt_seg_new))
+ nib_seg_new_data = nib_seg_new.get_fdata()
np.testing.assert_array_equal(data_seg_new, nib_seg_new_data)
if open_figures:
# figure before:
- cmd = 'itksnap -g {0} -s {1}'.format(
- jph(root_dir, 'data_examples', 'punt.nii.gz'),
- jph(root_dir, 'data_examples', fin_punt_seg_original))
+ cmd = "itksnap -g {0} -s {1}".format(
+ jph(root_dir, "data_examples", "punt.nii.gz"),
+ jph(root_dir, "data_examples", fin_punt_seg_original),
+ )
os.system(cmd)
# figure after
- cmd = 'itksnap -g {0} -s {1}'.format(
- jph(root_dir, 'data_examples', 'punt.nii.gz'),
- jph(root_dir, 'data_output', fin_punt_seg_new))
+ cmd = "itksnap -g {0} -s {1}".format(
+ jph(root_dir, "data_examples", "punt.nii.gz"),
+ jph(root_dir, "data_output", fin_punt_seg_new),
+ )
os.system(cmd)
"""
Example 2:
Permute labels according to a given permutation
"""
- if run_example['Permute']:
-
+ if run_example["Permute"]:
# data:
- fin_punt_seg_original = 'punt_seg.nii.gz'
- fin_punt_seg_new = 'punt_seg.nii.gz'
+ fin_punt_seg_original = "punt_seg.nii.gz"
+ fin_punt_seg_new = "punt_seg.nii.gz"
perm = [[1, 2, 3], [3, 1, 2]]
# Using the manager:
lt.manipulate_labels.permute_labels(fin_punt_seg_original, fin_punt_seg_new, perm)
# without the manager:
- im_seg = nib.load(jph(root_dir, 'data_examples', fin_punt_seg_original))
- data_seg = im_seg.get_data()
+ im_seg = nib.load(jph(root_dir, "data_examples", fin_punt_seg_original))
+ data_seg = im_seg.get_fdata()
data_seg_new = rel.permute_labels(data_seg, perm)
# Results comparison:
- nib_seg_new = nib.load(jph(root_dir, 'data_output', fin_punt_seg_new))
- nib_seg_new_data = nib_seg_new.get_data()
+ nib_seg_new = nib.load(jph(root_dir, "data_output", fin_punt_seg_new))
+ nib_seg_new_data = nib_seg_new.get_fdata()
np.testing.assert_array_equal(data_seg_new, nib_seg_new_data)
if open_figures:
# figure before:
- cmd = 'itksnap -g {0} -s {1}'.format(
- jph(root_dir, 'data_examples', 'punt.nii.gz'),
- jph(root_dir, 'data_examples', fin_punt_seg_original))
+ cmd = "itksnap -g {0} -s {1}".format(
+ jph(root_dir, "data_examples", "punt.nii.gz"),
+ jph(root_dir, "data_examples", fin_punt_seg_original),
+ )
os.system(cmd)
# figure after
- cmd = 'itksnap -g {0} -s {1}'.format(
- jph(root_dir, 'data_examples', 'punt.nii.gz'),
- jph(root_dir, 'data_output', fin_punt_seg_new))
+ cmd = "itksnap -g {0} -s {1}".format(
+ jph(root_dir, "data_examples", "punt.nii.gz"),
+ jph(root_dir, "data_output", fin_punt_seg_new),
+ )
os.system(cmd)
"""
Example 3:
Erase some labels.
"""
- if run_example['Erase']:
-
+ if run_example["Erase"]:
# data:
- fin_punt_seg_original = 'punt_seg.nii.gz'
- fin_punt_seg_new = 'punt_seg.nii.gz'
+ fin_punt_seg_original = "punt_seg.nii.gz"
+ fin_punt_seg_new = "punt_seg.nii.gz"
labels_to_erase = [4, 5, 6]
# using the manager:
lt.manipulate_labels.erase_labels(fin_punt_seg_original, fin_punt_seg_new, labels_to_erase=labels_to_erase)
# without the manager:
- im_seg = nib.load(jph(root_dir, 'data_examples', fin_punt_seg_original))
- data_seg = im_seg.get_data()
+ im_seg = nib.load(jph(root_dir, "data_examples", fin_punt_seg_original))
+ data_seg = im_seg.get_fdata()
data_seg_new = rel.erase_labels(data_seg, labels_to_erase)
# Results comparison:
- nib_seg_new = nib.load(jph(root_dir, 'data_output', fin_punt_seg_new))
- nib_seg_new_data = nib_seg_new.get_data()
+ nib_seg_new = nib.load(jph(root_dir, "data_output", fin_punt_seg_new))
+ nib_seg_new_data = nib_seg_new.get_fdata()
np.testing.assert_array_equal(data_seg_new, nib_seg_new_data)
if open_figures:
# figure before:
- cmd = 'itksnap -g {0} -s {1}'.format(
- jph(root_dir, 'data_examples', 'punt.nii.gz'),
- jph(root_dir, 'data_examples', fin_punt_seg_original))
+ cmd = "itksnap -g {0} -s {1}".format(
+ jph(root_dir, "data_examples", "punt.nii.gz"),
+ jph(root_dir, "data_examples", fin_punt_seg_original),
+ )
os.system(cmd)
# figure after
- cmd = 'itksnap -g {0} -s {1}'.format(
- jph(root_dir, 'data_examples', 'punt.nii.gz'),
- jph(root_dir, 'data_output', fin_punt_seg_new))
+ cmd = "itksnap -g {0} -s {1}".format(
+ jph(root_dir, "data_examples", "punt.nii.gz"),
+ jph(root_dir, "data_output", fin_punt_seg_new),
+ )
os.system(cmd)
"""
Example 4:
Assign to the excluded labels the same value.
"""
- if run_example['Assign all others a value']:
-
+ if run_example["Assign all others a value"]:
# data:
- fin_punt_seg_original = 'punt_seg.nii.gz'
- fin_punt_seg_new = 'punt_seg.nii.gz'
+ fin_punt_seg_original = "punt_seg.nii.gz"
+ fin_punt_seg_new = "punt_seg.nii.gz"
labels_to_keep = [1, 3, 5]
new_value = 100
# using the manager:
- lt.manipulate_labels.assign_all_other_labels_the_same_value(fin_punt_seg_original,
- fin_punt_seg_new, labels_to_keep=labels_to_keep, same_value_label=new_value)
+ lt.manipulate_labels.assign_all_other_labels_the_same_value(
+ fin_punt_seg_original,
+ fin_punt_seg_new,
+ labels_to_keep=labels_to_keep,
+ same_value_label=new_value,
+ )
# without the manager:
- im_seg = nib.load(jph(root_dir, 'data_examples', fin_punt_seg_original))
- data_seg = im_seg.get_data()
- data_seg_new = rel.assign_all_other_labels_the_same_value(data_seg,
- labels_to_keep=labels_to_keep, same_value_label=new_value)
+ im_seg = nib.load(jph(root_dir, "data_examples", fin_punt_seg_original))
+ data_seg = im_seg.get_fdata()
+ data_seg_new = rel.assign_all_other_labels_the_same_value(
+ data_seg,
+ labels_to_keep=labels_to_keep,
+ same_value_label=new_value,
+ )
# Results comparison:
- nib_seg_new = nib.load(jph(root_dir, 'data_output', fin_punt_seg_new))
- nib_seg_new_data = nib_seg_new.get_data()
+ nib_seg_new = nib.load(jph(root_dir, "data_output", fin_punt_seg_new))
+ nib_seg_new_data = nib_seg_new.get_fdata()
np.testing.assert_array_equal(data_seg_new, nib_seg_new_data)
if open_figures:
# figure before:
- cmd = 'itksnap -g {0} -s {1}'.format(
- jph(root_dir, 'data_examples', 'punt.nii.gz'),
- jph(root_dir, 'data_examples', fin_punt_seg_original))
+ cmd = "itksnap -g {0} -s {1}".format(
+ jph(root_dir, "data_examples", "punt.nii.gz"),
+ jph(root_dir, "data_examples", fin_punt_seg_original),
+ )
os.system(cmd)
# figure after
- cmd = 'itksnap -g {0} -s {1}'.format(
- jph(root_dir, 'data_examples', 'punt.nii.gz'),
- jph(root_dir, 'data_output', fin_punt_seg_new))
+ cmd = "itksnap -g {0} -s {1}".format(
+ jph(root_dir, "data_examples", "punt.nii.gz"),
+ jph(root_dir, "data_output", fin_punt_seg_new),
+ )
os.system(cmd)
"""
Example 5:
Keep only one label.
"""
- if run_example['Keep one label']:
+ if run_example["Keep one label"]:
# data:
- fin_punt_seg_original = 'punt_seg.nii.gz'
- fin_punt_seg_new = 'punt_seg.nii.gz'
+ fin_punt_seg_original = "punt_seg.nii.gz"
+ fin_punt_seg_new = "punt_seg.nii.gz"
label_to_keep = 6
# using the manager:
- lt.manipulate_labels.keep_one_label(fin_punt_seg_original, fin_punt_seg_new,
- label_to_keep=label_to_keep)
+ lt.manipulate_labels.keep_one_label(fin_punt_seg_original, fin_punt_seg_new, label_to_keep=label_to_keep)
# without the manager:
- im_seg = nib.load(jph(root_dir, 'data_examples', fin_punt_seg_original))
- data_seg = im_seg.get_data()
+ im_seg = nib.load(jph(root_dir, "data_examples", fin_punt_seg_original))
+ data_seg = im_seg.get_fdata()
data_seg_new = rel.keep_only_one_label(data_seg, label_to_keep=label_to_keep)
# Results comparison:
- nib_seg_new = nib.load(jph(root_dir, 'data_output', fin_punt_seg_new))
- nib_seg_new_data = nib_seg_new.get_data()
+ nib_seg_new = nib.load(jph(root_dir, "data_output", fin_punt_seg_new))
+ nib_seg_new_data = nib_seg_new.get_fdata()
np.testing.assert_array_equal(data_seg_new, nib_seg_new_data)
if open_figures:
# figure before:
- cmd = 'itksnap -g {0} -s {1}'.format(
- jph(root_dir, 'data_examples', 'punt.nii.gz'),
- jph(root_dir, 'data_examples', fin_punt_seg_original))
+ cmd = "itksnap -g {0} -s {1}".format(
+ jph(root_dir, "data_examples", "punt.nii.gz"),
+ jph(root_dir, "data_examples", fin_punt_seg_original),
+ )
os.system(cmd)
# figure after
- cmd = 'itksnap -g {0} -s {1}'.format(
- jph(root_dir, 'data_examples', 'punt.nii.gz'),
- jph(root_dir, 'data_output', fin_punt_seg_new))
+ cmd = "itksnap -g {0} -s {1}".format(
+ jph(root_dir, "data_examples", "punt.nii.gz"),
+ jph(root_dir, "data_output", fin_punt_seg_new),
+ )
os.system(cmd)
"""
Example 6:
extend slice new dimension (not only for segmentations).
"""
- if run_example['Extend slice']:
+ if run_example["Extend slice"]:
# data:
- fin_punt_original = 'punt.nii.gz'
- fin_punt_new = 'punt_extended.nii.gz'
+ fin_punt_original = "punt.nii.gz"
+ fin_punt_new = "punt_extended.nii.gz"
- fin_punt_seg_original = 'punt_seg.nii.gz'
- fin_punt_seg_new = 'punt_seg_extended.nii.gz'
+ fin_punt_seg_original = "punt_seg.nii.gz"
+ fin_punt_seg_new = "punt_seg_extended.nii.gz"
new_axis = 3
num_slices = 5
# using the manager:
- lt.manipulate_intensities.extend_slice_new_dimension(fin_punt_original, fin_punt_new,
- new_axis=new_axis, num_slices=num_slices)
- lt.manipulate_intensities.extend_slice_new_dimension(fin_punt_seg_original, fin_punt_seg_new,
- new_axis=new_axis, num_slices=num_slices)
+ lt.manipulate_intensities.extend_slice_new_dimension(
+ fin_punt_original,
+ fin_punt_new,
+ new_axis=new_axis,
+ num_slices=num_slices,
+ )
+ lt.manipulate_intensities.extend_slice_new_dimension(
+ fin_punt_seg_original,
+ fin_punt_seg_new,
+ new_axis=new_axis,
+ num_slices=num_slices,
+ )
# without the manager:
- im_seg = nib.load(jph(root_dir, 'data_examples', fin_punt_seg_original))
- data_seg = im_seg.get_data()
- data_seg_new = np.stack([data_seg, ] * num_slices, axis=new_axis)
+ im_seg = nib.load(jph(root_dir, "data_examples", fin_punt_seg_original))
+ data_seg = im_seg.get_fdata()
+ data_seg_new = np.stack([data_seg] * num_slices, axis=new_axis)
# Results comparison:
- nib_seg_new = nib.load(jph(root_dir, 'data_output', fin_punt_seg_new))
- nib_seg_new_data = nib_seg_new.get_data()
+ nib_seg_new = nib.load(jph(root_dir, "data_output", fin_punt_seg_new))
+ nib_seg_new_data = nib_seg_new.get_fdata()
np.testing.assert_array_equal(data_seg_new, nib_seg_new_data)
if open_figures:
# figure before:
- cmd = 'itksnap -g {0} -s {1}'.format(
- jph(root_dir, 'data_examples', fin_punt_original),
- jph(root_dir, 'data_examples', fin_punt_seg_original))
+ cmd = "itksnap -g {0} -s {1}".format(
+ jph(root_dir, "data_examples", fin_punt_original),
+ jph(root_dir, "data_examples", fin_punt_seg_original),
+ )
os.system(cmd)
# figure after - (*) the segmentation can be loaded in itksnap in only one slice.
- cmd = 'itksnap -g {0} -s {1}'.format(
- jph(root_dir, 'data_output', fin_punt_new),
- jph(root_dir, 'data_examples', fin_punt_seg_original)) # (*)
+ cmd = "itksnap -g {0} -s {1}".format(
+ jph(root_dir, "data_output", fin_punt_new),
+ jph(root_dir, "data_examples", fin_punt_seg_original),
+ ) # (*)
os.system(cmd)
"""
Example 7:
Split labels in 4d
"""
- if run_example['Split in 4d']:
+ if run_example["Split in 4d"]:
# data:
- fin_seg_original = 'mes_seg.nii.gz'
- fin_seg_new = 'mes_seg_4d.nii.gz'
+ fin_seg_original = "mes_seg.nii.gz"
+ fin_seg_new = "mes_seg_4d.nii.gz"
# using the manager
lt.manipulate_intensities.split_in_4d(fin_seg_original, fin_seg_new)
# without the manager:
- im_seg = nib.load(jph(root_dir, 'data_examples', fin_seg_original))
- data_seg = im_seg.get_data()
+ im_seg = nib.load(jph(root_dir, "data_examples", fin_seg_original))
+ data_seg = im_seg.get_fdata()
list_labels = list(np.sort(list(set(data_seg.flat))))
data_seg_new = sp.split_labels_to_4d(data_seg, list_labels)
# results comparison
- nib_seg_new = nib.load(jph(root_dir, 'data_output', fin_seg_new))
- nib_seg_new_data = nib_seg_new.get_data()
+ nib_seg_new = nib.load(jph(root_dir, "data_output", fin_seg_new))
+ nib_seg_new_data = nib_seg_new.get_fdata()
np.testing.assert_array_equal(data_seg_new, nib_seg_new_data)
if open_figures:
# figure before and after:
- cmd = 'itksnap -g {0}; itksnap -g {1};'.format(
- jph(root_dir, 'data_examples', fin_seg_original), jph(root_dir, 'data_output', fin_seg_new))
+ cmd = "itksnap -g {0}; itksnap -g {1};".format(
+ jph(root_dir, "data_examples", fin_seg_original),
+ jph(root_dir, "data_output", fin_seg_new),
+ )
os.system(cmd)
"""
@@ -296,84 +320,98 @@
Split labels in 4d, only some labels.
"""
- if run_example['Split in 4d only some']:
+ if run_example["Split in 4d only some"]:
# data:
- fin_punt_seg_original = 'punt_seg.nii.gz'
- fin_punt_seg_new = 'punt_seg_4d.nii.gz'
+ fin_punt_seg_original = "punt_seg.nii.gz"
+ fin_punt_seg_new = "punt_seg_4d.nii.gz"
list_labels = [4, 5, 6]
# using the manager
- lt.manipulate_intensities.split_in_4d(fin_punt_seg_original, fin_punt_seg_new, list_labels=list_labels,
- keep_original_values=False)
+ lt.manipulate_intensities.split_in_4d(
+ fin_punt_seg_original,
+ fin_punt_seg_new,
+ list_labels=list_labels,
+ keep_original_values=False,
+ )
# without the manager:
- im_seg = nib.load(jph(root_dir, 'data_examples', fin_punt_seg_original))
- data_seg = im_seg.get_data()
+ im_seg = nib.load(jph(root_dir, "data_examples", fin_punt_seg_original))
+ data_seg = im_seg.get_fdata()
data_seg_new = sp.split_labels_to_4d(data_seg, list_labels, keep_original_values=False)
# results comparison
- nib_seg_new = nib.load(jph(root_dir, 'data_output', fin_punt_seg_new))
- nib_seg_new_data = nib_seg_new.get_data()
+ nib_seg_new = nib.load(jph(root_dir, "data_output", fin_punt_seg_new))
+ nib_seg_new_data = nib_seg_new.get_fdata()
np.testing.assert_array_equal(data_seg_new, nib_seg_new_data)
if open_figures:
# figure after:
- cmd = 'itksnap -g {0}'.format(
- jph(root_dir, 'data_output', fin_punt_seg_new))
+ cmd = "itksnap -g {0}".format(jph(root_dir, "data_output", fin_punt_seg_new))
os.system(cmd)
"""
Example 9:
Merge labels in 4d
"""
- if run_example['Merge in 4d']:
+ if run_example["Merge in 4d"]:
# data:
- fin_seg_original = 'mes_seg.nii.gz'
- fin_seg_splitted = 'mes_seg_splitted.nii.gz'
- fin_seg_merged = 'mes_seg_merged.nii.gz'
+ fin_seg_original = "mes_seg.nii.gz"
+ fin_seg_splitted = "mes_seg_splitted.nii.gz"
+ fin_seg_merged = "mes_seg_merged.nii.gz"
# split:
lt.manipulate_intensities.split_in_4d(fin_seg_original, fin_seg_splitted)
# merge
- lt.set_input_data_folder(jph(root_dir, 'data_output'))
+ lt.set_input_data_folder(jph(root_dir, "data_output"))
lt.manipulate_intensities.merge_from_4d(fin_seg_splitted, fin_seg_merged)
# check you got back the same:
- pfi_seg_before = jph(root_dir, 'data_examples', fin_seg_original)
- pfi_seg_after = jph(root_dir, 'data_output', fin_seg_merged)
+ pfi_seg_before = jph(root_dir, "data_examples", fin_seg_original)
+ pfi_seg_after = jph(root_dir, "data_output", fin_seg_merged)
im_before = nib.load(pfi_seg_before)
- im_after = nib.load(pfi_seg_after)
- np.testing.assert_array_equal(im_before.get_data(), im_after.get_data())
+ im_after = nib.load(pfi_seg_after)
+ np.testing.assert_array_equal(im_before.get_fdata(), im_after.get_fdata())
# set back the input data folder to data_examples
- lt.set_input_data_folder(jph(root_dir, 'data_examples'))
+ lt.set_input_data_folder(jph(root_dir, "data_examples"))
"""
Example 10:
Symmetrise axial
"""
- if run_example['Axial symmetrisation']:
- fin_original = 'mes_seg.nii.gz'
- fin_transformed = 'mes_now_punt.nii.gz'
+ if run_example["Axial symmetrisation"]:
+ fin_original = "mes_seg.nii.gz"
+ fin_transformed = "mes_now_punt.nii.gz"
# transform mes in a punt
- lt.symmetrize.symmetrise_axial(fin_original, fin_transformed, axis='x', plane_intercept=128,
- side_to_copy='above', keep_in_data_dimensions=True)
+ lt.symmetrize.symmetrise_axial(
+ fin_original,
+ fin_transformed,
+ axis="x",
+ plane_intercept=128,
+ side_to_copy="above",
+ keep_in_data_dimensions=True,
+ )
if open_figures:
- cmd = 'itksnap -g {0}'.format(jph(root_dir, 'data_output', fin_transformed))
+ cmd = "itksnap -g {0}".format(jph(root_dir, "data_output", fin_transformed))
os.system(cmd)
"""
Example 11:
Symmetrise with registration
"""
- if run_example['Symmetrisation with registration']:
- fin_anatomy = 'ellipsoids.nii.gz'
- fin_seg = 'ellipsoids_seg_half.nii.gz'
+ if run_example["Symmetrisation with registration"]:
+ fin_anatomy = "ellipsoids.nii.gz"
+ fin_seg = "ellipsoids_seg_half.nii.gz"
# transform mes in a punt
- lt.symmetrize.symmetrise_with_registration(fin_anatomy, fin_seg,
- list_labels_input=[1, 2, 3, 4, 5, 6],
- results_folder_path=jph(root_dir, 'data_output'),
- result_img_path=jph(root_dir, 'data_output','ellipsoid_seg_SYM.nii.gz'),
- list_labels_transformed=[7, 8, 9, 10, 11, 12])
+ lt.symmetrize.symmetrise_with_registration(
+ fin_anatomy,
+ fin_seg,
+ list_labels_input=[1, 2, 3, 4, 5, 6],
+ results_folder_path=jph(root_dir, "data_output"),
+ result_img_path=jph(root_dir, "data_output", "ellipsoid_seg_SYM.nii.gz"),
+ list_labels_transformed=[7, 8, 9, 10, 11, 12],
+ )
if open_figures:
- cmd = 'itksnap -g {0} -s {1}'.format(jph(root_dir, 'data_examples', fin_anatomy),
- jph(root_dir, 'data_output','ellipsoid_seg_SYM.nii.gz'))
+ cmd = "itksnap -g {0} -s {1}".format(
+ jph(root_dir, "data_examples", fin_anatomy),
+ jph(root_dir, "data_output", "ellipsoid_seg_SYM.nii.gz"),
+ )
os.system(cmd)
diff --git a/examples/prototype_check_imperfections.py b/examples/prototype_check_imperfections.py
index c57194a..c60c620 100644
--- a/examples/prototype_check_imperfections.py
+++ b/examples/prototype_check_imperfections.py
@@ -2,6 +2,7 @@
from os.path import join as jph
import a_generate_phantoms_for_examples as gen
+
from nilabels.agents.agents_controller import AgentsController as NiL
from nilabels.definitions import root_dir
from nilabels.tools.aux_methods.label_descriptor_manager import generate_dummy_label_descriptor
@@ -9,27 +10,27 @@
# ---- GENERATE DATA ----
-if not os.path.exists(jph(root_dir, 'data_examples', 'ellipsoids_seg.nii.gz')):
+if not os.path.exists(jph(root_dir, "data_examples", "ellipsoids_seg.nii.gz")):
- creation_list = {'Examples folder' : True,
- 'Punt e mes' : False,
- 'C' : False,
- 'Planetaruim' : False,
- 'Buckle ellipsoids' : True,
- 'Ellipsoids family' : False,
- 'Cubes in the sky' : False,
- 'Sandwich' : False,
- 'Four-folds' : False}
+ creation_list = {"Examples folder" : True,
+ "Punt e mes" : False,
+ "C" : False,
+ "Planetaruim" : False,
+ "Buckle ellipsoids" : True,
+ "Ellipsoids family" : False,
+ "Cubes in the sky" : False,
+ "Sandwich" : False,
+ "Four-folds" : False}
gen.generate_figures(creation_list)
# ---- PATH MANAGER ----
-pfi_input_segm = jph(root_dir, 'data_examples', 'ellipsoids_seg.nii.gz')
+pfi_input_segm = jph(root_dir, "data_examples", "ellipsoids_seg.nii.gz")
# ---- CREATE LABELS DESCRIPTOR FOR PHANTOM ellipsoids with 0 to 6 labels ----
-pfi_labels_descriptor = jph(root_dir, 'data_examples', 'labels_descriptor_ellipsoids.txt')
+pfi_labels_descriptor = jph(root_dir, "data_examples", "labels_descriptor_ellipsoids.txt")
generate_dummy_label_descriptor(pfi_labels_descriptor, list_labels=[0, 1, 4, 5, 6, 7, 8]) # add extra labels to test
# ---- PERFORM the check ----
diff --git a/examples/prototype_clean_a_segmentation.py b/examples/prototype_clean_a_segmentation.py
index 2aea74b..b0129a4 100644
--- a/examples/prototype_clean_a_segmentation.py
+++ b/examples/prototype_clean_a_segmentation.py
@@ -1,48 +1,47 @@
import os
from os.path import join as jph
+import a_generate_phantoms_for_examples as gen
import nibabel as nib
import numpy as np
-import a_generate_phantoms_for_examples as gen
-from nilabels.agents.agents_controller import AgentsController as NiL
-from nilabels.definitions import root_dir
-
import nilabels as nil
+from nilabels.definitions import root_dir
# ---- GENERATE DATA ----
-if not os.path.exists(jph(root_dir, 'data_examples', 'ellipsoids.nii.gz')):
-
- creation_list = {'Examples folder' : True,
- 'Punt e mes' : False,
- 'C' : False,
- 'Planetaruim' : False,
- 'Buckle ellipsoids' : True,
- 'Ellipsoids family' : False,
- 'Cubes in the sky' : False,
- 'Sandwich' : False,
- 'Four-folds' : False}
+if not os.path.exists(jph(root_dir, "data_examples", "ellipsoids.nii.gz")):
+ creation_list = {
+ "Examples folder": True,
+ "Punt e mes": False,
+ "C": False,
+ "Planetaruim": False,
+ "Buckle ellipsoids": True,
+ "Ellipsoids family": False,
+ "Cubes in the sky": False,
+ "Sandwich": False,
+ "Four-folds": False,
+ }
gen.generate_figures(creation_list)
# ---- PATH MANAGER ----
# input
-pfi_input_anatomy = jph(root_dir, 'data_examples', 'ellipsoids.nii.gz')
-pfi_input_segmentation_noisy = jph(root_dir, 'data_examples', 'ellipsoids_seg_noisy.nii.gz')
-pfo_output_folder = jph(root_dir, 'data_output')
+pfi_input_anatomy = jph(root_dir, "data_examples", "ellipsoids.nii.gz")
+pfi_input_segmentation_noisy = jph(root_dir, "data_examples", "ellipsoids_seg_noisy.nii.gz")
+pfo_output_folder = jph(root_dir, "data_output")
assert os.path.exists(pfi_input_anatomy), pfi_input_anatomy
assert os.path.exists(pfi_input_segmentation_noisy), pfi_input_segmentation_noisy
assert os.path.exists(pfo_output_folder), pfo_output_folder
# Output
-log_file_before_cleaning = jph(pfo_output_folder, 'log_before_cleaning.txt')
-pfi_output_cleaned_segmentation = jph(pfo_output_folder, 'ellipsoids_segm_cleaned.nii.gz')
-log_file_after_cleaning = jph(pfo_output_folder, 'log_after_cleaning.txt')
-pfi_differece_cleaned_non_cleaned = jph(pfo_output_folder, 'difference_half_cleaned_uncleaned.nii.gz')
+log_file_before_cleaning = jph(pfo_output_folder, "log_before_cleaning.txt")
+pfi_output_cleaned_segmentation = jph(pfo_output_folder, "ellipsoids_segm_cleaned.nii.gz")
+log_file_after_cleaning = jph(pfo_output_folder, "log_after_cleaning.txt")
+pfi_differece_cleaned_non_cleaned = jph(pfo_output_folder, "difference_half_cleaned_uncleaned.nii.gz")
# ---- PROCESS ----
@@ -50,35 +49,40 @@
nl = nil.App()
# get the report before cleaning
-nl.check.number_connected_components_per_label(pfi_input_segmentation_noisy,
- where_to_save_the_log_file=log_file_before_cleaning)
+nl.check.number_connected_components_per_label(
+ pfi_input_segmentation_noisy, where_to_save_the_log_file=log_file_before_cleaning,
+)
-print('Wanted final number of components per label:')
+print("Wanted final number of components per label:")
im_input_segmentation_noisy = nib.load(pfi_input_segmentation_noisy)
-correspondences_labels_components = [[k, 1] for k in range(np.max(im_input_segmentation_noisy.get_data()) + 1) ]
+correspondences_labels_components = [[k, 1] for k in range(np.max(im_input_segmentation_noisy.get_fdata()) + 1)]
print(correspondences_labels_components)
# get the cleaned segmentation
-nl.manipulate_labels.clean_segmentation(pfi_input_segmentation_noisy, pfi_output_cleaned_segmentation,
- labels_to_clean=correspondences_labels_components, force_overwriting=True)
+nl.manipulate_labels.clean_segmentation(
+ pfi_input_segmentation_noisy,
+ pfi_output_cleaned_segmentation,
+ labels_to_clean=correspondences_labels_components,
+ force_overwriting=True,
+)
# get the report of the connected components afterwards
-nl.check.number_connected_components_per_label(pfi_output_cleaned_segmentation,
- where_to_save_the_log_file=log_file_after_cleaning)
+nl.check.number_connected_components_per_label(
+ pfi_output_cleaned_segmentation, where_to_save_the_log_file=log_file_after_cleaning,
+)
# ---- GET DIFFERENCE ----
-cmd = 'seg_maths {0} -sub {1} {2}'.format(pfi_input_segmentation_noisy, pfi_output_cleaned_segmentation,
- pfi_differece_cleaned_non_cleaned)
+cmd = f"seg_maths {pfi_input_segmentation_noisy} -sub {pfi_output_cleaned_segmentation} {pfi_differece_cleaned_non_cleaned}"
os.system(cmd)
-cmd = 'seg_maths {0} -bin {0}'.format(pfi_differece_cleaned_non_cleaned)
+cmd = f"seg_maths {pfi_differece_cleaned_non_cleaned} -bin {pfi_differece_cleaned_non_cleaned}"
os.system(cmd)
# ---- VISUALISE OUTPUT ----
-opener1 = 'itksnap -g {} -s {}'.format(pfi_input_anatomy, pfi_input_segmentation_noisy)
-opener2 = 'itksnap -g {} -s {}'.format(pfi_input_anatomy, pfi_output_cleaned_segmentation)
-opener3 = 'itksnap -g {} -s {}'.format(pfi_input_anatomy, pfi_differece_cleaned_non_cleaned)
+opener1 = f"itksnap -g {pfi_input_anatomy} -s {pfi_input_segmentation_noisy}"
+opener2 = f"itksnap -g {pfi_input_anatomy} -s {pfi_output_cleaned_segmentation}"
+opener3 = f"itksnap -g {pfi_input_anatomy} -s {pfi_differece_cleaned_non_cleaned}"
os.system(opener1)
os.system(opener2)
diff --git a/examples/prototype_open_segmentations_in_freeview.py b/examples/prototype_open_segmentations_in_freeview.py
index 10c9d87..5428933 100644
--- a/examples/prototype_open_segmentations_in_freeview.py
+++ b/examples/prototype_open_segmentations_in_freeview.py
@@ -1,5 +1,4 @@
-"""
-Use ITK-snap, its labels_descriptor.txt and freeview to get the surfaces and overlay the surface to the main image
+"""Use ITK-snap, its labels_descriptor.txt and freeview to get the surfaces and overlay the surface to the main image
directly in freeview with correct naming convention.
"""
@@ -8,10 +7,9 @@
from nilabels.tools.aux_methods.label_descriptor_manager import LabelsDescriptorManager
-def freesurfer_surface_overlayed(pfi_anatomy, pfo_stl_surfaces, pfi_descriptor, convention_descriptor='itk-snap',
- suffix_surf='surf', add_colors=True, labels_to_delineate='all'):
- """
- Manual step: from a segmentation export all the labels in stand-alone .stl files with ITK-snap, in a folder
+def freesurfer_surface_overlayed(pfi_anatomy, pfo_stl_surfaces, pfi_descriptor, convention_descriptor="itk-snap",
+ suffix_surf="surf", add_colors=True, labels_to_delineate="all"):
+ """Manual step: from a segmentation export all the labels in stand-alone .stl files with ITK-snap, in a folder
pfo_stl_surfaces, with suffix suffix_surf
:param pfi_anatomy:
:param pfo_stl_surfaces:
@@ -24,28 +22,26 @@ def freesurfer_surface_overlayed(pfi_anatomy, pfo_stl_surfaces, pfi_descriptor,
"""
ldm = LabelsDescriptorManager(pfi_descriptor, labels_descriptor_convention=convention_descriptor)
- cmd = 'source $FREESURFER_HOME/SetUpFreeSurfer.sh; freeview -v {0} -f '.format(pfi_anatomy)
+ cmd = f"source $FREESURFER_HOME/SetUpFreeSurfer.sh; freeview -v {pfi_anatomy} -f "
if labels_to_delineate:
labels_to_delineate = ldm.dict_label_descriptor.keys()[1:-1]
for k in labels_to_delineate:
- pfi_surface = os.path.join(pfo_stl_surfaces, '{0}{1:05d}.stl'.format(suffix_surf, k))
+ pfi_surface = os.path.join(pfo_stl_surfaces, f"{suffix_surf}{k:05d}.stl")
assert os.path.exists(pfi_surface), pfi_surface
if add_colors:
- triplet_rgb = '{0},{1},{2}'.format(ldm.dict_label_descriptor[k][0][0],
- ldm.dict_label_descriptor[k][0][1],
- ldm.dict_label_descriptor[k][0][2])
+ triplet_rgb = f"{ldm.dict_label_descriptor[k][0][0]},{ldm.dict_label_descriptor[k][0][1]},{ldm.dict_label_descriptor[k][0][2]}"
- cmd += ' {0}:edgecolor={1}:color={1} '.format(pfi_surface, triplet_rgb)
+ cmd += f" {pfi_surface}:edgecolor={triplet_rgb}:color={triplet_rgb} "
else:
- cmd += ' {0} '.format(pfi_surface)
+ cmd += f" {pfi_surface} "
os.system(cmd)
-if __name__ == '__main__':
- print('Step 0: create segmented atlas with phantom generator.')
- print('Step 1: Manual step - open the segmentation in ITK-Snap and export all the surfaces in .stl in the '
- 'specified folder')
- print('Step 2: run freesurfer_surface_overlayed')
+if __name__ == "__main__":
+ print("Step 0: create segmented atlas with phantom generator.")
+ print("Step 1: Manual step - open the segmentation in ITK-Snap and export all the surfaces in .stl in the "
+ "specified folder")
+ print("Step 2: run freesurfer_surface_overlayed")
diff --git a/examples/prototype_segment_an_image.py b/examples/prototype_segment_an_image.py
index 46de041..a248141 100644
--- a/examples/prototype_segment_an_image.py
+++ b/examples/prototype_segment_an_image.py
@@ -2,23 +2,24 @@
from os.path import join as jph
import a_generate_phantoms_for_examples as gen
+
from nilabels.agents.agents_controller import AgentsController as NiL
from nilabels.definitions import root_dir
# ---- GENERATE DATA ----
-if not os.path.exists(jph(root_dir, 'data_examples', 'ellipsoids.nii.gz')):
+if not os.path.exists(jph(root_dir, "data_examples", "ellipsoids.nii.gz")):
- creation_list = {'Examples folder' : True,
- 'Punt e mes' : False,
- 'C' : False,
- 'Planetaruim' : False,
- 'Buckle ellipsoids' : True,
- 'Ellipsoids family' : False,
- 'Cubes in the sky' : False,
- 'Sandwich' : False,
- 'Four-folds' : False}
+ creation_list = {"Examples folder" : True,
+ "Punt e mes" : False,
+ "C" : False,
+ "Planetaruim" : False,
+ "Buckle ellipsoids" : True,
+ "Ellipsoids family" : False,
+ "Cubes in the sky" : False,
+ "Sandwich" : False,
+ "Four-folds" : False}
gen.generate_figures(creation_list)
@@ -26,40 +27,40 @@
# ---- PATH MANAGER ----
# input:
-pfi_input_anatomy = jph(root_dir, 'data_examples', 'ellipsoids.nii.gz')
-pfo_output_folder = jph(root_dir, 'data_output')
+pfi_input_anatomy = jph(root_dir, "data_examples", "ellipsoids.nii.gz")
+pfo_output_folder = jph(root_dir, "data_output")
assert os.path.exists(pfi_input_anatomy), pfi_input_anatomy
assert os.path.exists(pfo_output_folder)
# output:
-pfi_intensities_segmentation = jph(pfo_output_folder, 'ellipsoids_segm_int.nii.gz')
-pfi_otsu_segmentation = jph(pfo_output_folder, 'ellipsoids_segm_otsu.nii.gz')
-pfi_mog_segmentation_crisp = jph(pfo_output_folder, 'ellipsoids_segm_mog_crisp.nii.gz')
-pfi_mog_segmentation_prob = jph(pfo_output_folder, 'ellipsoids_segm_mog_prob.nii.gz')
+pfi_intensities_segmentation = jph(pfo_output_folder, "ellipsoids_segm_int.nii.gz")
+pfi_otsu_segmentation = jph(pfo_output_folder, "ellipsoids_segm_otsu.nii.gz")
+pfi_mog_segmentation_crisp = jph(pfo_output_folder, "ellipsoids_segm_mog_crisp.nii.gz")
+pfi_mog_segmentation_prob = jph(pfo_output_folder, "ellipsoids_segm_mog_prob.nii.gz")
-print('---- PROCESS 1: intensities segmentation ----')
+print("---- PROCESS 1: intensities segmentation ----")
la = NiL()
la.segment.simple_intensities_thresholding(pfi_input_anatomy, pfi_intensities_segmentation, number_of_levels=5)
-print('---- PROCESS 2: Otsu ----')
+print("---- PROCESS 2: Otsu ----")
la = NiL()
-la.segment.otsu_thresholding(pfi_input_anatomy, pfi_otsu_segmentation, side='above', return_as_mask=False)
+la.segment.otsu_thresholding(pfi_input_anatomy, pfi_otsu_segmentation, side="above", return_as_mask=False)
-print('---- PROCESS 2: MoG ----')
+print("---- PROCESS 2: MoG ----")
la = NiL()
la.segment.mixture_of_gaussians(pfi_input_anatomy, pfi_mog_segmentation_crisp, pfi_mog_segmentation_prob,
K=5, see_histogram=True)
-print('---- VIEW ----')
+print("---- VIEW ----")
-opener1 = 'itksnap -g {} -s {}'.format(pfi_input_anatomy, pfi_intensities_segmentation)
-opener2 = 'itksnap -g {} -s {}'.format(pfi_input_anatomy, pfi_otsu_segmentation)
-opener3 = 'itksnap -g {} -s {}'.format(pfi_input_anatomy, pfi_mog_segmentation_crisp)
-opener4 = 'itksnap -g {} -o {}'.format(pfi_input_anatomy, pfi_mog_segmentation_prob)
+opener1 = f"itksnap -g {pfi_input_anatomy} -s {pfi_intensities_segmentation}"
+opener2 = f"itksnap -g {pfi_input_anatomy} -s {pfi_otsu_segmentation}"
+opener3 = f"itksnap -g {pfi_input_anatomy} -s {pfi_mog_segmentation_crisp}"
+opener4 = f"itksnap -g {pfi_input_anatomy} -o {pfi_mog_segmentation_prob}"
os.system(opener1)
os.system(opener2)
diff --git a/examples/prototype_symmetrise_a_segmentation.py b/examples/prototype_symmetrise_a_segmentation.py
index 5a2ea9f..3ad1081 100644
--- a/examples/prototype_symmetrise_a_segmentation.py
+++ b/examples/prototype_symmetrise_a_segmentation.py
@@ -2,23 +2,24 @@
from os.path import join as jph
import a_generate_phantoms_for_examples as gen
+
from nilabels.agents.agents_controller import AgentsController as NiL
from nilabels.definitions import root_dir
# ---- GENERATE DATA ----
-if not os.path.exists(jph(root_dir, 'data_examples', 'ellipsoids.nii.gz')):
+if not os.path.exists(jph(root_dir, "data_examples", "ellipsoids.nii.gz")):
- creation_list = {'Examples folder' : False,
- 'Punt e mes' : False,
- 'C' : False,
- 'Planetaruim' : False,
- 'Buckle ellipsoids' : True,
- 'Ellipsoids family' : False,
- 'Cubes in the sky' : False,
- 'Sandwich' : False,
- 'Four-folds' : False}
+ creation_list = {"Examples folder" : False,
+ "Punt e mes" : False,
+ "C" : False,
+ "Planetaruim" : False,
+ "Buckle ellipsoids" : True,
+ "Ellipsoids family" : False,
+ "Cubes in the sky" : False,
+ "Sandwich" : False,
+ "Four-folds" : False}
gen.generate_figures(creation_list)
@@ -27,16 +28,16 @@
# input
-pfi_input_anatomy = jph(root_dir, 'data_examples', 'ellipsoids.nii.gz')
-pfi_input_segmentation = jph(root_dir, 'data_examples', 'ellipsoids_seg_half.nii.gz')
-pfo_output_folder = jph(root_dir, 'data_output')
+pfi_input_anatomy = jph(root_dir, "data_examples", "ellipsoids.nii.gz")
+pfi_input_segmentation = jph(root_dir, "data_examples", "ellipsoids_seg_half.nii.gz")
+pfo_output_folder = jph(root_dir, "data_output")
assert os.path.exists(pfi_input_anatomy), pfi_input_anatomy
assert os.path.exists(pfi_input_segmentation), pfi_input_segmentation
assert os.path.exists(pfo_output_folder), pfo_output_folder
# output
-pfi_output_segmentation = jph(root_dir, 'data_examples', 'ellipsoids_seg_symmetrised.nii.gz')
+pfi_output_segmentation = jph(root_dir, "data_examples", "ellipsoids_seg_symmetrised.nii.gz")
# ---- LABELS LIST ----
@@ -60,13 +61,13 @@
pfi_output_segmentation,
results_folder_path=pfo_output_folder,
list_labels_transformed=labels_sym_right,
- coord='z',
+ coord="z",
reuse_registration=False)
# --- SEE RESULTS ----
-opener1 = 'itksnap -g {} -s {}'.format(pfi_input_anatomy, pfi_input_segmentation)
-opener2 = 'itksnap -g {} -s {}'.format(pfi_input_anatomy, pfi_output_segmentation)
+opener1 = f"itksnap -g {pfi_input_anatomy} -s {pfi_input_segmentation}"
+opener2 = f"itksnap -g {pfi_input_anatomy} -s {pfi_output_segmentation}"
os.system(opener1)
os.system(opener2)
diff --git a/examples/simple_label_fusion.py b/examples/simple_label_fusion.py
index 9bc29ef..7228cdb 100644
--- a/examples/simple_label_fusion.py
+++ b/examples/simple_label_fusion.py
@@ -4,73 +4,73 @@
from nilabels.agents.agents_controller import AgentsController as NiL
from nilabels.definitions import root_dir
-if __name__ == '__main__':
+if __name__ == "__main__":
- run_steps = {'Generate results folders': False,
- 'Register and propagate': False,
- 'Fuse seg_LabFusion': True}
+ run_steps = {"Generate results folders": False,
+ "Register and propagate": False,
+ "Fuse seg_LabFusion": True}
- pfo_input_dataset = jph(root_dir, 'data_examples', 'ellipsoids_family')
- pfo_results_propagation = jph(root_dir, 'data_output', 'results_propagation')
- pfo_results_label_fusion = jph(root_dir, 'data_output', 'results_label_fusion')
+ pfo_input_dataset = jph(root_dir, "data_examples", "ellipsoids_family")
+ pfo_results_propagation = jph(root_dir, "data_output", "results_propagation")
+ pfo_results_label_fusion = jph(root_dir, "data_output", "results_label_fusion")
- if run_steps['Generate results folders']:
- cmd0 = 'mkdir -p {0}'.format(pfo_results_propagation)
- cmd1 = 'mkdir -p {0}'.format(pfo_results_label_fusion)
+ if run_steps["Generate results folders"]:
+ cmd0 = f"mkdir -p {pfo_results_propagation}"
+ cmd1 = f"mkdir -p {pfo_results_label_fusion}"
os.system(cmd0)
os.system(cmd1)
- if run_steps['Register and propagate']:
- fin_target = 'target.nii.gz'
- fin_target_seg = 'target_seg.nii.gz'
+ if run_steps["Register and propagate"]:
+ fin_target = "target.nii.gz"
+ fin_target_seg = "target_seg.nii.gz"
for k in range(1, 11):
# affine registration
- cmd_aff = 'reg_aladin -ref {0} -flo {1} -aff {2} -res {3}'.format(jph(pfo_input_dataset, 'target.nii.gz'),
- jph(pfo_input_dataset, 'ellipsoid' + str(k) + '.nii.gz'),
- jph(pfo_results_propagation, 'aff_ellipsoid' + str(k) + '_on_target.txt'),
- jph(pfo_results_propagation, 'aff_ellipsoid' + str(k) + '_on_target.nii.gz'))
+ cmd_aff = "reg_aladin -ref {0} -flo {1} -aff {2} -res {3}".format(jph(pfo_input_dataset, "target.nii.gz"),
+ jph(pfo_input_dataset, "ellipsoid" + str(k) + ".nii.gz"),
+ jph(pfo_results_propagation, "aff_ellipsoid" + str(k) + "_on_target.txt"),
+ jph(pfo_results_propagation, "aff_ellipsoid" + str(k) + "_on_target.nii.gz"))
os.system(cmd_aff)
# non-rigid registration
- cmd_nrig = 'reg_f3d -ref {0} -flo {1} -cpp {2} -res {3}'.format(jph(pfo_input_dataset, 'target.nii.gz'),
- jph(pfo_results_propagation, 'aff_ellipsoid' + str(k) + '_on_target.nii.gz'),
- jph(pfo_results_propagation, 'cpp_ellipsoid' + str(k) + '_on_target.nii.gz'),
- jph(pfo_results_propagation, 'nrig_ellipsoid' + str(k) + '_on_target.nii.gz'))
+ cmd_nrig = "reg_f3d -ref {0} -flo {1} -cpp {2} -res {3}".format(jph(pfo_input_dataset, "target.nii.gz"),
+ jph(pfo_results_propagation, "aff_ellipsoid" + str(k) + "_on_target.nii.gz"),
+ jph(pfo_results_propagation, "cpp_ellipsoid" + str(k) + "_on_target.nii.gz"),
+ jph(pfo_results_propagation, "nrig_ellipsoid" + str(k) + "_on_target.nii.gz"))
os.system(cmd_nrig)
# affine propagation
- cmd_resample_aff = 'reg_resample -ref {0} -flo {1} -trans {2} -res {3} -inter 0'.format(
- jph(pfo_input_dataset, 'target.nii.gz'),
- jph(pfo_input_dataset, 'ellipsoid' + str(k) + '_seg.nii.gz'),
- jph(pfo_results_propagation, 'aff_ellipsoid' + str(k) + '_on_target.txt'),
- jph(pfo_results_propagation, 'seg_aff_ellipsoid' + str(k) + '_on_target.nii.gz')
+ cmd_resample_aff = "reg_resample -ref {0} -flo {1} -trans {2} -res {3} -inter 0".format(
+ jph(pfo_input_dataset, "target.nii.gz"),
+ jph(pfo_input_dataset, "ellipsoid" + str(k) + "_seg.nii.gz"),
+ jph(pfo_results_propagation, "aff_ellipsoid" + str(k) + "_on_target.txt"),
+ jph(pfo_results_propagation, "seg_aff_ellipsoid" + str(k) + "_on_target.nii.gz"),
)
os.system(cmd_resample_aff)
# non-rigid propagation
- cmd_resample_nrig = 'reg_resample -ref {0} -flo {1} -trans {2} -res {3} -inter 0'.format(
- jph(pfo_input_dataset, 'target.nii.gz'),
- jph(pfo_results_propagation, 'seg_aff_ellipsoid' + str(k) + '_on_target.nii.gz'),
- jph(pfo_results_propagation, 'cpp_ellipsoid' + str(k) + '_on_target.nii.gz'),
- jph(pfo_results_propagation, 'seg_nrig_ellipsoid' + str(k) + '_on_target.nii.gz')
+ cmd_resample_nrig = "reg_resample -ref {0} -flo {1} -trans {2} -res {3} -inter 0".format(
+ jph(pfo_input_dataset, "target.nii.gz"),
+ jph(pfo_results_propagation, "seg_aff_ellipsoid" + str(k) + "_on_target.nii.gz"),
+ jph(pfo_results_propagation, "cpp_ellipsoid" + str(k) + "_on_target.nii.gz"),
+ jph(pfo_results_propagation, "seg_nrig_ellipsoid" + str(k) + "_on_target.nii.gz"),
)
os.system(cmd_resample_nrig)
- if run_steps['Fuse seg_LabFusion']:
+ if run_steps["Fuse seg_LabFusion"]:
# instantiate a label manager:
- lm = NiL(jph(root_dir, 'data_examples', 'ellipsoids_family'), jph(root_dir, 'data_output'))
+ lm = NiL(jph(root_dir, "data_examples", "ellipsoids_family"), jph(root_dir, "data_output"))
# With majority voting
- options_seg = '-MV'
+ options_seg = "-MV"
- list_pfi_segmentations = [jph(pfo_results_propagation, 'seg_nrig_ellipsoid' + str(k) + '_on_target.nii.gz') for k in range(1, 11)]
- list_pfi_warped = [jph(pfo_results_propagation, 'seg_nrig_ellipsoid' + str(k) + '_on_target.nii.gz') for k in range(1, 11)]
+ list_pfi_segmentations = [jph(pfo_results_propagation, "seg_nrig_ellipsoid" + str(k) + "_on_target.nii.gz") for k in range(1, 11)]
+ list_pfi_warped = [jph(pfo_results_propagation, "seg_nrig_ellipsoid" + str(k) + "_on_target.nii.gz") for k in range(1, 11)]
- lm.fuse.create_stack_for_labels_fusion('target.nii.gz', jph('results_label_fusion', 'output' + options_seg + '.nii.gz'),
+ lm.fuse.create_stack_for_labels_fusion("target.nii.gz", jph("results_label_fusion", "output" + options_seg + ".nii.gz"),
list_pfi_segmentations, options=options_seg)
# If something more sophisticated needs to be done, it returns the paths to the stacks of images:
- options_seg = '_test2'
- list_paths = lm.fuse.create_stack_for_labels_fusion('target.nii.gz', jph('results_label_fusion', 'output' + options_seg + '.nii.gz'),
+ options_seg = "_test2"
+ list_paths = lm.fuse.create_stack_for_labels_fusion("target.nii.gz", jph("results_label_fusion", "output" + options_seg + ".nii.gz"),
list_pfi_segmentations, list_pfi_warped, options=options_seg, prepare_data_only=True)
print(list_paths)
diff --git a/examples/simple_relabelling_example.py b/examples/simple_relabelling_example.py
index 05ebde9..c75748d 100644
--- a/examples/simple_relabelling_example.py
+++ b/examples/simple_relabelling_example.py
@@ -4,18 +4,18 @@
from nilabels.agents.agents_controller import AgentsController as NiL
from nilabels.definitions import root_dir
-if __name__ == '__main__':
+if __name__ == "__main__":
# generate output folder for examples:
- cmd = 'mkdir -p {}'.format(jph(root_dir, 'data_output'))
+ cmd = "mkdir -p {}".format(jph(root_dir, "data_output"))
os.system(cmd)
# instantiate a label manager:
- lt = NiL(jph(root_dir, 'data_examples'), jph(root_dir, 'data_output'))
+ lt = NiL(jph(root_dir, "data_examples"), jph(root_dir, "data_output"))
# data:
- fin_punt_seg_original = 'mes_seg.nii.gz'
- fin_punt_seg_new = 'mes_seg_relabelled.nii.gz'
+ fin_punt_seg_original = "mes_seg.nii.gz"
+ fin_punt_seg_new = "mes_seg_relabelled.nii.gz"
list_old_labels = [1, 2, 3, 4, 5, 6]
list_new_labels = [2, 3, 4, 5, 6, 7]
@@ -25,12 +25,12 @@
list_old_labels, list_new_labels)
# figure before:
- cmd = 'itksnap -g {0} -s {1}'.format(
- jph(root_dir, 'data_examples', 'mes.nii.gz'),
- jph(root_dir, 'data_examples', fin_punt_seg_original))
+ cmd = "itksnap -g {0} -s {1}".format(
+ jph(root_dir, "data_examples", "mes.nii.gz"),
+ jph(root_dir, "data_examples", fin_punt_seg_original))
os.system(cmd)
# figure after
- cmd = 'itksnap -g {0} -s {1}'.format(
- jph(root_dir, 'data_examples', 'mes.nii.gz'),
- jph(root_dir, 'data_output', fin_punt_seg_new))
+ cmd = "itksnap -g {0} -s {1}".format(
+ jph(root_dir, "data_examples", "mes.nii.gz"),
+ jph(root_dir, "data_output", fin_punt_seg_new))
os.system(cmd)
diff --git a/nilabels/__init__.py b/nilabels/__init__.py
index a77a769..25dfe78 100644
--- a/nilabels/__init__.py
+++ b/nilabels/__init__.py
@@ -1 +1 @@
-from .agents.agents_controller import AgentsController as App
+from .agents.agents_controller import AgentsController as App # noqa: F401
diff --git a/nilabels/agents/agents_controller.py b/nilabels/agents/agents_controller.py
index eaf838a..61973f2 100644
--- a/nilabels/agents/agents_controller.py
+++ b/nilabels/agents/agents_controller.py
@@ -1,30 +1,27 @@
import os
-from nilabels.agents.labels_manipulator import LabelsManipulator
-from nilabels.agents.shape_manipulator import ShapeManipulator
-from nilabels.agents.intensities_manipulator import IntensitiesManipulator
-from nilabels.agents.measurer import LabelsMeasure
-from nilabels.agents.fuser import LabelsFuser
-from nilabels.agents.symmetrizer import SegmentationSymmetrizer
from nilabels.agents.checker import LabelsChecker
+from nilabels.agents.fuser import LabelsFuser
from nilabels.agents.header_controller import HeaderController
-from nilabels.agents.segmenter import LabelsSegmenter
+from nilabels.agents.intensities_manipulator import IntensitiesManipulator
+from nilabels.agents.labels_manipulator import LabelsManipulator
from nilabels.agents.math import Math
+from nilabels.agents.measurer import LabelsMeasure
+from nilabels.agents.segmenter import LabelsSegmenter
+from nilabels.agents.shape_manipulator import ShapeManipulator
+from nilabels.agents.symmetrizer import SegmentationSymmetrizer
-class AgentsController(object):
-
+class AgentsController:
def __init__(self, input_data_folder=None, output_data_folder=None):
- """
- Main agent-class that access all the tools methods given input paths through agents.
+ """Main agent-class that access all the tools methods given input paths through agents.
Each agent has a different semantic task, so that recover the tool to be applied to a path will be easier.
The nomenclature and functionality of each tool is decoupled from the agents that are using them.
> The input of a method in tools is typically a nibabel image.
> The input of a method in the agents it typically a path to a nifti image.
"""
- if input_data_folder is not None:
- if not os.path.isdir(input_data_folder):
- raise IOError('Selected path must be None or must point to an existing folder.')
+ if input_data_folder is not None and not os.path.isdir(input_data_folder):
+ raise OSError("Selected path must be None or must point to an existing folder.")
self._pfo_in = input_data_folder
@@ -43,13 +40,13 @@ def set_output_data_folder(self, output_data_folder):
self._set_attribute_agents()
def _set_attribute_agents(self):
- self.manipulate_labels = LabelsManipulator(self._pfo_in, self._pfo_out)
+ self.manipulate_labels = LabelsManipulator(self._pfo_in, self._pfo_out)
self.manipulate_intensities = IntensitiesManipulator(self._pfo_in, self._pfo_out)
- self.manipulate_shape = ShapeManipulator(self._pfo_in, self._pfo_out)
- self.measure = LabelsMeasure(self._pfo_in, self._pfo_out)
- self.fuse = LabelsFuser(self._pfo_in, self._pfo_out)
- self.symmetrize = SegmentationSymmetrizer(self._pfo_in, self._pfo_out)
- self.check = LabelsChecker(self._pfo_in, self._pfo_out)
- self.header = HeaderController(self._pfo_in, self._pfo_out)
- self.segment = LabelsSegmenter(self._pfo_in, self._pfo_out)
- self.math = Math(self._pfo_in, self._pfo_out)
+ self.manipulate_shape = ShapeManipulator(self._pfo_in, self._pfo_out)
+ self.measure = LabelsMeasure(self._pfo_in, self._pfo_out)
+ self.fuse = LabelsFuser(self._pfo_in, self._pfo_out)
+ self.symmetrize = SegmentationSymmetrizer(self._pfo_in, self._pfo_out)
+ self.check = LabelsChecker(self._pfo_in, self._pfo_out)
+ self.header = HeaderController(self._pfo_in, self._pfo_out)
+ self.segment = LabelsSegmenter(self._pfo_in, self._pfo_out)
+ self.math = Math(self._pfo_in, self._pfo_out)
diff --git a/nilabels/agents/checker.py b/nilabels/agents/checker.py
index af5a6ec..17f8792 100644
--- a/nilabels/agents/checker.py
+++ b/nilabels/agents/checker.py
@@ -7,44 +7,52 @@
from nilabels.tools.detections.check_imperfections import check_missing_labels
-class LabelsChecker(object):
- """
- Facade of the methods in tools, for work with paths to images rather than
+class LabelsChecker:
+ """Facade of the methods in tools, for work with paths to images rather than
with data. Methods under LabelsManagerFuse access label fusion methods.
"""
+
def __init__(self, input_data_folder=None, output_data_folder=None):
self.pfo_in = input_data_folder
self.pfo_out = output_data_folder
- def missing_labels(self, path_input_segmentation, path_input_labels_descriptor, pfi_where_to_save_the_log_file=None):
+ def missing_labels(
+ self,
+ path_input_segmentation,
+ path_input_labels_descriptor,
+ pfi_where_to_save_the_log_file=None,
+ ):
pfi_segm = connect_path_tail_head(self.pfo_in, path_input_segmentation)
- pfi_ld = connect_path_tail_head(self.pfo_in, path_input_labels_descriptor)
+ pfi_ld = connect_path_tail_head(self.pfo_in, path_input_labels_descriptor)
ldm = LabelsDescriptorManager(pfi_ld)
im_se = nib.load(pfi_segm)
- in_descriptor_not_delineated, delineated_not_in_descriptor = \
- check_missing_labels(im_se, ldm, pfi_where_log=pfi_where_to_save_the_log_file)
+ in_descriptor_not_delineated, delineated_not_in_descriptor = check_missing_labels(
+ im_se,
+ ldm,
+ pfi_where_log=pfi_where_to_save_the_log_file,
+ )
return in_descriptor_not_delineated, delineated_not_in_descriptor
def number_connected_components_per_label(self, input_segmentation, where_to_save_the_log_file=None):
pfi_segm = connect_path_tail_head(self.pfo_in, input_segmentation)
im = nib.load(pfi_segm)
- msg = 'Labels check number of connected components for segmentation {} \n\n'.format(pfi_segm)
+ msg = f"Labels check number of connected components for segmentation {pfi_segm} \n\n"
labs_list, cc_list = [], []
- for lab in sorted(list(set(im.get_data().flat))):
- cc = ndimage.label(im.get_data() == lab)[1]
- msg_l = 'Label {} has {} connected components'.format(lab, cc)
+ for lab in sorted(set(im.get_fdata().flat)):
+ cc = ndimage.label(im.get_fdata() == lab)[1]
+ msg_l = f"Label {lab} has {cc} connected components"
print(msg_l)
- msg += msg_l + '\n'
+ msg += msg_l + "\n"
labs_list.append(lab)
- cc_list. append(cc)
+ cc_list.append(cc)
if where_to_save_the_log_file is not None:
- f = open(where_to_save_the_log_file, 'w')
+ f = open(where_to_save_the_log_file, "w")
f.write(msg)
f.close()
return labs_list, cc_list
def compare_two_nifti_images(self, path_first_image, path_second_image):
- pfi_first = connect_path_tail_head(self.pfo_in, path_first_image)
+ pfi_first = connect_path_tail_head(self.pfo_in, path_first_image)
pfi_second = connect_path_tail_head(self.pfo_in, path_second_image)
im1 = nib.load(pfi_first)
im2 = nib.load(pfi_second)
diff --git a/nilabels/agents/fuser.py b/nilabels/agents/fuser.py
index cd2cc1e..940bab1 100644
--- a/nilabels/agents/fuser.py
+++ b/nilabels/agents/fuser.py
@@ -1,24 +1,30 @@
-import os
-import numpy as np
import nibabel as nib
+import numpy as np
-from nilabels.tools.aux_methods.utils_path import connect_path_tail_head
from nilabels.tools.aux_methods.utils_nib import set_new_data
+from nilabels.tools.aux_methods.utils_path import connect_path_tail_head
-class LabelsFuser(object):
- """
- Facade of the methods in tools, for work with paths to images rather than
+class LabelsFuser:
+ """Facade of the methods in tools, for work with paths to images rather than
with data.
"""
+
def __init__(self, input_data_folder=None, output_data_folder=None):
self.pfo_in = input_data_folder
self.pfo_out = output_data_folder
- def create_stack_for_labels_fusion(self, pfi_target, pfi_result, list_pfi_segmentations, list_pfi_warped=None,
- seg_output_name='res_4d_seg', warp_output_name='res_4d_warp', output_tag=''):
- """
- Stack and fuse anatomical images and segmentations in a single command.
+ def create_stack_for_labels_fusion(
+ self,
+ pfi_target,
+ pfi_result,
+ list_pfi_segmentations,
+ list_pfi_warped=None,
+ seg_output_name="res_4d_seg",
+ warp_output_name="res_4d_warp",
+ output_tag="",
+ ):
+ """Stack and fuse anatomical images and segmentations in a single command.
:param pfi_target: path to file to the target of the segmentation
:param pfi_result: path to file where to store the result.
:param list_pfi_segmentations: list of the segmentations to fuse
@@ -36,11 +42,11 @@ def create_stack_for_labels_fusion(self, pfi_target, pfi_result, list_pfi_segmen
# save 4d segmentations in stack_seg
list_pfi_segmentations = [connect_path_tail_head(self.pfo_in, j) for j in list_pfi_segmentations]
#
- list_stack_seg = [nib.load(pfi).get_data() for pfi in list_pfi_segmentations]
+ list_stack_seg = [nib.load(pfi).get_fdata() for pfi in list_pfi_segmentations]
stack_seg = np.stack(list_stack_seg, axis=3)
del list_stack_seg
im_4d_seg = set_new_data(nib.load(list_pfi_segmentations[0]), stack_seg)
- pfi_4d_seg = connect_path_tail_head(self.pfo_out, '{0}_{1}.nii.gz'.format(seg_output_name, output_tag))
+ pfi_4d_seg = connect_path_tail_head(self.pfo_out, f"{seg_output_name}_{output_tag}.nii.gz")
nib.save(im_4d_seg, pfi_4d_seg)
# save 4d warped if available
@@ -49,12 +55,11 @@ def create_stack_for_labels_fusion(self, pfi_target, pfi_result, list_pfi_segmen
else:
list_pfi_warped = [connect_path_tail_head(self.pfo_in, j) for j in list_pfi_warped]
#
- list_stack_warp = [nib.load(pfi).get_data() for pfi in list_pfi_warped]
+ list_stack_warp = [nib.load(pfi).get_fdata() for pfi in list_pfi_warped]
stack_warp = np.stack(list_stack_warp, axis=3)
del list_stack_warp
im_4d_warp = set_new_data(nib.load(list_pfi_warped[0]), stack_warp)
- pfi_4d_warp = connect_path_tail_head(self.pfo_out, '{0}_{1}.nii.gz'.format(warp_output_name, output_tag))
+ pfi_4d_warp = connect_path_tail_head(self.pfo_out, f"{warp_output_name}_{output_tag}.nii.gz")
nib.save(im_4d_warp, pfi_4d_warp)
-
return pfi_target, pfi_result, pfi_4d_seg, pfi_4d_warp
diff --git a/nilabels/agents/header_controller.py b/nilabels/agents/header_controller.py
index cf92bec..c2972ce 100644
--- a/nilabels/agents/header_controller.py
+++ b/nilabels/agents/header_controller.py
@@ -1,15 +1,17 @@
import nibabel as nib
import numpy as np
+from nilabels.tools.aux_methods.utils_nib import (
+ modify_affine_transformation,
+ modify_image_data_type,
+ replace_translational_part,
+)
+from nilabels.tools.aux_methods.utils_path import connect_path_tail_head, get_pfi_in_pfi_out
from nilabels.tools.aux_methods.utils_rotations import get_small_orthogonal_rotation
-from nilabels.tools.aux_methods.utils_path import get_pfi_in_pfi_out, connect_path_tail_head
-from nilabels.tools.aux_methods.utils_nib import modify_image_data_type, \
- modify_affine_transformation, replace_translational_part
-class HeaderController(object):
- """
- Facade of the methods in tools. symmetrizer, for work with paths to images rather than
+class HeaderController:
+ """Facade of the methods in tools. symmetrizer, for work with paths to images rather than
with data. Methods under LabelsManagerManipulate are taking in general
one or more input manipulate them according to some rule and save the
output in the output_data_folder or in the specified paths.
@@ -20,8 +22,7 @@ def __init__(self, input_data_folder=None, output_data_folder=None):
self.pfo_out = output_data_folder
def modify_image_type(self, filename_in, filename_out, new_dtype, update_description=None, verbose=1):
- """
- Change data type and optionally update the nifti field descriptor.
+ """Change data type and optionally update the nifti field descriptor.
:param filename_in: path to filename input
:param filename_out: path to filename output
:param new_dtype: numpy data type compatible input
@@ -29,7 +30,6 @@ def modify_image_type(self, filename_in, filename_out, new_dtype, update_descrip
:param verbose:
:return: image with new dtype and descriptor updated.
"""
-
pfi_in, pfi_out = get_pfi_in_pfi_out(filename_in, filename_out, self.pfo_in, self.pfo_out)
im = nib.load(pfi_in)
@@ -37,9 +37,8 @@ def modify_image_type(self, filename_in, filename_out, new_dtype, update_descrip
nib.save(new_im, pfi_out)
def modify_affine(self, filename_in, affine_in, filename_out, q_form=True, s_form=True,
- multiplication_side='left'):
- """
- Modify the affine transformation by substitution or by left or right multiplication
+ multiplication_side="left"):
+ """Modify the affine transformation by substitution or by left or right multiplication
:param filename_in: path to filename input
:param affine_in: path to affine matrix input, or nd.array or .npy array
:param filename_out: path to filename output
@@ -55,7 +54,7 @@ def modify_affine(self, filename_in, affine_in, filename_out, q_form=True, s_for
if isinstance(affine_in, str):
- if affine_in.endswith('.txt'):
+ if affine_in.endswith(".txt"):
aff = np.loadtxt(connect_path_tail_head(self.pfo_in, affine_in))
else:
aff = np.load(connect_path_tail_head(self.pfo_in, affine_in))
@@ -63,26 +62,23 @@ def modify_affine(self, filename_in, affine_in, filename_out, q_form=True, s_for
elif isinstance(affine_in, np.ndarray):
aff = affine_in
else:
- raise IOError('parameter affine_in can be path to an affine matrix .txt or .npy or the numpy array'
- 'corresponding to the affine transformation.')
+ raise OSError("parameter affine_in can be path to an affine matrix .txt or .npy or the numpy array"
+ "corresponding to the affine transformation.")
im = nib.load(pfi_in)
new_im = modify_affine_transformation(im, aff, q_form=q_form, s_form=s_form,
multiplication_side=multiplication_side)
nib.save(new_im, pfi_out)
- def apply_small_rotation(self, filename_in, filename_out, angle=np.pi/6, principal_axis='pitch',
+ def apply_small_rotation(self, filename_in, filename_out, angle=np.pi/6, principal_axis="pitch",
respect_to_centre=True):
- """
-
- :param filename_in: path to filename input
+ """:param filename_in: path to filename input
:param filename_out: path to filename output
:param angle: rotation angle in radiants
:param principal_axis: 'yaw', 'pitch' or 'roll'
:param respect_to_centre: by default is True. If False, respect to the origin.
:return:
"""
-
if isinstance(angle, list):
assert isinstance(principal_axis, list)
assert len(principal_axis) == len(angle)
@@ -92,7 +88,7 @@ def apply_small_rotation(self, filename_in, filename_out, angle=np.pi/6, princip
rot = rot.dot(aff)
else:
rot = get_small_orthogonal_rotation(theta=angle, principal_axis=principal_axis)
-
+
pfi_in, pfi_out = get_pfi_in_pfi_out(filename_in, filename_out, self.pfo_in, self.pfo_out)
im = nib.load(pfi_in)
@@ -113,13 +109,12 @@ def apply_small_rotation(self, filename_in, filename_out, angle=np.pi/6, princip
new_aff[:3, :3] = rot[:3, :3].dot(new_aff[:3, :3])
new_im = modify_affine_transformation(im_input=im, new_aff=new_aff, q_form=True, s_form=True,
- multiplication_side='replace')
+ multiplication_side="replace")
nib.save(new_im, pfi_out)
def modify_translational_part(self, filename_in, filename_out, new_translation):
- """
- :param filename_in: path to filename input
+ """:param filename_in: path to filename input
:param filename_out: path to filename output
:param new_translation: translation that will replace the existing one.
:return:
@@ -129,7 +124,7 @@ def modify_translational_part(self, filename_in, filename_out, new_translation):
if isinstance(new_translation, str):
- if new_translation.endswith('.txt'):
+ if new_translation.endswith(".txt"):
tr = np.loadtxt(connect_path_tail_head(self.pfo_in, new_translation))
else:
tr = np.load(connect_path_tail_head(self.pfo_in, new_translation))
@@ -139,8 +134,8 @@ def modify_translational_part(self, filename_in, filename_out, new_translation):
elif isinstance(new_translation, list):
tr = np.array(new_translation)
else:
- raise IOError('parameter new_translation can be path to an affine matrix .txt or .npy or the numpy array'
- 'corresponding to the new intended translational part.')
+ raise OSError("parameter new_translation can be path to an affine matrix .txt or .npy or the numpy array"
+ "corresponding to the new intended translational part.")
new_im = replace_translational_part(im, tr)
nib.save(new_im, pfi_out)
diff --git a/nilabels/agents/intensities_manipulator.py b/nilabels/agents/intensities_manipulator.py
index aeba447..ce1b316 100644
--- a/nilabels/agents/intensities_manipulator.py
+++ b/nilabels/agents/intensities_manipulator.py
@@ -1,30 +1,28 @@
import nibabel as nib
import numpy as np
-from nilabels.tools.aux_methods.utils_path import get_pfi_in_pfi_out, connect_path_tail_head
from nilabels.tools.aux_methods.utils import labels_query
-from nilabels.tools.image_colors_manipulations.normaliser import normalise_below_labels
+from nilabels.tools.aux_methods.utils_path import connect_path_tail_head, get_pfi_in_pfi_out
from nilabels.tools.detections.contours import contour_from_segmentation
-from nilabels.tools.image_shape_manipulations.merger import grafting
from nilabels.tools.image_colors_manipulations.cutter import apply_a_mask_nib
+from nilabels.tools.image_colors_manipulations.normaliser import normalise_below_labels
+from nilabels.tools.image_shape_manipulations.merger import grafting
-class IntensitiesManipulator(object):
- """
- Facade of the methods in tools, for work with paths to images rather than
+class IntensitiesManipulator:
+ """Facade of the methods in tools, for work with paths to images rather than
with data. Methods under LabelsManagerManipulate are taking in general
one or more input manipulate them according to some rule and save the
output in the output_data_folder or in the specified paths.
"""
+
def __init__(self, input_data_folder=None, output_data_folder=None, path_label_descriptor=None):
self.pfo_in = input_data_folder
self.pfo_out = output_data_folder
self.path_label_descriptor = path_label_descriptor
def normalise_below_label(self, filename_image_in, filename_image_out, filename_segm, labels, stats=np.median):
- """
-
- :param filename_image_in: path to image input
+ """:param filename_image_in: path to image input
:param filename_image_out: path to image output
:param filename_segm: path to segmentation
:param labels: list of labels below which the voxels are collected
@@ -38,24 +36,31 @@ def normalise_below_label(self, filename_image_in, filename_image_out, filename_
im_input = nib.load(pfi_in)
im_segm = nib.load(pfi_segm)
- labels_list, labels_names = labels_query(labels, im_segm.get_data())
+ labels_list, labels_names = labels_query(labels, im_segm.get_fdata())
im_out = normalise_below_labels(im_input, labels_list, labels, stats=stats, exclude_first_label=True)
nib.save(im_out, pfi_out)
- def get_contour_from_segmentation(self, filename_input_segmentation, filename_output_contour,
- omit_axis=None, verbose=0):
- """
- Get the contour from a segmentation.
+ def get_contour_from_segmentation(
+ self,
+ filename_input_segmentation,
+ filename_output_contour,
+ omit_axis=None,
+ verbose=0,
+ ):
+ """Get the contour from a segmentation.
:param filename_input_segmentation: input segmentation
:param filename_output_contour: output contour
:param omit_axis: meant to avoid "walls" in the output segmentation
:param verbose:
:return:
"""
-
- pfi_in, pfi_out = get_pfi_in_pfi_out(filename_input_segmentation, filename_output_contour,
- self.pfo_in, self.pfo_out)
+ pfi_in, pfi_out = get_pfi_in_pfi_out(
+ filename_input_segmentation,
+ filename_output_contour,
+ self.pfo_in,
+ self.pfo_out,
+ )
im_segm = nib.load(pfi_in)
im_contour = contour_from_segmentation(im_segm, omit_axis=omit_axis, verbose=verbose)
@@ -63,8 +68,7 @@ def get_contour_from_segmentation(self, filename_input_segmentation, filename_ou
nib.save(im_contour, filename_output_contour)
def get_grafting(self, pfi_input_hosting_mould, pfi_input_patch, pfi_output_grafted, pfi_input_patch_mask=None):
- """
- :param pfi_input_hosting_mould: base image where the grafting should happen
+ """:param pfi_input_hosting_mould: base image where the grafting should happen
:param pfi_input_patch: patch to be grafted in the hosting_mould
:param pfi_output_grafted: output image with the grafting
:param pfi_input_patch_mask: optional additional mask, where the grafting will take place.
@@ -74,12 +78,12 @@ def get_grafting(self, pfi_input_hosting_mould, pfi_input_patch, pfi_output_graf
pfi_patch = connect_path_tail_head(self.pfo_in, pfi_input_patch)
im_hosting = nib.load(pfi_hosting)
- im_patch = nib.load(pfi_patch)
- im_mask = None
+ im_patch = nib.load(pfi_patch)
+ im_mask = None
if pfi_input_patch_mask is not None:
pfi_mask = connect_path_tail_head(self.pfo_in, pfi_input_patch_mask)
- im_mask = nib.load(pfi_mask)
+ im_mask = nib.load(pfi_mask)
im_grafted = grafting(im_hosting, im_patch, im_patch_mask=im_mask)
@@ -87,8 +91,7 @@ def get_grafting(self, pfi_input_hosting_mould, pfi_input_patch, pfi_output_graf
nib.save(im_grafted, pfi_output)
def crop_outside_mask(self, filename_input_image, filename_mask, filename_output_image_masked):
- """
- Set to zero all the values outside the mask.
+ """Set to zero all the values outside the mask.
Adaptative - if the mask is 3D and the image is 4D, will create a temporary mask,
generate the stack of masks, and apply the stacks to the image.
:param filename_input_image: path to file 3d x T image
@@ -96,8 +99,12 @@ def crop_outside_mask(self, filename_input_image, filename_mask, filename_output
:param filename_output_image_masked: apply the mask to each time point T in the fourth dimension if any.
:return: None, it saves the output in pfi_output.
"""
- pfi_in, pfi_out = get_pfi_in_pfi_out(filename_input_image, filename_output_image_masked,
- self.pfo_in, self.pfo_out)
+ pfi_in, pfi_out = get_pfi_in_pfi_out(
+ filename_input_image,
+ filename_output_image_masked,
+ self.pfo_in,
+ self.pfo_out,
+ )
pfi_mask = connect_path_tail_head(self.pfo_in, filename_mask)
diff --git a/nilabels/agents/labels_manipulator.py b/nilabels/agents/labels_manipulator.py
index 3833bb9..f0aa068 100644
--- a/nilabels/agents/labels_manipulator.py
+++ b/nilabels/agents/labels_manipulator.py
@@ -5,41 +5,53 @@
from nilabels.tools.aux_methods.label_descriptor_manager import LabelsDescriptorManager as LdM
from nilabels.tools.aux_methods.utils_nib import set_new_data
-from nilabels.tools.aux_methods.utils_path import get_pfi_in_pfi_out, connect_path_tail_head
+from nilabels.tools.aux_methods.utils_path import connect_path_tail_head, get_pfi_in_pfi_out
from nilabels.tools.cleaning.labels_cleaner import clean_semgentation
-from nilabels.tools.image_colors_manipulations.relabeller import relabeller, \
- permute_labels, erase_labels, assign_all_other_labels_the_same_value, keep_only_one_label
-from nilabels.tools.image_colors_manipulations.segmentation_to_rgb import \
- get_rgb_image_from_segmentation_and_label_descriptor
+from nilabels.tools.image_colors_manipulations.relabeller import (
+ assign_all_other_labels_the_same_value,
+ erase_labels,
+ keep_only_one_label,
+ permute_labels,
+ relabeller,
+)
+from nilabels.tools.image_colors_manipulations.segmentation_to_rgb import (
+ get_rgb_image_from_segmentation_and_label_descriptor,
+)
from nilabels.tools.image_shape_manipulations.merger import from_segmentations_stack_to_probabilistic_segmentation
-class LabelsManipulator(object):
- """
- Facade of the methods in tools, for work with paths to images rather than
+class LabelsManipulator:
+ """Facade of the methods in tools, for work with paths to images rather than
with data. Methods under LabelsManagerManipulate are taking in general
one or more input manipulate them according to some rule and save the
output in the output_data_folder or in the specified paths.
"""
+
def __init__(self, input_data_folder=None, output_data_folder=None):
- self.pfo_in = input_data_folder
+ self.pfo_in = input_data_folder
self.pfo_out = output_data_folder
self.verbose = True
- self.labels_descriptor_convention = 'itk-snap'
-
- def relabel(self, path_to_input_segmentation, path_to_output_segmentation=None,
- list_old_labels=(), list_new_labels=(), path_to_input_labels_descriptor=None,
- path_to_output_labels_descriptor=None):
- """
- Masks of :func:`labels_manager.tools.manipulations.relabeller.relabeller` using filenames
- """
-
- pfi_in, pfi_out = get_pfi_in_pfi_out(path_to_input_segmentation, path_to_output_segmentation, self.pfo_in,
- self.pfo_out)
+ self.labels_descriptor_convention = "itk-snap"
+
+ def relabel(
+ self,
+ path_to_input_segmentation,
+ path_to_output_segmentation=None,
+ list_old_labels=(),
+ list_new_labels=(),
+ path_to_input_labels_descriptor=None,
+ path_to_output_labels_descriptor=None,
+ ):
+ """Masks of :func:`labels_manager.tools.manipulations.relabeller.relabeller` using filenames"""
+ pfi_in, pfi_out = get_pfi_in_pfi_out(
+ path_to_input_segmentation,
+ path_to_output_segmentation,
+ self.pfo_in,
+ self.pfo_out,
+ )
im_labels = nib.load(pfi_in)
- data_labels = im_labels.get_data()
- data_relabelled = relabeller(data_labels, list_old_labels=list_old_labels,
- list_new_labels=list_new_labels)
+ data_labels = im_labels.get_fdata()
+ data_relabelled = relabeller(data_labels, list_old_labels=list_old_labels, list_new_labels=list_new_labels)
im_relabelled = set_new_data(im_labels, data_relabelled)
@@ -57,17 +69,26 @@ def relabel(self, path_to_input_segmentation, path_to_output_segmentation=None,
pfi_out_ld = connect_path_tail_head(self.pfo_out, path_to_output_labels_descriptor)
ldm_relabelled.save_label_descriptor(pfi_out_ld)
- print('Relabelled image {0} saved in {1}.'.format(pfi_in, pfi_out))
+ print(f"Relabelled image {pfi_in} saved in {pfi_out}.")
return pfi_out
- def permute_labels(self, path_to_input_segmentation, path_to_output_segmentation=None, permutation=(),
- path_to_input_labels_descriptor=None, path_to_output_labels_descriptor=None):
-
- pfi_in, pfi_out = get_pfi_in_pfi_out(path_to_input_segmentation, path_to_output_segmentation,
- self.pfo_in, self.pfo_out)
+ def permute_labels(
+ self,
+ path_to_input_segmentation,
+ path_to_output_segmentation=None,
+ permutation=(),
+ path_to_input_labels_descriptor=None,
+ path_to_output_labels_descriptor=None,
+ ):
+ pfi_in, pfi_out = get_pfi_in_pfi_out(
+ path_to_input_segmentation,
+ path_to_output_segmentation,
+ self.pfo_in,
+ self.pfo_out,
+ )
im_labels = nib.load(pfi_in)
- data_labels = im_labels.get_data()
+ data_labels = im_labels.get_fdata()
data_permuted = permute_labels(data_labels, permutation=permutation)
im_permuted = set_new_data(im_labels, data_permuted)
@@ -85,17 +106,26 @@ def permute_labels(self, path_to_input_segmentation, path_to_output_segmentation
pfi_out_ld = connect_path_tail_head(self.pfo_out, path_to_output_labels_descriptor)
ldm_relabelled.save_label_descriptor(pfi_out_ld)
- print('Permuted labels from image {0} saved in {1}.'.format(pfi_in, pfi_out))
+ print(f"Permuted labels from image {pfi_in} saved in {pfi_out}.")
return pfi_out
- def erase_labels(self, path_to_input_segmentation, path_to_output_segmentation=None, labels_to_erase=(),
- path_to_input_labels_descriptor=None, path_to_output_labels_descriptor=None):
-
- pfi_in, pfi_out = get_pfi_in_pfi_out(path_to_input_segmentation, path_to_output_segmentation,
- self.pfo_in, self.pfo_out)
+ def erase_labels(
+ self,
+ path_to_input_segmentation,
+ path_to_output_segmentation=None,
+ labels_to_erase=(),
+ path_to_input_labels_descriptor=None,
+ path_to_output_labels_descriptor=None,
+ ):
+ pfi_in, pfi_out = get_pfi_in_pfi_out(
+ path_to_input_segmentation,
+ path_to_output_segmentation,
+ self.pfo_in,
+ self.pfo_out,
+ )
im_labels = nib.load(pfi_in)
- data_labels = im_labels.get_data()
+ data_labels = im_labels.get_fdata()
data_erased = erase_labels(data_labels, labels_to_erase=labels_to_erase)
im_erased = set_new_data(im_labels, data_erased)
@@ -113,15 +143,19 @@ def erase_labels(self, path_to_input_segmentation, path_to_output_segmentation=N
pfi_out_ld = connect_path_tail_head(self.pfo_out, path_to_output_labels_descriptor)
ldm_relabelled.save_label_descriptor(pfi_out_ld)
- print('Erased labels from image {0} saved in {1}.'.format(pfi_in, pfi_out))
+ print(f"Erased labels from image {pfi_in} saved in {pfi_out}.")
return pfi_out
- def assign_all_other_labels_the_same_value(self, path_to_input_segmentation, path_to_output_segmentation=None,
- labels_to_keep=(), same_value_label=255,
- path_to_input_labels_descriptor=None,
- path_to_output_labels_descriptor=None):
- """
- :param path_to_input_segmentation:
+ def assign_all_other_labels_the_same_value(
+ self,
+ path_to_input_segmentation,
+ path_to_output_segmentation=None,
+ labels_to_keep=(),
+ same_value_label=255,
+ path_to_input_labels_descriptor=None,
+ path_to_output_labels_descriptor=None,
+ ):
+ """:param path_to_input_segmentation:
:param path_to_output_segmentation:
:param labels_to_keep:
:param same_value_label:
@@ -129,13 +163,20 @@ def assign_all_other_labels_the_same_value(self, path_to_input_segmentation, pat
:param path_to_output_labels_descriptor:
:return:
"""
- pfi_in, pfi_out = get_pfi_in_pfi_out(path_to_input_segmentation, path_to_output_segmentation,
- self.pfo_in, self.pfo_out)
+ pfi_in, pfi_out = get_pfi_in_pfi_out(
+ path_to_input_segmentation,
+ path_to_output_segmentation,
+ self.pfo_in,
+ self.pfo_out,
+ )
im_labels = nib.load(pfi_in)
- data_labels = im_labels.get_data()
- data_reassigned = assign_all_other_labels_the_same_value(data_labels,
- labels_to_keep=labels_to_keep, same_value_label=same_value_label)
+ data_labels = im_labels.get_fdata()
+ data_reassigned = assign_all_other_labels_the_same_value(
+ data_labels,
+ labels_to_keep=labels_to_keep,
+ same_value_label=same_value_label,
+ )
im_reassigned = set_new_data(im_labels, data_reassigned)
nib.save(im_reassigned, pfi_out)
@@ -144,8 +185,10 @@ def assign_all_other_labels_the_same_value(self, path_to_input_segmentation, pat
pfi_in_ld = connect_path_tail_head(self.pfo_in, path_to_input_labels_descriptor)
ldm_input = LdM(pfi_in_ld, labels_descriptor_convention=self.labels_descriptor_convention)
- ldm_relabelled = ldm_input.assign_all_other_labels_the_same_value(labels_to_keep=labels_to_keep,
- same_value_label=same_value_label)
+ ldm_relabelled = ldm_input.assign_all_other_labels_the_same_value(
+ labels_to_keep=labels_to_keep,
+ same_value_label=same_value_label,
+ )
if path_to_output_labels_descriptor is None:
ldm_relabelled.save_label_descriptor(pfi_in_ld)
@@ -153,16 +196,26 @@ def assign_all_other_labels_the_same_value(self, path_to_input_segmentation, pat
pfi_out_ld = connect_path_tail_head(self.pfo_out, path_to_output_labels_descriptor)
ldm_relabelled.save_label_descriptor(pfi_out_ld)
- print('Reassigned labels from image {0} saved in {1}.'.format(pfi_in, pfi_out))
+ print(f"Reassigned labels from image {pfi_in} saved in {pfi_out}.")
return pfi_out
- def keep_one_label(self, path_to_input_segmentation, path_to_output_segmentation=None, label_to_keep=1,
- path_to_input_labels_descriptor=None, path_to_output_labels_descriptor=None):
- pfi_in, pfi_out = get_pfi_in_pfi_out(path_to_input_segmentation, path_to_output_segmentation,
- self.pfo_in, self.pfo_out)
+ def keep_one_label(
+ self,
+ path_to_input_segmentation,
+ path_to_output_segmentation=None,
+ label_to_keep=1,
+ path_to_input_labels_descriptor=None,
+ path_to_output_labels_descriptor=None,
+ ):
+ pfi_in, pfi_out = get_pfi_in_pfi_out(
+ path_to_input_segmentation,
+ path_to_output_segmentation,
+ self.pfo_in,
+ self.pfo_out,
+ )
- im_labels = nib.load(pfi_in)
- data_labels = im_labels.get_data()
+ im_labels = nib.load(pfi_in)
+ data_labels = im_labels.get_fdata()
data_one_label = keep_only_one_label(data_labels, label_to_keep)
im_one_label = set_new_data(im_labels, data_one_label)
@@ -180,7 +233,7 @@ def keep_one_label(self, path_to_input_segmentation, path_to_output_segmentation
pfi_out_ld = connect_path_tail_head(self.pfo_out, path_to_output_labels_descriptor)
ldm_relabelled.save_label_descriptor(pfi_out_ld)
- print('Label {0} kept from image {1} and saved in {2}.'.format(label_to_keep, pfi_in, pfi_out))
+ print(f"Label {label_to_keep} kept from image {pfi_in} and saved in {pfi_out}.")
return pfi_out
def get_probabilistic_prior_from_stack_segmentations(self, path_to_stack_crisp_segm, path_to_fuzzy_output):
@@ -193,7 +246,8 @@ def get_probabilistic_prior_from_stack_segmentations(self, path_to_stack_crisp_s
vec = [np.prod(im_stack_crisp.shape[:3])] + [dims[3]]
array_output = from_segmentations_stack_to_probabilistic_segmentation(
- im_stack_crisp.get_data().reshape(vec).T).T
+ im_stack_crisp.get_fdata().reshape(vec).T,
+ ).T
data_output = array_output.reshape(list(im_stack_crisp.shape[:3]) + [array_output.shape[1]])
@@ -202,10 +256,16 @@ def get_probabilistic_prior_from_stack_segmentations(self, path_to_stack_crisp_s
return pfi_out
- def clean_segmentation(self, path_to_input_segmentation, path_to_output_cleaned_segmentation, labels_to_clean=(),
- verbose=1, special_label=None, force_overwriting=False):
- """
- Clean the segmentation merging the small connected components with the surrounding tissue.
+ def clean_segmentation(
+ self,
+ path_to_input_segmentation,
+ path_to_output_cleaned_segmentation,
+ labels_to_clean=(),
+ verbose=1,
+ special_label=None,
+ force_overwriting=False,
+ ):
+ """Clean the segmentation merging the small connected components with the surrounding tissue.
:param path_to_input_segmentation: path to the input segmentation
:param path_to_output_cleaned_segmentation: path to the output cleaned segmentation. For safety this file can
not exist before calling this method.
@@ -219,31 +279,41 @@ def clean_segmentation(self, path_to_input_segmentation, path_to_output_cleaned_
Note: as a feature (really!) after the holes identification, all labels, and not only the ones indicated in the
input dilate iteratively over the 'holes'.
"""
- pfi_in, pfi_out = get_pfi_in_pfi_out(path_to_input_segmentation, path_to_output_cleaned_segmentation,
- self.pfo_in, self.pfo_out)
- if not force_overwriting:
- if os.path.exists(pfi_out):
- raise IOError('File {} already exists. Cleaner can not overwrite a segmentation'.format(pfi_out))
+ pfi_in, pfi_out = get_pfi_in_pfi_out(
+ path_to_input_segmentation,
+ path_to_output_cleaned_segmentation,
+ self.pfo_in,
+ self.pfo_out,
+ )
+ if not force_overwriting and os.path.exists(pfi_out):
+ msg = f"File {pfi_out} already exists. Cleaner can not overwrite a segmentation"
+ raise OSError(msg)
im_segm = nib.load(pfi_in)
if special_label is None:
- special_label = np.max(im_segm.get_data()) + 1
+ special_label = np.max(im_segm.get_fdata()) + 1
- new_segm_data = clean_semgentation(im_segm.get_data(), labels_to_clean=labels_to_clean,
- label_for_holes=special_label)
+ new_segm_data = clean_semgentation(
+ im_segm.get_fdata(),
+ labels_to_clean=labels_to_clean,
+ label_for_holes=special_label,
+ )
im_segm_cleaned = set_new_data(im_segm, new_segm_data.astype(im_segm.get_data_dtype()))
nib.save(im_segm_cleaned, pfi_out)
if verbose:
- print('Segmentation {} cleaned saved to {}'.format(pfi_in, pfi_out))
-
- def from_segmentation_and_labels_descriptor_to_rgb(self, path_to_input_segmentation,
- path_to_input_txt_labels_descriptor,
- path_to_output_4d_rgb_image,
- invert_black_white=False, dtype_output=np.int32):
- """
- + Masks of :func:`nilabel.tools.image_colors_manipulations.segmentation_to_rgb.
+ print(f"Segmentation {pfi_in} cleaned saved to {pfi_out}")
+
+ def from_segmentation_and_labels_descriptor_to_rgb(
+ self,
+ path_to_input_segmentation,
+ path_to_input_txt_labels_descriptor,
+ path_to_output_4d_rgb_image,
+ invert_black_white=False,
+ dtype_output=np.int32,
+ ):
+ """+ Masks of :func:`nilabel.tools.image_colors_manipulations.segmentation_to_rgb.
get_rgb_image_from_segmentation_and_label_descriptor` using filenames.
From a segmentation and its label descriptro in itk-snap convention or in fsl convention
it creates a corresponding 4d image with the 3 R G B channels in the fourth dimension.
@@ -259,9 +329,12 @@ def from_segmentation_and_labels_descriptor_to_rgb(self, path_to_input_segmentat
im_segm = nib.load(pfi_segm)
ldm = LdM(pfi_ld, labels_descriptor_convention=self.labels_descriptor_convention)
- im_rgb_4d_image = get_rgb_image_from_segmentation_and_label_descriptor(im_segm, ldm,
- invert_black_white=invert_black_white,
- dtype_output=dtype_output)
+ im_rgb_4d_image = get_rgb_image_from_segmentation_and_label_descriptor(
+ im_segm,
+ ldm,
+ invert_black_white=invert_black_white,
+ dtype_output=dtype_output,
+ )
pfi_out = connect_path_tail_head(self.pfo_out, path_to_output_4d_rgb_image)
nib.save(im_rgb_4d_image, pfi_out)
diff --git a/nilabels/agents/math.py b/nilabels/agents/math.py
index 7949067..f9a59b5 100644
--- a/nilabels/agents/math.py
+++ b/nilabels/agents/math.py
@@ -1,17 +1,15 @@
import nibabel as nib
-from nilabels.tools.aux_methods.utils_path import get_pfi_in_pfi_out, connect_path_tail_head
from nilabels.tools.aux_methods.utils_nib import set_new_data
+from nilabels.tools.aux_methods.utils_path import connect_path_tail_head, get_pfi_in_pfi_out
-class Math(object):
- """
- Facade of no external methods. Simple class for quick algebraic manipulations of images with the same grid
- """
+class Math:
+ """Facade of no external methods. Simple class for quick algebraic manipulations of images with the same grid"""
def __init__(self, input_data_folder=None, output_data_folder=None):
- self.pfo_in = input_data_folder
- self.pfo_out = output_data_folder
+ self.pfo_in = input_data_folder
+ self.pfo_out = output_data_folder
def sum(self, path_first_image, path_second_image, path_resulting_image):
pfi_im1, pfi_im2 = get_pfi_in_pfi_out(path_first_image, path_second_image, self.pfo_in, self.pfo_in)
@@ -21,12 +19,12 @@ def sum(self, path_first_image, path_second_image, path_resulting_image):
im2 = nib.load(pfi_im2)
if not im1.shape == im2.shape:
- raise IOError('Input images must have the same dimensions.')
+ raise OSError("Input images must have the same dimensions.")
- im_result = set_new_data(im1, new_data=im1.get_data() + im2.get_data())
+ im_result = set_new_data(im1, new_data=im1.get_fdata() + im2.get_fdata())
nib.save(im_result, pfi_result)
- print('Image sum of {0} {1} saved under {2}.'.format(pfi_im1, pfi_im2, pfi_result))
+ print(f"Image sum of {pfi_im1} {pfi_im2} saved under {pfi_result}.")
return pfi_result
def sub(self, path_first_image, path_second_image, path_resulting_image):
@@ -37,12 +35,12 @@ def sub(self, path_first_image, path_second_image, path_resulting_image):
im2 = nib.load(pfi_im2)
if not im1.shape == im2.shape:
- raise IOError('Input images must have the same dimensions.')
+ raise OSError("Input images must have the same dimensions.")
- im_result = set_new_data(im1, new_data=im1.get_data() - im2.get_data())
+ im_result = set_new_data(im1, new_data=im1.get_fdata() - im2.get_fdata())
nib.save(im_result, pfi_result)
- print('Image difference of {0} {1} saved under {2}.'.format(pfi_im1, pfi_im2, pfi_result))
+ print(f"Image difference of {pfi_im1} {pfi_im2} saved under {pfi_result}.")
return pfi_result
def prod(self, path_first_image, path_second_image, path_resulting_image):
@@ -53,12 +51,12 @@ def prod(self, path_first_image, path_second_image, path_resulting_image):
im2 = nib.load(pfi_im2)
if not im1.shape == im2.shape:
- raise IOError('Input images must have the same dimensions.')
+ raise OSError("Input images must have the same dimensions.")
- im_result = set_new_data(im1, new_data=im1.get_data() * im2.get_data())
+ im_result = set_new_data(im1, new_data=im1.get_fdata() * im2.get_fdata())
nib.save(im_result, pfi_result)
- print('Image product of {0} {1} saved under {2}.'.format(pfi_im1, pfi_im2, pfi_result))
+ print(f"Image product of {pfi_im1} {pfi_im2} saved under {pfi_result}.")
return pfi_result
def scalar_prod(self, scalar, path_image, path_resulting_image):
@@ -66,8 +64,8 @@ def scalar_prod(self, scalar, path_image, path_resulting_image):
pfi_result = connect_path_tail_head(self.pfo_out, path_resulting_image)
im = nib.load(pfi_image)
- im_result = set_new_data(im, new_data=scalar * im.get_data())
+ im_result = set_new_data(im, new_data=scalar * im.get_fdata())
nib.save(im_result, pfi_result)
- print('Image {0} times {1} saved under {2}.'.format(pfi_image, scalar, pfi_result))
+ print(f"Image {pfi_image} times {scalar} saved under {pfi_result}.")
return pfi_result
diff --git a/nilabels/agents/measurer.py b/nilabels/agents/measurer.py
index ab0ff78..5d7f242 100644
--- a/nilabels/agents/measurer.py
+++ b/nilabels/agents/measurer.py
@@ -2,33 +2,37 @@
import nibabel as nib
import numpy as np
-import pandas as pa
+import pandas as pd
-from nilabels.definitions import definition_label
from nilabels.tools.aux_methods.utils import labels_query
from nilabels.tools.aux_methods.utils_path import connect_path_tail_head
-from nilabels.tools.caliber.distances import dice_score, covariance_distance, \
- hausdorff_distance, global_outline_error, global_dice_score, normalised_symmetric_contour_distance
+from nilabels.tools.caliber.distances import (
+ covariance_distance,
+ dice_score,
+ global_dice_score,
+ global_outline_error,
+ hausdorff_distance,
+ normalised_symmetric_contour_distance,
+)
from nilabels.tools.caliber.volumes_and_values import get_values_below_labels_list, get_volumes_per_label
-class LabelsMeasure(object):
+class LabelsMeasure:
"""
Facade of the methods in tools.detections and tools.caliber, where methods are accessed through
paths to images rather than with data.
Methods under LabelsManagerDetect are taking in general one or more input
- and return some feature of the segmentations \n {}.
- """.format(definition_label)
+ and return some feature of the segmentations.
+ """
def __init__(self, input_data_folder=None, output_data_folder=None, return_mm3=True, verbose=0):
- self.pfo_in = input_data_folder
- self.pfo_out = output_data_folder
+ self.pfo_in = input_data_folder
+ self.pfo_out = output_data_folder
self.return_mm3 = return_mm3
- self.verbose = verbose
+ self.verbose = verbose
def volume(self, segmentation_filename, labels=None, tot_volume_prior=None, where_to_save=None):
- """
- :param segmentation_filename: filename of the segmentation S.
+ """:param segmentation_filename: filename of the segmentation S.
:param labels: list of labels, multi-labels (as sublists, e.g. left right will be considered one label)
:param tot_volume_prior: as an intra-cranial volume factor.
:param where_to_save:
@@ -37,9 +41,14 @@ def volume(self, segmentation_filename, labels=None, tot_volume_prior=None, wher
pfi_segm = connect_path_tail_head(self.pfo_in, segmentation_filename)
assert os.path.exists(pfi_segm), pfi_segm
im_segm = nib.load(pfi_segm)
- labels_list, labels_names = labels_query(labels, im_segm.get_data())
- df_volumes_per_label = get_volumes_per_label(im_segm, labels=labels_list, labels_names=labels_names,
- tot_volume_prior=tot_volume_prior, verbose=self.verbose)
+ labels_list, labels_names = labels_query(labels, im_segm.get_fdata())
+ df_volumes_per_label = get_volumes_per_label(
+ im_segm,
+ labels=labels_list,
+ labels_names=labels_names,
+ tot_volume_prior=tot_volume_prior,
+ verbose=self.verbose,
+ )
if self.verbose > 0:
print(df_volumes_per_label)
if where_to_save is not None:
@@ -48,11 +57,10 @@ def volume(self, segmentation_filename, labels=None, tot_volume_prior=None, wher
return df_volumes_per_label
def get_total_volume(self, segmentation_filename):
- return self.volume(segmentation_filename, labels='tot')
+ return self.volume(segmentation_filename, labels="tot")
def values_below_labels(self, segmentation_filename, anatomy_filename, labels=None):
- """
- :param segmentation_filename:
+ """:param segmentation_filename:
:param anatomy_filename:
:param labels:
:return: pandas series with label names and corresponding vectors of labels values
@@ -64,28 +72,33 @@ def values_below_labels(self, segmentation_filename, anatomy_filename, labels=No
im_anat = nib.load(pfi_anat)
im_segm = nib.load(pfi_segm)
- labels_list, labels_names = labels_query(labels, segmentation_array=im_segm.get_data())
+ labels_list, labels_names = labels_query(labels, segmentation_array=im_segm.get_fdata())
labels_values = get_values_below_labels_list(im_segm, im_anat, labels_list)
- return pa.Series(labels_values, index=labels_names)
-
- def dist(self, segm_1_filename, segm_2_filename, labels_list=None, labels_names=None,
- metrics=(dice_score, covariance_distance, hausdorff_distance, normalised_symmetric_contour_distance),
- where_to_save=None):
-
+ return pd.Series(labels_values, index=labels_names)
+
+ def dist(
+ self,
+ segm_1_filename,
+ segm_2_filename,
+ labels_list=None,
+ labels_names=None,
+ metrics=(dice_score, covariance_distance, hausdorff_distance, normalised_symmetric_contour_distance),
+ where_to_save=None,
+ ):
pfi_segm1 = connect_path_tail_head(self.pfo_in, segm_1_filename)
pfi_segm2 = connect_path_tail_head(self.pfo_in, segm_2_filename)
assert os.path.exists(pfi_segm1), pfi_segm1
assert os.path.exists(pfi_segm2), pfi_segm2
if self.verbose > 0:
- print("Distances between segmentations: \n -> {0} \n -> {1} \n...started!".format(pfi_segm1, pfi_segm2))
+ print(f"Distances between segmentations: \n -> {pfi_segm1} \n -> {pfi_segm2} \n...started!")
im_segm1 = nib.load(pfi_segm1)
im_segm2 = nib.load(pfi_segm2)
if labels_list is None:
- labels_list1, labels_names1 = labels_query('all', im_segm1.get_data())
- labels_list2, labels_names2 = labels_query('all', im_segm2.get_data())
- labels_list = list(set(labels_list1) & set(labels_list2))
+ labels_list1, labels_names1 = labels_query("all", im_segm1.get_fdata())
+ labels_list2, labels_names2 = labels_query("all", im_segm2.get_fdata())
+ labels_list = list(set(labels_list1) & set(labels_list2))
labels_list.sort(key=int)
labels_names = None
@@ -96,12 +109,17 @@ def dist(self, segm_1_filename, segm_2_filename, labels_list=None, labels_names=
for d in metrics:
if self.verbose > 0:
- print('{} computation started'.format(d.func_name))
- pa_se = d(im_segm1, im_segm2, labels_list, labels_names, self.return_mm3) # TODO get as function with variable number of arguments
- dict_distances_per_label.update({d.func_name : pa_se})
-
- df_distances_per_label = pa.DataFrame(dict_distances_per_label,
- columns=dict_distances_per_label.keys())
+ print(f"{d.func_name} computation started")
+ pa_se = d(
+ im_segm1,
+ im_segm2,
+ labels_list,
+ labels_names,
+ self.return_mm3,
+ ) # TODO get as function with variable number of arguments
+ dict_distances_per_label.update({d.func_name: pa_se})
+
+ df_distances_per_label = pd.DataFrame(dict_distances_per_label, columns=dict_distances_per_label.keys())
# df_distances_per_label.loc['mean'] = df_distances_per_label.mean()
# df_distances_per_label.loc['std'] = df_distances_per_label.std()
@@ -115,9 +133,13 @@ def dist(self, segm_1_filename, segm_2_filename, labels_list=None, labels_names=
return df_distances_per_label
- def global_dist(self, segm_1_filename, segm_2_filename, where_to_save=None,
- global_metrics=(global_outline_error, global_dice_score)):
-
+ def global_dist(
+ self,
+ segm_1_filename,
+ segm_2_filename,
+ where_to_save=None,
+ global_metrics=(global_outline_error, global_dice_score),
+ ):
pfi_segm1 = connect_path_tail_head(self.pfo_in, segm_1_filename)
pfi_segm2 = connect_path_tail_head(self.pfo_in, segm_2_filename)
@@ -125,14 +147,18 @@ def global_dist(self, segm_1_filename, segm_2_filename, where_to_save=None,
assert os.path.exists(pfi_segm2), pfi_segm2
if self.verbose > 0:
- print("\nGlobal distances between segmentations: \n -> {0} \n -> {1} "
- "\nComputations started!".format(pfi_segm1, pfi_segm2))
+ print(
+ f"\nGlobal distances between segmentations: \n -> {pfi_segm1} \n -> {pfi_segm2} "
+ "\nComputations started!",
+ )
im_segm1 = nib.load(pfi_segm1)
im_segm2 = nib.load(pfi_segm2)
- se_global_distances = pa.Series(np.array([d(im_segm1, im_segm2) for d in global_metrics]),
- index=[d.__name__ for d in global_metrics])
+ se_global_distances = pd.Series(
+ np.array([d(im_segm1, im_segm2) for d in global_metrics]),
+ index=[d.__name__ for d in global_metrics],
+ )
if where_to_save is not None:
where_to_save = connect_path_tail_head(self.pfo_out, where_to_save)
se_global_distances.to_pickle(where_to_save)
@@ -141,13 +167,20 @@ def global_dist(self, segm_1_filename, segm_2_filename, where_to_save=None,
def topology(self):
# TODO: island detections, graph detections, cc detections from detector tools
- print('topology for {} is in the TODO list!'.format(self.__class__))
-
- def groupwise_global_measures_comparisons(self, list_path_A, list_path_B, pfo_where_to_save,
- name_list_path_A=None, name_list_path_B=None,
- list_distances=(global_dice_score, global_outline_error),
- prefix_output='distances_comparison', save_human_readable=True,
- verbose=1):
+ print(f"topology for {self.__class__} is in the TODO list!")
+
+ def groupwise_global_measures_comparisons(
+ self,
+ list_path_A,
+ list_path_B,
+ pfo_where_to_save,
+ name_list_path_A=None,
+ name_list_path_B=None,
+ list_distances=(global_dice_score, global_outline_error),
+ prefix_output="distances_comparison",
+ save_human_readable=True,
+ verbose=1,
+ ):
list_path_A = [connect_path_tail_head(self.pfo_in, ph) for ph in list_path_A]
list_path_B = [connect_path_tail_head(self.pfo_in, ph) for ph in list_path_B]
@@ -157,37 +190,43 @@ def groupwise_global_measures_comparisons(self, list_path_A, list_path_B, pfo_wh
for pfi_im_B in list_path_B:
assert os.path.exists(pfi_im_B), pfi_im_B
if name_list_path_A is None:
- name_list_path_A = [os.path.basename(n).replace('.nii', '').replace('.gz', '') for n in list_path_A]
+ name_list_path_A = [os.path.basename(n).replace(".nii", "").replace(".gz", "") for n in list_path_A]
if name_list_path_B is None:
- name_list_path_B = [os.path.basename(n).replace('.nii', '').replace('.gz', '') for n in list_path_B]
+ name_list_path_B = [os.path.basename(n).replace(".nii", "").replace(".gz", "") for n in list_path_B]
# Initialise one data-frame for each metric/score selected
dictionary_of_measurements = {}
for d in list_distances:
- dictionary_of_measurements.update({d.__name__: pa.DataFrame(np.zeros([len(list_path_A), len(list_path_B)]),
- index=name_list_path_A, columns=name_list_path_B)})
+ dictionary_of_measurements.update(
+ {
+ d.__name__: pd.DataFrame(
+ np.zeros([len(list_path_A), len(list_path_B)]),
+ index=name_list_path_A,
+ columns=name_list_path_B,
+ ),
+ },
+ )
# Fill values in each data-frame
for pfi_im_A, name_A in zip(list_path_A, name_list_path_A):
for pfi_im_B, name_B in zip(list_path_B, name_list_path_B):
-
im_A = nib.load(pfi_im_A)
im_B = nib.load(pfi_im_B)
for d in list_distances:
d_A_B = d(im_A, im_B)
dictionary_of_measurements[d.__name__][name_B][name_A] = d_A_B
if verbose > 0:
- print('{0:<15} {1:<15} \n{2:<15} : {3}'.format(pfi_im_A, pfi_im_B, d.__name__, d_A_B))
+ print(f"{pfi_im_A:<15} {pfi_im_B:<15} \n{d.__name__:<15} : {d_A_B}")
print(dictionary_of_measurements[d.__name__])
# prepare output folder
- os.system('mkdir -p {}'.format(pfo_where_to_save))
+ os.system(f"mkdir -p {pfo_where_to_save}")
for d in list_distances:
# Save each dataframe independently
- pfi_df_global_dice_score = os.path.join(pfo_where_to_save, '{0}_{1}.pickle'.format(prefix_output, d.__name__))
+ pfi_df_global_dice_score = os.path.join(pfo_where_to_save, f"{prefix_output}_{d.__name__}.pickle")
dictionary_of_measurements[d.__name__].to_pickle(pfi_df_global_dice_score)
if save_human_readable:
- pfi_df_global_dice_score_txt = os.path.join(pfo_where_to_save, '{0}_{1}.txt'.format(prefix_output, d.__name__))
- with open(pfi_df_global_dice_score_txt, 'w') as outfile:
+ pfi_df_global_dice_score_txt = os.path.join(pfo_where_to_save, f"{prefix_output}_{d.__name__}.txt")
+ with open(pfi_df_global_dice_score_txt, "w") as outfile:
dictionary_of_measurements[d.__name__].to_string(outfile)
diff --git a/nilabels/agents/segmenter.py b/nilabels/agents/segmenter.py
index c44f2d6..433ba82 100644
--- a/nilabels/agents/segmenter.py
+++ b/nilabels/agents/segmenter.py
@@ -1,25 +1,24 @@
-import numpy as np
import nibabel as nib
+import numpy as np
-from nilabels.tools.aux_methods.utils_path import connect_path_tail_head
from nilabels.tools.aux_methods.utils_nib import set_new_data
-from nilabels.tools.detections.get_segmentation import intensity_segmentation, otsu_threshold, \
- MoG_array
+from nilabels.tools.aux_methods.utils_path import connect_path_tail_head
+from nilabels.tools.detections.get_segmentation import MoG_array, intensity_segmentation, otsu_threshold
-class LabelsSegmenter(object):
- """
- Facade for the simple segmentation methods based on intensities, Otsu thresholding and
+class LabelsSegmenter:
+ """Facade for the simple segmentation methods based on intensities, Otsu thresholding and
mixture of gaussians.
"""
+
def __init__(self, input_data_folder=None, output_data_folder=None):
self.pfo_in = input_data_folder
self.pfo_out = output_data_folder
- def simple_intensities_thresholding(self, path_to_input_image, path_to_output_segmentation, number_of_levels=5,
- output_dtype=np.uint16):
- """
- Simple level intensity-based segmentation.
+ def simple_intensities_thresholding(
+ self, path_to_input_image, path_to_output_segmentation, number_of_levels=5, output_dtype=np.uint16,
+ ):
+ """Simple level intensity-based segmentation.
:param path_to_input_image:
:param path_to_output_segmentation:
:param number_of_levels: number of levels in the output segmentations
@@ -29,15 +28,14 @@ def simple_intensities_thresholding(self, path_to_input_image, path_to_output_se
pfi_input_image = connect_path_tail_head(self.pfo_in, path_to_input_image)
input_im = nib.load(pfi_input_image)
- output_array = intensity_segmentation(input_im.get_data(), num_levels=number_of_levels)
+ output_array = intensity_segmentation(input_im.get_fdata(), num_levels=number_of_levels)
output_im = set_new_data(input_im, output_array, new_dtype=output_dtype)
pfi_output_segm = connect_path_tail_head(self.pfo_out, path_to_output_segmentation)
nib.save(output_im, pfi_output_segm)
- def otsu_thresholding(self, path_to_input_image, path_to_output_segmentation, side='above', return_as_mask=True):
- """
- Binary segmentation with Otsu thresholding parameters from skimage filters.
+ def otsu_thresholding(self, path_to_input_image, path_to_output_segmentation, side="above", return_as_mask=True):
+ """Binary segmentation with Otsu thresholding parameters from skimage filters.
:param path_to_input_image:
:param path_to_output_segmentation:
:param side: can be 'above' or 'below' if the user requires to mask the values above the Otsu threshold or
@@ -49,18 +47,27 @@ def otsu_thresholding(self, path_to_input_image, path_to_output_segmentation, si
pfi_input_image = connect_path_tail_head(self.pfo_in, path_to_input_image)
input_im = nib.load(pfi_input_image)
- output_array = otsu_threshold(input_im.get_data(), side=side, return_as_mask=return_as_mask)
+ output_array = otsu_threshold(input_im.get_fdata(), side=side, return_as_mask=return_as_mask)
output_im = set_new_data(input_im, output_array, new_dtype=output_array.dtype)
pfi_output_segm = connect_path_tail_head(self.pfo_out, path_to_output_segmentation)
nib.save(output_im, pfi_output_segm)
- def mixture_of_gaussians(self, path_to_input_image, path_to_output_segmentation_crisp,
- path_to_output_segmentation_prob, K=None, mask_im=None, pre_process_median_filter=False,
- pre_process_only_interquartile=False, see_histogram=None, reorder_mus=True,
- output_dtype_crisp=np.uint16, output_dtype_prob=np.float32):
- """
- Wrap of MoG_array for nibabel images.
+ def mixture_of_gaussians(
+ self,
+ path_to_input_image,
+ path_to_output_segmentation_crisp,
+ path_to_output_segmentation_prob,
+ K=None,
+ mask_im=None,
+ pre_process_median_filter=False,
+ pre_process_only_interquartile=False,
+ see_histogram=None,
+ reorder_mus=True,
+ output_dtype_crisp=np.uint16,
+ output_dtype_prob=np.float32,
+ ):
+ """Wrap of MoG_array for nibabel images.
-----
:param path_to_input_image: path to input image format to be segmented with a MOG method.
:param path_to_output_segmentation_crisp: path to output crisp segmentation
@@ -76,19 +83,24 @@ def mixture_of_gaussians(self, path_to_input_image, path_to_output_segmentation_
:param output_dtype_prob: data type output probabilistic segmentation (float32)
:return: save crisp and probabilistic segmentation at the specified files after sklearn.mixture.GaussianMixture
"""
-
pfi_input_image = connect_path_tail_head(self.pfo_in, path_to_input_image)
input_im = nib.load(pfi_input_image)
if mask_im is not None:
- mask_array = mask_im.get_data()
+ mask_array = mask_im.get_fdata()
else:
mask_array = None
- ans = MoG_array(input_im.get_data(), K=K, mask_array=mask_array,
- pre_process_median_filter=pre_process_median_filter,
- pre_process_only_interquartile=pre_process_only_interquartile,
- output_gmm_class=False, see_histogram=see_histogram, reorder_mus=reorder_mus)
+ ans = MoG_array(
+ input_im.get_fdata(),
+ K=K,
+ mask_array=mask_array,
+ pre_process_median_filter=pre_process_median_filter,
+ pre_process_only_interquartile=pre_process_only_interquartile,
+ output_gmm_class=False,
+ see_histogram=see_histogram,
+ reorder_mus=reorder_mus,
+ )
crisp, prob = ans[0], ans[1]
diff --git a/nilabels/agents/shape_manipulator.py b/nilabels/agents/shape_manipulator.py
index ba60e20..5f15c24 100644
--- a/nilabels/agents/shape_manipulator.py
+++ b/nilabels/agents/shape_manipulator.py
@@ -1,15 +1,14 @@
import nibabel as nib
import numpy as np
-from nilabels.tools.aux_methods.utils_path import get_pfi_in_pfi_out, connect_path_tail_head
from nilabels.tools.aux_methods.utils_nib import set_new_data
+from nilabels.tools.aux_methods.utils_path import connect_path_tail_head, get_pfi_in_pfi_out
from nilabels.tools.image_colors_manipulations.cutter import cut_4d_volume_with_a_1_slice_mask_nib
from nilabels.tools.image_shape_manipulations.merger import merge_labels_from_4d, stack_images
from nilabels.tools.image_shape_manipulations.splitter import split_labels_to_4d
-class ShapeManipulator(object):
-
+class ShapeManipulator:
def __init__(self, input_data_folder=None, output_data_folder=None):
self.pfo_in = input_data_folder
self.pfo_out = output_data_folder
@@ -18,42 +17,43 @@ def extend_slice_new_dimension(self, pfi_input, pfi_output=None, new_axis=3, num
pfi_in, pfi_out = get_pfi_in_pfi_out(pfi_input, pfi_output, self.pfo_in, self.pfo_out)
im_slice = nib.load(pfi_in)
- data_slice = im_slice.get_data()
+ data_slice = im_slice.get_fdata()
- data_extended = np.stack([data_slice, ] * num_slices, axis=new_axis)
+ data_extended = np.stack([data_slice] * num_slices, axis=new_axis)
im_extended = set_new_data(im_slice, data_extended)
nib.save(im_extended, pfi_out)
- print('Extended image of {0} saved in {1}.'.format(pfi_in, pfi_out))
+ print(f"Extended image of {pfi_in} saved in {pfi_out}.")
return pfi_out
def split_in_4d(self, pfi_input, pfi_output=None, list_labels=None, keep_original_values=True):
pfi_in, pfi_out = get_pfi_in_pfi_out(pfi_input, pfi_output, self.pfo_in, self.pfo_out)
im_labels_3d = nib.load(pfi_in)
- data_labels_3d = im_labels_3d.get_data()
+ data_labels_3d = im_labels_3d.get_fdata()
assert len(data_labels_3d.shape) == 3
if list_labels is None:
list_labels = list(np.sort(list(set(data_labels_3d.flat))))
- data_split_in_4d = split_labels_to_4d(data_labels_3d, list_labels=list_labels,
- keep_original_values=keep_original_values)
+ data_split_in_4d = split_labels_to_4d(
+ data_labels_3d, list_labels=list_labels, keep_original_values=keep_original_values,
+ )
im_split_in_4d = set_new_data(im_labels_3d, data_split_in_4d)
nib.save(im_split_in_4d, pfi_out)
- print('Split labels from image {0} saved in {1}.'.format(pfi_in, pfi_out))
+ print(f"Split labels from image {pfi_in} saved in {pfi_out}.")
return pfi_out
def merge_from_4d(self, pfi_input, pfi_output=None):
pfi_in, pfi_out = get_pfi_in_pfi_out(pfi_input, pfi_output, self.pfo_in, self.pfo_out)
im_labels_4d = nib.load(pfi_in)
- data_labels_4d = im_labels_4d.get_data()
+ data_labels_4d = im_labels_4d.get_fdata()
assert len(data_labels_4d.shape) == 4
data_merged_in_3d = merge_labels_from_4d(data_labels_4d)
im_merged_in_3d = set_new_data(im_labels_4d, data_merged_in_3d)
nib.save(im_merged_in_3d, pfi_out)
- print('Merged labels from 4d image {0} saved in {1}.'.format(pfi_in, pfi_out))
+ print(f"Merged labels from 4d image {pfi_in} saved in {pfi_out}.")
return pfi_out
def cut_4d_volume_with_a_1_slice_mask(self, pfi_input, filename_mask, pfi_output=None):
diff --git a/nilabels/agents/symmetrizer.py b/nilabels/agents/symmetrizer.py
index 8cfed64..b748e9a 100644
--- a/nilabels/agents/symmetrizer.py
+++ b/nilabels/agents/symmetrizer.py
@@ -3,15 +3,14 @@
import nibabel as nib
import numpy as np
-from nilabels.tools.aux_methods.utils_path import get_pfi_in_pfi_out, connect_path_tail_head
from nilabels.tools.aux_methods.utils_nib import set_new_data
-from nilabels.tools.image_colors_manipulations.relabeller import relabeller
+from nilabels.tools.aux_methods.utils_path import connect_path_tail_head, get_pfi_in_pfi_out
from nilabels.tools.aux_methods.utils_rotations import flip_data, symmetrise_data
+from nilabels.tools.image_colors_manipulations.relabeller import relabeller
-class SegmentationSymmetrizer(object):
- """
- Facade of the methods in tools. symmetrizer, for work with paths to images rather than
+class SegmentationSymmetrizer:
+ """Facade of the methods in tools. symmetrizer, for work with paths to images rather than
with data. Methods under LabelsManagerManipulate are taking in general
one or more input manipulate them according to some rule and save the
output in the output_data_folder or in the specified paths.
@@ -21,35 +20,44 @@ def __init__(self, input_data_folder=None, output_data_folder=None):
self.pfo_in = input_data_folder
self.pfo_out = output_data_folder
- def symmetrise_axial(self, filename_in, filename_out=None, axis='x', plane_intercept=10,
- side_to_copy='below', keep_in_data_dimensions=True):
-
+ def symmetrise_axial(
+ self,
+ filename_in,
+ filename_out=None,
+ axis="x",
+ plane_intercept=10,
+ side_to_copy="below",
+ keep_in_data_dimensions=True,
+ ):
pfi_in, pfi_out = get_pfi_in_pfi_out(filename_in, filename_out, self.pfo_in, self.pfo_out)
im_segm = nib.load(pfi_in)
- data_labels = im_segm.get_data()
- data_symmetrised = symmetrise_data(data_labels,
- axis_direction=axis,
- plane_intercept=plane_intercept,
- side_to_copy=side_to_copy,
- keep_in_data_dimensions_boundaries=keep_in_data_dimensions)
+ data_labels = im_segm.get_fdata()
+ data_symmetrised = symmetrise_data(
+ data_labels,
+ axis_direction=axis,
+ plane_intercept=plane_intercept,
+ side_to_copy=side_to_copy,
+ keep_in_data_dimensions_boundaries=keep_in_data_dimensions,
+ )
im_symmetrised = set_new_data(im_segm, data_symmetrised)
nib.save(im_symmetrised, pfi_out)
- print('Symmetrised axis {0}, plane_intercept {1}, image of {2} saved in {3}.'.format(axis, plane_intercept, pfi_in, pfi_out))
+ print(f"Symmetrised axis {axis}, plane_intercept {plane_intercept}, image of {pfi_in} saved in {pfi_out}.")
return pfi_out
- def symmetrise_with_registration(self,
- filename_anatomy,
- filename_segmentation,
- list_labels_input,
- result_img_path,
- results_folder_path=None,
- list_labels_transformed=None,
- coord='z',
- reuse_registration=False):
- """
- Symmetrise a segmentation with registration: it uses NiftyReg.
+ def symmetrise_with_registration(
+ self,
+ filename_anatomy,
+ filename_segmentation,
+ list_labels_input,
+ result_img_path,
+ results_folder_path=None,
+ list_labels_transformed=None,
+ coord="z",
+ reuse_registration=False,
+ ):
+ """Symmetrise a segmentation with registration: it uses NiftyReg.
The old side is symmetrised in the new side, with new relabelling.
Method based on paths even if in tools
:param filename_anatomy: Path to File anatomical image
@@ -66,7 +74,6 @@ def symmetrise_with_registration(self,
---
NOTE: requires niftyreg.
"""
-
pfi_in_anatomy = connect_path_tail_head(self.pfo_in, filename_anatomy)
pfi_in_segmentation = connect_path_tail_head(self.pfo_in, filename_segmentation)
@@ -80,13 +87,13 @@ def symmetrise_with_registration(self,
pfi_out_segmentation = connect_path_tail_head(results_folder_path, result_img_path)
- def flip_data_path(input_im_path, output_im_path, axis='x'):
+ def flip_data_path(input_im_path, output_im_path, axis="x"):
# wrap flip data, having path for inputs and outputs.
if not os.path.isfile(input_im_path):
- raise IOError('input image file does not exist.')
+ raise OSError("input image file does not exist.")
im_labels = nib.load(input_im_path)
- data_labels = im_labels.get_data()
+ data_labels = im_labels.get_fdata()
data_flipped = flip_data(data_labels, axis_direction=axis)
im_relabelled = set_new_data(im_labels, data_flipped)
@@ -97,73 +104,75 @@ def flip_data_path(input_im_path, output_im_path, axis='x'):
# check input:
if not os.path.isfile(pfi_in_anatomy):
- raise IOError('input image file {} does not exist.'.format(pfi_in_anatomy))
+ msg = f"input image file {pfi_in_anatomy} does not exist."
+ raise OSError(msg)
if not os.path.isfile(pfi_in_segmentation):
- raise IOError('input segmentation file {} does not exist.'.format(pfi_in_segmentation))
+ msg = f"input segmentation file {pfi_in_segmentation} does not exist."
+ raise OSError(msg)
# erase labels that are not in the list from image and descriptor
- out_labels_side_A_path = os.path.join(results_folder_path, 'z_labels_side_A.nii.gz')
+ out_labels_side_A_path = os.path.join(results_folder_path, "z_labels_side_A.nii.gz")
labels_im = nib.load(pfi_in_segmentation)
- labels_data = labels_im.get_data()
+ labels_data = labels_im.get_fdata()
labels_to_erase = list(set(labels_data.flat) - set(list_labels_input + [0]))
# Relabel: from pfi_segmentation to out_labels_side_A_path
im_pfi_segmentation = nib.load(pfi_in_segmentation)
- segmentation_data_relabelled = relabeller(im_pfi_segmentation.get_data(), list_old_labels=labels_to_erase,
- list_new_labels=[0, ] * len(labels_to_erase))
+ segmentation_data_relabelled = relabeller(
+ im_pfi_segmentation.get_fdata(),
+ list_old_labels=labels_to_erase,
+ list_new_labels=[0] * len(labels_to_erase),
+ )
nib_labels_side_A_path = set_new_data(im_pfi_segmentation, segmentation_data_relabelled)
nib.save(nib_labels_side_A_path, out_labels_side_A_path)
# --- Create side B --- #
# flip anatomical image and register it over the non flipped
- out_anatomical_flipped_path = os.path.join(results_folder_path, 'z_anatomical_flipped.nii.gz')
+ out_anatomical_flipped_path = os.path.join(results_folder_path, "z_anatomical_flipped.nii.gz")
flip_data_path(pfi_in_anatomy, out_anatomical_flipped_path, axis=coord)
# flip the labels
- out_labels_flipped_path = os.path.join(results_folder_path, 'z_labels_flipped.nii.gz')
+ out_labels_flipped_path = os.path.join(results_folder_path, "z_labels_flipped.nii.gz")
flip_data_path(out_labels_side_A_path, out_labels_flipped_path, axis=coord)
# register anatomical flipped over non flipped
- out_anatomical_flipped_warped_path = os.path.join(results_folder_path, 'z_anatomical_flipped_warped.nii.gz')
- out_affine_transf_path = os.path.join(results_folder_path, 'z_affine_transformation.txt')
+ out_anatomical_flipped_warped_path = os.path.join(results_folder_path, "z_anatomical_flipped_warped.nii.gz")
+ out_affine_transf_path = os.path.join(results_folder_path, "z_affine_transformation.txt")
if not reuse_registration:
- cmd = 'reg_aladin -ref {0} -flo {1} -aff {2} -res {3}'.format(pfi_in_anatomy,
- out_anatomical_flipped_path,
- out_affine_transf_path,
- out_anatomical_flipped_warped_path)
- print('Registration started!\n')
+ cmd = f"reg_aladin -ref {pfi_in_anatomy} -flo {out_anatomical_flipped_path} -aff {out_affine_transf_path} -res {out_anatomical_flipped_warped_path}"
+ print("Registration started!\n")
os.system(cmd)
# propagate the registration to the flipped labels
- out_labels_side_B_path = os.path.join(results_folder_path, 'z_labels_side_B.nii.gz')
- cmd = 'reg_resample -ref {0} -flo {1} ' \
- '-res {2} -trans {3} -inter {4}'.format(out_labels_side_A_path,
- out_labels_flipped_path,
- out_labels_side_B_path,
- out_affine_transf_path,
- 0)
-
- print('Resampling started!\n')
+ out_labels_side_B_path = os.path.join(results_folder_path, "z_labels_side_B.nii.gz")
+ cmd = (
+ f"reg_resample -ref {out_labels_side_A_path} -flo {out_labels_flipped_path} "
+ f"-res {out_labels_side_B_path} -trans {out_affine_transf_path} -inter {0}"
+ )
+
+ print("Resampling started!\n")
os.system(cmd)
else:
- out_labels_side_B_path = os.path.join(results_folder_path, 'z_labels_side_B.nii.gz')
+ out_labels_side_B_path = os.path.join(results_folder_path, "z_labels_side_B.nii.gz")
# update labels of the side B if necessarily
if list_labels_transformed is not None:
- print('relabelling step!')
+ print("relabelling step!")
assert len(list_labels_transformed) == len(list_labels_input)
# relabel from out_labels_side_B_path to out_labels_side_B_path
im_segmentation_side_B = nib.load(out_labels_side_B_path)
- data_segmentation_side_B_new = relabeller(im_segmentation_side_B.get_data(),
- list_old_labels=list_labels_input,
- list_new_labels=list_labels_transformed)
+ data_segmentation_side_B_new = relabeller(
+ im_segmentation_side_B.get_fdata(),
+ list_old_labels=list_labels_input,
+ list_new_labels=list_labels_transformed,
+ )
nib_segmentation_side_B_new = set_new_data(im_segmentation_side_B, data_segmentation_side_B_new)
nib.save(nib_segmentation_side_B_new, out_labels_side_B_path)
@@ -173,21 +182,22 @@ def flip_data_path(input_im_path, output_im_path, axis='x'):
nib_side_A = nib.load(out_labels_side_A_path)
nib_side_B = nib.load(out_labels_side_B_path)
- data_side_A = nib_side_A.get_data()
- data_side_B = nib_side_B.get_data()
+ data_side_A = nib_side_A.get_fdata()
+ data_side_B = nib_side_B.get_fdata()
symmetrised_data = np.zeros_like(data_side_A)
# To manage the intersections of labels between old and new side. Vectorize later...
dims = data_side_A.shape
- print('Pointwise symmetrisation started!')
+ print("Pointwise symmetrisation started!")
for z in range(dims[0]):
for x in range(dims[1]):
for y in range(dims[2]):
- if (data_side_A[z, x, y] == 0 and data_side_B[z, x, y] != 0) or \
- (data_side_A[z, x, y] != 0 and data_side_B[z, x, y] == 0):
+ if (data_side_A[z, x, y] == 0 and data_side_B[z, x, y] != 0) or (
+ data_side_A[z, x, y] != 0 and data_side_B[z, x, y] == 0
+ ):
symmetrised_data[z, x, y] = np.max([data_side_A[z, x, y], data_side_B[z, x, y]])
elif data_side_A[z, x, y] != 0 and data_side_B[z, x, y] != 0:
if data_side_A[z, x, y] == data_side_B[z, x, y]:
@@ -197,4 +207,3 @@ def flip_data_path(input_im_path, output_im_path, axis='x'):
im_symmetrised = set_new_data(nib_side_A, symmetrised_data)
nib.save(im_symmetrised, pfi_out_segmentation)
-
diff --git a/nilabels/definitions.py b/nilabels/definitions.py
index 36711cc..5ba721d 100644
--- a/nilabels/definitions.py
+++ b/nilabels/definitions.py
@@ -1,6 +1,6 @@
import os
-__version__ = 'v0.0.8' # update also in setup.py
+__version__ = "v0.0.8" # update also in setup.py
root_dir = os.path.dirname(os.path.abspath(os.path.dirname(__file__)))
info = {
@@ -9,13 +9,13 @@
"description": "",
"repository": {
"type": "git",
- "url": ""
+ "url": "",
},
"author": "Sebastiano Ferraris",
"dependencies": {
# requirements.txt automatically generated using pipreqs
- "python requirements" : "{0}/requirements.txt".format(root_dir)
- }
+ "python requirements" : f"{root_dir}/requirements.txt",
+ },
}
diff --git a/nilabels/tools/aux_methods/label_descriptor_manager.py b/nilabels/tools/aux_methods/label_descriptor_manager.py
index 93ffc8b..224a415 100644
--- a/nilabels/tools/aux_methods/label_descriptor_manager.py
+++ b/nilabels/tools/aux_methods/label_descriptor_manager.py
@@ -1,5 +1,4 @@
-"""
-Module to manipulate descriptors as formatted by ITK-snap,
+"""Module to manipulate descriptors as formatted by ITK-snap,
as the one in the descriptor below:
################################################
@@ -30,17 +29,18 @@
An instance of a label descriptor manager is contains the path to a file with
a label descriptor in itk-snap convention or fsl convention and a parameter specifying the convetion.
"""
+
import collections
-import os
import copy
+import logging
+import os
+
import numpy as np
from nilabels.tools.aux_methods.sanity_checks import is_valid_permutation
from nilabels.tools.aux_methods.utils import permutation_from_cauchy_to_disjoints_cycles
-
-descriptor_standard_header = \
-"""################################################
+descriptor_standard_header = """################################################
# ITK-SnAP Label Description File
# File format:
# IDX -R- -G- -B- -A-- VIS MSH LABEL
@@ -56,8 +56,7 @@
################################################
"""
-descriptor_data_examples = \
-""" 0 0 0 0 0 0 0 "background"
+descriptor_data_examples = """ 0 0 0 0 0 0 0 "background"
1 255 0 0 1 1 1 "label one (l1)"
2 204 0 0 1 1 1 "label two (l2)"
3 51 51 255 1 1 1 "label three"
@@ -69,10 +68,8 @@
"""
-def generate_dummy_label_descriptor(pfi_output, list_labels=range(5),
- list_roi_names=None, list_colors_triplets=None):
- """
- For testing purposes, it creates a dummy label descriptor with the itk-snap convention
+def generate_dummy_label_descriptor(pfi_output, list_labels=range(5), list_roi_names=None, list_colors_triplets=None):
+ """For testing purposes, it creates a dummy label descriptor with the itk-snap convention
:param pfi_output: where to save the eventual label descriptor
:param list_labels: list of labels range, default 0:5
:param list_roi_names: names of the regions of interests. If None, default names are assigned.
@@ -82,58 +79,52 @@ def generate_dummy_label_descriptor(pfi_output, list_labels=range(5),
d = collections.OrderedDict()
visibility = [(1.0, 1, 1)] * len(list_labels)
if list_roi_names is None:
- list_roi_names = ["label {}".format(j) for j in list_labels]
- else:
- if not len(list_labels) == len(list_roi_names):
- raise IOError('Wrong input data')
+ list_roi_names = [f"label {j}" for j in list_labels]
+ elif len(list_labels) != len(list_roi_names):
+ raise OSError("Wrong input data")
if list_colors_triplets is None:
list_colors_triplets = [list(np.random.choice(range(256), 3)) for _ in list_labels]
- else:
- if not len(list_labels) == len(list(list_colors_triplets)):
- raise IOError('Wrong input data')
+ elif len(list_labels) != len(list(list_colors_triplets)):
+ raise OSError("Wrong input data")
for j_in, j in enumerate(list_labels):
up_d = {str(j): [list_colors_triplets[j_in], visibility[j_in], list_roi_names[j_in]]}
d.update(up_d)
- f = open(pfi_output, 'w+')
+ f = open(pfi_output, "w+")
f.write(descriptor_standard_header)
- for j in d.keys():
+ for j in d:
if j.isdigit():
- line = '{0: >5}{1: >6}{2: >6}{3: >6}{4: >9}{5: >6}{6: >6} "{7}"\n'.format(
- j, d[j][0][0], d[j][0][1], d[j][0][2], d[j][1][0], d[j][1][1], d[j][1][2], d[j][2])
+ line = f'{j: >5}{d[j][0][0]: >6}{d[j][0][1]: >6}{d[j][0][2]: >6}{d[j][1][0]: >9}{d[j][1][1]: >6}{d[j][1][2]: >6} "{d[j][2]}"\n'
f.write(line)
f.close()
return d
-class LabelsDescriptorManager(object):
-
- def __init__(self, pfi_label_descriptor, labels_descriptor_convention='itk-snap'):
+class LabelsDescriptorManager:
+ def __init__(self, pfi_label_descriptor, labels_descriptor_convention="itk-snap"):
self.pfi_label_descriptor = pfi_label_descriptor
self.convention = labels_descriptor_convention
self._check_path()
- if self.convention == 'itk-snap':
+ if self.convention == "itk-snap":
self.dict_label_descriptor = self.get_dict_itk_snap()
- elif self.convention == 'fsl':
+ elif self.convention == "fsl":
self.dict_label_descriptor = self.get_dict_fsl()
else:
- raise IOError("Signature for the variable **convention** can be only 'itk-snap' or 'fsl'.")
+ raise OSError("Signature for the variable **convention** can be only 'itk-snap' or 'fsl'.")
# ----------- Sanity checks -----------
def _check_path(self):
- """
- The path to labels descriptor it must exist.
+ """The path to labels descriptor it must exist.
:return: input path sanity check.
"""
if not os.path.exists(self.pfi_label_descriptor):
- msg = 'Label descriptor file {} does not exist'.format(self.pfi_label_descriptor)
- raise IOError(msg)
+ msg = f"Label descriptor file {self.pfi_label_descriptor} does not exist"
+ raise OSError(msg)
# ---------- Getters-setters -----------
def get_dict_itk_snap(self):
- """
- Parse the ITK-Snap label descriptor into an ordered dict data structure.
+ """Parse the ITK-Snap label descriptor into an ordered dict data structure.
Each element of the ordered dict is of the kind
{218 : [[128, 0, 128], [1.0, 1.0, 1.0], 'Corpus callosum']}
key : [[RGB], [B A vis], "label_name"]
@@ -142,33 +133,30 @@ def get_dict_itk_snap(self):
id : ''
"""
label_descriptor_dict = collections.OrderedDict()
- for l in open(self.pfi_label_descriptor, 'r'):
- if not l.strip().startswith('#') and not l == '':
- parsed_line = [j.strip() for j in l.split(' ') if not j == '']
- args = [tuple(parsed_line[1:4]), tuple(parsed_line[4:7]), parsed_line[7].replace('"', '')]
- args[0] = [int(k) for k in args[0]]
- args[1] = [float(k) for k in args[1]]
+ for l in open(self.pfi_label_descriptor):
+ if not l.strip().startswith("#") and l != "":
+ parsed_line = [j.strip() for j in l.split(" ") if j != ""]
+ args = [tuple(parsed_line[1:4]), tuple(parsed_line[4:7]), parsed_line[7].replace('"', "")]
+ args[0] = [int(k) for k in args[0]]
+ args[1] = [float(k) for k in args[1]]
label_descriptor_dict.update({int(parsed_line[0]): args})
return label_descriptor_dict
def get_dict_fsl(self):
- """
- Parse the fsl label descriptor into an ordered dict data structure.
- """
+ """Parse the fsl label descriptor into an ordered dict data structure."""
label_descriptor_dict = collections.OrderedDict()
- for l in open(self.pfi_label_descriptor, 'r'):
- if not l.strip().startswith('#'):
- parsed_line = [j.strip() for j in l.split(' ') if not j == '']
- args = [list(parsed_line[2:5]), float(parsed_line[5]), parsed_line[1]]
- args[0] = [int(k) for k in list(args[0])]
- args[1] = [args[1], 1, 1]
+ for l in open(self.pfi_label_descriptor):
+ if not l.strip().startswith("#"):
+ parsed_line = [j.strip() for j in l.split(" ") if j != ""]
+ args = [list(parsed_line[2:5]), float(parsed_line[5]), parsed_line[1]]
+ args[0] = [int(k) for k in list(args[0])]
+ args[1] = [args[1], 1, 1]
label_descriptor_dict.update({int(parsed_line[0]): args})
return label_descriptor_dict
def get_multi_label_dict(self, combine_right_left=True):
- """
- Different data structure to allow for multiple labels with the same name Left/Right (capital letters)
+ """Different data structure to allow for multiple labels with the same name Left/Right (capital letters)
:param combine_right_left:
:return:
"""
@@ -177,35 +165,35 @@ def get_multi_label_dict(self, combine_right_left=True):
# Fill mld_tmp, with the same values in the label descriptor switching the label name with
# the label id - note: possible abbreviations gets lost.
- for k_label_id in self.dict_label_descriptor.keys():
- mld_tmp.update({self.dict_label_descriptor[k_label_id][2].replace('"', '').strip(): [int(k_label_id)]})
+ for k_label_id in self.dict_label_descriptor:
+ mld_tmp.update({self.dict_label_descriptor[k_label_id][2].replace('"', "").strip(): [int(k_label_id)]})
mld_tmp_keys = list(mld_tmp.keys())
while len(mld_tmp_keys) > 0:
- if 'Right' in mld_tmp_keys[0]:
+ if "Right" in mld_tmp_keys[0]:
right_key = mld_tmp_keys[0]
- left_key = right_key.replace('Right', 'Left')
+ left_key = right_key.replace("Right", "Left")
if left_key in mld_tmp_keys:
mld.update({right_key: mld_tmp[right_key]})
mld.update({left_key: mld_tmp[left_key]})
mld_tmp_keys.remove(left_key)
if combine_right_left:
- mld.update({right_key.replace('Right', '').strip(): mld_tmp[right_key] + mld_tmp[left_key]})
+ mld.update({right_key.replace("Right", "").strip(): mld_tmp[right_key] + mld_tmp[left_key]})
else:
- print('Warning: Left key for {} not present'.format(right_key))
+ print(f"Warning: Left key for {right_key} not present")
mld.update({right_key: mld_tmp[right_key]})
- elif 'Left' in mld_tmp_keys[0]:
+ elif "Left" in mld_tmp_keys[0]:
left_key = mld_tmp_keys[0]
- right_key = left_key.replace('Left', 'Right')
+ right_key = left_key.replace("Left", "Right")
if right_key in mld_tmp_keys:
mld.update({left_key: mld_tmp[left_key]})
mld.update({right_key: mld_tmp[right_key]})
mld_tmp_keys.remove(right_key)
if combine_right_left:
- mld.update({left_key.replace('Left', '').strip(): mld_tmp[left_key] + mld_tmp[right_key]})
+ mld.update({left_key.replace("Left", "").strip(): mld_tmp[left_key] + mld_tmp[right_key]})
else:
- print('Warning: Right key for {} not present'.format(left_key))
+ print(f"Warning: Right key for {left_key} not present")
mld.update({left_key: mld_tmp[left_key]})
else:
mld.update({mld_tmp_keys[0]: mld_tmp[mld_tmp_keys[0]]})
@@ -215,38 +203,30 @@ def get_multi_label_dict(self, combine_right_left=True):
return mld
def save_label_descriptor(self, pfi_where_to_save):
- f = open(pfi_where_to_save, 'w+')
- if self.convention == 'itk-snap':
+ f = open(pfi_where_to_save, "w+")
+ if self.convention == "itk-snap":
f.write(descriptor_standard_header)
- for j in self.dict_label_descriptor.keys():
- if self.convention == 'itk-snap':
- line = '{0: >5}{1: >6}{2: >6}{3: >6}{4: >9}{5: >6}{6: >6} "{7}"\n'.format(
- j,
- self.dict_label_descriptor[j][0][0],
- self.dict_label_descriptor[j][0][1],
- self.dict_label_descriptor[j][0][2],
- self.dict_label_descriptor[j][1][0],
- int(self.dict_label_descriptor[j][1][1]),
- int(self.dict_label_descriptor[j][1][2]),
- self.dict_label_descriptor[j][2])
-
- elif self.convention == 'fsl':
- line = '{0} {1} {2} {3} {4} {5}\n'.format(
- j,
- self.dict_label_descriptor[j][2].replace(' ', '-'),
- self.dict_label_descriptor[j][0][0],
- self.dict_label_descriptor[j][0][1],
- self.dict_label_descriptor[j][0][2],
- self.dict_label_descriptor[j][1][0])
+ for j in self.dict_label_descriptor:
+ if self.convention == "itk-snap":
+ line = f'{j: >5}{self.dict_label_descriptor[j][0][0]: >6}{self.dict_label_descriptor[j][0][1]: >6}{self.dict_label_descriptor[j][0][2]: >6}{self.dict_label_descriptor[j][1][0]: >9}{int(self.dict_label_descriptor[j][1][1]): >6}{int(self.dict_label_descriptor[j][1][2]): >6} "{self.dict_label_descriptor[j][2]}"\n'
+
+ elif self.convention == "fsl":
+ line = "{0} {1} {2} {3} {4} {5}\n".format(
+ j,
+ self.dict_label_descriptor[j][2].replace(" ", "-"),
+ self.dict_label_descriptor[j][0][0],
+ self.dict_label_descriptor[j][0][1],
+ self.dict_label_descriptor[j][0][2],
+ self.dict_label_descriptor[j][1][0],
+ )
else:
- raise IOError("Assigned variable for **convention** can be only 'itk-snap' or 'fsl'.")
+ raise OSError("Assigned variable for **convention** can be only 'itk-snap' or 'fsl'.")
f.write(line)
f.close()
def save_as_multi_label_descriptor(self, pfi_destination):
- """
- Multi label descriptor is a custom file. It shows labels name and labels number keeping the Left and Right
+ """Multi label descriptor is a custom file. It shows labels name and labels number keeping the Left and Right
joined together in an additional line. It looks like:
---
Clear Label & 0
@@ -261,20 +241,19 @@ def save_as_multi_label_descriptor(self, pfi_destination):
:return: a saved multi label descriptor
"""
mld = self.get_multi_label_dict(combine_right_left=True)
- f = open(pfi_destination, 'w+')
- for k in mld.keys():
- line = '{0: <40}'.format(k)
+ f = open(pfi_destination, "w+")
+ for k in mld:
+ line = f"{k: <40}"
for j in mld[k]:
- line += '&{0: ^10}'.format(j)
+ line += f"&{j: ^10}"
f.write(line)
- f.write('\n')
+ f.write("\n")
f.close()
# ----------- Methods for labels manipulations - all these methods are not destructive -----------
def relabel(self, list_old_labels, list_new_labels, sort=True):
- """
- :param list_old_labels: list of existing labels
+ """:param list_old_labels: list of existing labels
:param list_new_labels: list of new labels to substitute the previous one
:param sort: provide sorted OrderedDict output.
E.G. old_labels = [3, 6, 9] new_labels = [4, 4, 100]
@@ -286,30 +265,30 @@ def relabel(self, list_old_labels, list_new_labels, sort=True):
"""
ldm_new = copy.deepcopy(self)
for k in list_old_labels:
- if k not in ldm_new.dict_label_descriptor.keys():
- raise IOError('Label {} in the input list not present'.format(k))
+ if k not in ldm_new.dict_label_descriptor:
+ msg = f"Label {k} in the input list not present"
+ raise OSError(msg)
del ldm_new.dict_label_descriptor[k]
for k_new in list_new_labels:
k_old = list_old_labels[list_new_labels.index(k_new)]
- if k_old in self.dict_label_descriptor.keys():
- ldm_new.dict_label_descriptor.update({k_new : self.dict_label_descriptor[k_old]})
+ if k_old in self.dict_label_descriptor:
+ ldm_new.dict_label_descriptor.update({k_new: self.dict_label_descriptor[k_old]})
if sort:
d_sorted = collections.OrderedDict()
for k in sorted(ldm_new.dict_label_descriptor.keys()):
- d_sorted.update({k : ldm_new.dict_label_descriptor[k]})
+ d_sorted.update({k: ldm_new.dict_label_descriptor[k]})
ldm_new.dict_label_descriptor = d_sorted
return ldm_new
def permute_labels(self, permutation):
- """
- Given a permutation, as specified in is_valid_permutation, provides a new labels descriptor manager
+ """Given a permutation, as specified in is_valid_permutation, provides a new labels descriptor manager
with the permuted labels.
:param permutation: a permutation defined by a list of integers (checked by is_valid_permutation under
sanity_check).
:return:
"""
if not is_valid_permutation(permutation):
- raise IOError('Not valid input permutation, please see the documentation.')
+ raise OSError("Not valid input permutation, please see the documentation.")
ldm_new = copy.deepcopy(self)
cycles = permutation_from_cauchy_to_disjoints_cycles(permutation)
@@ -320,31 +299,31 @@ def permute_labels(self, permutation):
return ldm_new
def erase_labels(self, labels_to_erase, verbose=True):
- """
- :param labels_to_erase: is a list of labels that will be erased.
+ """:param labels_to_erase: is a list of labels that will be erased.
:param verbose: raise warning message if label to erase is not present
"""
ldm_new = copy.deepcopy(self)
for lab in labels_to_erase:
if verbose and lab not in ldm_new.dict_label_descriptor.keys():
- print('Labels descriptor manager: Label {} can not be erased as not present.'.format(lab))
+ msg = f"Labels descriptor manager: Label {lab} can not be erased as not present."
+ logging.info(msg)
else:
del ldm_new.dict_label_descriptor[lab]
return ldm_new
def assign_all_other_labels_the_same_value(self, labels_to_keep, same_value_label):
- """
- :param labels_to_keep:
+ """:param labels_to_keep:
:param same_value_label:
:return:
"""
labels_that_will_have_the_same_value = list(set(self.dict_label_descriptor.keys()) - set(labels_to_keep) - {0})
- return self.relabel(labels_that_will_have_the_same_value,
- [same_value_label] * len(labels_that_will_have_the_same_value))
+ return self.relabel(
+ labels_that_will_have_the_same_value,
+ [same_value_label] * len(labels_that_will_have_the_same_value),
+ )
def keep_one_label(self, label_to_keep=1):
- """
- :param label_to_keep: all other values will be set to zero.
+ """:param label_to_keep: all other values will be set to zero.
:return:
"""
labels_not_to_keep = list(set(self.dict_label_descriptor.keys()) - {label_to_keep})
diff --git a/nilabels/tools/aux_methods/morpological_operations.py b/nilabels/tools/aux_methods/morpological_operations.py
index 0839656..d197506 100644
--- a/nilabels/tools/aux_methods/morpological_operations.py
+++ b/nilabels/tools/aux_methods/morpological_operations.py
@@ -3,24 +3,22 @@
def get_morphological_patch(dimension, shape):
- """
- :param dimension: dimension of the image (NOT the shape).
+ """:param dimension: dimension of the image (NOT the shape).
:param shape: circle or square.
:return: morphological patch as ndimage
"""
- if shape == 'circle':
+ if shape == "circle":
morpho_patch = ndimage.generate_binary_structure(dimension, 1)
- elif shape == 'square':
+ elif shape == "square":
morpho_patch = ndimage.generate_binary_structure(dimension, 3)
else:
- raise IOError
+ raise OSError
return morpho_patch
-def get_morphological_mask(point, omega, radius=5, shape='circle', morpho_patch=None):
- """
- Helper to obtain a morphological mask based on get_morphological_patch
+def get_morphological_mask(point, omega, radius=5, shape="circle", morpho_patch=None):
+ """Helper to obtain a morphological mask based on get_morphological_patch
:param point: centre of the mask
:param omega: grid dimension of the image domain. E.g. [256, 256, 128].
:param radius: radius of the mask if morpho_patch not given.
@@ -32,16 +30,15 @@ def get_morphological_mask(point, omega, radius=5, shape='circle', morpho_patch=
d = len(omega)
morpho_patch = get_morphological_patch(d, shape=shape)
- array_mask = np.zeros(omega, dtype=np.bool)
+ array_mask = np.zeros(omega, dtype=bool)
array_mask.itemset(tuple(point), 1)
for _ in range(radius):
array_mask = ndimage.binary_dilation(array_mask, structure=morpho_patch).astype(array_mask.dtype)
return array_mask
-def get_values_below_patch(point, target_image, radius=5, shape='circle', morpho_mask=None):
- """
- To obtain the list of the values below a maks.
+def get_values_below_patch(point, target_image, radius=5, shape="circle", morpho_mask=None):
+ """To obtain the list of the values below a maks.
:param point: central point of the patch
:param target_image: array image whose values we are interested into.
:param radius: patch radius if morpho_patch is not given.
@@ -57,8 +54,7 @@ def get_values_below_patch(point, target_image, radius=5, shape='circle', morpho
def get_circle_shell_for_given_radius(radius, dimension=3):
- """
- :param radius: radius of the circle.
+ """:param radius: radius of the circle.
:param dimension: must be 2 or 3.
:return: matrix coordinate values for a circle of given input radius and dimension centered at the origin.
E.G.
@@ -81,5 +77,5 @@ def get_circle_shell_for_given_radius(radius, dimension=3):
if (radius - 1) ** 2 < xi ** 2 + yi ** 2 <= radius ** 2:
circle.append((xi, yi))
else:
- raise IOError('Dimensions allowed are 2 or 3.')
+ raise OSError("Dimensions allowed are 2 or 3.")
return circle
diff --git a/nilabels/tools/aux_methods/sanity_checks.py b/nilabels/tools/aux_methods/sanity_checks.py
index 97ebdec..d135596 100644
--- a/nilabels/tools/aux_methods/sanity_checks.py
+++ b/nilabels/tools/aux_methods/sanity_checks.py
@@ -1,28 +1,26 @@
+import itertools
import os
-import time
import subprocess
-import itertools
+import time
def check_pfi_io(pfi_input, pfi_output):
- """
- Check if input and output files exist
+ """Check if input and output files exist
:param pfi_input: path to file input
:param pfi_output: path to file output. Can be None.
:return: True if pfi_input exists and if output is provided and exists.
"""
if not os.path.exists(pfi_input):
- raise IOError('Input file {} does not exists.'.format(pfi_input))
- if pfi_output is not None:
- if not os.path.exists(os.path.dirname(pfi_output)):
- raise IOError('Output file {} is located in a non-existing folder.'.format(
- pfi_output))
+ msg = f"Input file {pfi_input} does not exists."
+ raise OSError(msg)
+ if pfi_output is not None and not os.path.exists(os.path.dirname(pfi_output)):
+ msg = f"Output file {pfi_output} is located in a non-existing folder."
+ raise OSError(msg)
return True
def check_path_validity(pfi, interval=1, timeout=100):
- """
- Workaround function to cope with delayed operations in the cluster.
+ """Workaround function to cope with delayed operations in the cluster.
Boringly asking if something exists, until timeout or appearance of the sought file happen.
:param pfi: path to file to assess
:param interval: seconds
@@ -30,41 +28,37 @@ def check_path_validity(pfi, interval=1, timeout=100):
:return:
"""
if os.path.exists(pfi):
- if pfi.endswith('.nii.gz'):
+ if pfi.endswith(".nii.gz"):
mustend = time.time() + timeout
while time.time() < mustend:
try:
- subprocess.check_output('gunzip -t {}'.format(pfi), shell=True)
+ subprocess.check_output(f"gunzip -t {pfi}", shell=True)
except subprocess.CalledProcessError:
print("Caught CalledProcessError")
else:
return True
time.sleep(interval)
- msg = 'File {0} corrupted after 100 tests. \n'.format(pfi)
- raise IOError(msg)
- else:
- return True
- else:
- msg = '{} does not exist!'.format(pfi)
- raise IOError(msg)
+ msg = f"File {pfi} corrupted after 100 tests. \n"
+ raise OSError(msg)
+ return True
+ msg = f"{pfi} does not exist!"
+ raise OSError(msg)
def is_valid_permutation(in_perm, for_labels=True):
- """
- A permutation in generalised Cauchy notation is a list of 2 lists of same size:
+ """A permutation in generalised Cauchy notation is a list of 2 lists of same size:
a = [[1,2,3], [2,3,1]]
means permute 1 with 2, 2 with 3, 3 with 1.
:param for_labels: if True the permutation elements must be int.
:param in_perm: input permutation.
"""
- if not len(in_perm) == 2:
+ if len(in_perm) != 2:
return False
if not len(in_perm[0]) == len(in_perm[1]) == len(set(in_perm[0])) == len(set(in_perm[1])):
return False
- if not (sorted(set(in_perm[0])) > sorted(set(in_perm[1])))-(sorted(set(in_perm[0])) < sorted(set(in_perm[1]))) == 0:
+ if (sorted(set(in_perm[0])) > sorted(set(in_perm[1]))) - (sorted(set(in_perm[0])) < sorted(set(in_perm[1]))) != 0:
return False
- if for_labels:
+ if for_labels and not all(isinstance(x, int) for x in list(itertools.chain(*in_perm))):
# as dealing with labels, all the elements must be int
- if not all(isinstance(x, int) for x in list(itertools.chain(*in_perm))):
- return False
+ return False
return True
diff --git a/nilabels/tools/aux_methods/utils.py b/nilabels/tools/aux_methods/utils.py
index efb73a9..84f78ed 100644
--- a/nilabels/tools/aux_methods/utils.py
+++ b/nilabels/tools/aux_methods/utils.py
@@ -1,17 +1,15 @@
-import numpy as np
import os
import subprocess
-from sympy.combinatorics import Permutation
-from nilabels.tools.aux_methods.sanity_checks import is_valid_permutation
+import numpy as np
+from nilabels.tools.aux_methods.sanity_checks import is_valid_permutation
# ---- List utils ----
def lift_list(input_list):
- """
- List of nested lists becomes a list with the element exposed in the main list.
+ """List of nested lists becomes a list with the element exposed in the main list.
:param input_list: a list of lists.
:return: eliminates the first nesting levels of lists.
E.G.
@@ -20,19 +18,21 @@ def lift_list(input_list):
"""
if input_list == []:
return []
- else:
- return lift_list(input_list[0]) + (lift_list(input_list[1:]) if len(input_list) > 1 else []) \
- if type(input_list) is list else [input_list]
+
+ return (
+ lift_list(input_list[0]) + (lift_list(input_list[1:]) if len(input_list) > 1 else [])
+ if isinstance(input_list, list)
+ else [input_list]
+ )
def eliminates_consecutive_duplicates(input_list):
- """
- :param input_list: a list
+ """:param input_list: a list
:return: the same list with no consecutive duplicates.
"""
- output_list = [input_list[0], ]
+ output_list = [input_list[0]]
for i in range(1, len(input_list)):
- if not input_list[i] == input_list[i-1]:
+ if input_list[i] != input_list[i - 1]:
output_list.append(input_list[i])
return output_list
@@ -41,8 +41,7 @@ def eliminates_consecutive_duplicates(input_list):
def print_and_run(cmd, msg=None, safety_on=False, short_path_output=True):
- """
- run the command to console and print the message.
+ """Run the command to console and print the message.
if msg is None print the command itself.
:param cmd: command for the terminal
:param msg: message to show before running the command
@@ -51,12 +50,11 @@ def print_and_run(cmd, msg=None, safety_on=False, short_path_output=True):
:param safety_on: safety, in case you want to see the messages at a first run.
:return:
"""
+
def scan_and_remove_path(msg):
- """
- Take a string with a series of paths separated by a space and keeps only the base-names of each path.
- """
- a = [os.path.basename(p) for p in msg.split(' ')]
- return ' '.join(a)
+ """Take a string with a series of paths separated by a space and keeps only the base-names of each path."""
+ a = [os.path.basename(p) for p in msg.split(" ")]
+ return " ".join(a)
if short_path_output:
output_msg = scan_and_remove_path(cmd)
@@ -64,9 +62,9 @@ def scan_and_remove_path(msg):
output_msg = cmd
if msg is not None:
- output_msg += '{}\n{}'.format(msg, output_msg)
+ output_msg += f"{msg}\n{output_msg}"
- print('\n-> {}\n'.format(output_msg))
+ print(f"\n-> {output_msg}\n")
if not safety_on:
subprocess.call(cmd, shell=True)
@@ -78,8 +76,7 @@ def scan_and_remove_path(msg):
def labels_query(labels, segmentation_array=None, remove_zero=True):
- """
- labels_list can be a list or a list of lists in case some labels have to be considered together. labels_names
+ """labels_list can be a list or a list of lists in case some labels have to be considered together. labels_names
:param labels: can be int, list, string as 'all' or 'tot', or a string containing a path to a .txt or a numpy array
:param segmentation_array: optional segmentation image data (array)
:param remove_zero: do not return zero
@@ -87,32 +84,34 @@ def labels_query(labels, segmentation_array=None, remove_zero=True):
"""
labels_names = []
if labels is None:
- labels = 'all'
+ labels = "all"
if isinstance(labels, int):
if segmentation_array is not None:
assert labels in segmentation_array
- labels_list = [labels, ]
+ labels_list = [labels]
elif isinstance(labels, list):
labels_list = labels
elif isinstance(labels, str):
- if labels == 'all' and segmentation_array is not None:
- labels_list = list(np.sort(list(set(segmentation_array.astype(np.int).flat))))
- elif labels == 'tot' and segmentation_array is not None:
- labels_list = [list(np.sort(list(set(segmentation_array.astype(np.int).flat))))]
+ if labels == "all" and segmentation_array is not None:
+ labels_list = list(np.sort(list(set(segmentation_array.astype(int).flat))))
+ elif labels == "tot" and segmentation_array is not None:
+ labels_list = [list(np.sort(list(set(segmentation_array.astype(int).flat))))]
if labels_list[0][0] == 0:
if remove_zero:
labels_list = labels_list[0][1:] # remove initial zero
else:
labels_list = labels_list[0]
elif os.path.exists(labels):
- if labels.endswith('.txt'):
+ if labels.endswith(".txt"):
labels_list = list(np.loadtxt(labels))
else:
labels_list = list(np.load(labels))
else:
- raise IOError("Input labels must be a list, a list of lists, or an int or the string 'all' (with "
- "segmentation array not set to none)) or the path to a file with the labels.")
+ raise OSError(
+ "Input labels must be a list, a list of lists, or an int or the string 'all' (with "
+ "segmentation array not set to none)) or the path to a file with the labels.",
+ )
elif isinstance(labels, dict):
# expected input is the output of manipulate_descriptor.get_multi_label_dict (keys are labels names id are
# list of labels)
@@ -124,8 +123,10 @@ def labels_query(labels, segmentation_array=None, remove_zero=True):
else:
labels_list.append(labels[k][0])
else:
- raise IOError("Input labels must be a list, a list of lists, or an int or the string 'all' or the path to a"
- "file with the labels.")
+ raise OSError(
+ "Input labels must be a list, a list of lists, or an int or the string 'all' or the path to a"
+ "file with the labels.",
+ )
if not isinstance(labels, dict):
labels_names = [str(l) for l in labels_list]
@@ -138,15 +139,14 @@ def labels_query(labels, segmentation_array=None, remove_zero=True):
def permutation_from_cauchy_to_disjoints_cycles(cauchy_perm):
- """
- from [[1, 2, 3, 4, 5], [3, 4, 5, 2, 1]]
+ """From [[1, 2, 3, 4, 5], [3, 4, 5, 2, 1]]
to [[1, 3, 5], [2, 4]]
:param cauchy_perm: permutation in generalised Cauchy convention (any object, not necessarily numbers from 1 to n
or from 0 ot n-1) where the objects that are not permuted do not appear.
:return: input permutation written in disjoint cycles.
"""
if not is_valid_permutation(cauchy_perm):
- raise IOError('Input permutation is not valid')
+ raise OSError("Input permutation is not valid")
list_cycles = []
cycle = [cauchy_perm[0][0]]
while len(cauchy_perm[0]) > 0:
@@ -163,10 +163,8 @@ def permutation_from_cauchy_to_disjoints_cycles(cauchy_perm):
def permutation_from_disjoint_cycles_to_cauchy(cyclic_perm):
- """
- from [[1, 3, 5], [2, 4]]
+ """From [[1, 3, 5], [2, 4]]
to [[1, 2, 3, 4, 5], [3, 4, 5, 2, 1]]
"""
pairs = sorted([(a, b) for cycle in cyclic_perm for a, b in zip(cycle, cycle[1:] + cycle[:1])])
return [list(i) for i in zip(*pairs)]
-
diff --git a/nilabels/tools/aux_methods/utils_nib.py b/nilabels/tools/aux_methods/utils_nib.py
index b46324b..55d99b6 100644
--- a/nilabels/tools/aux_methods/utils_nib.py
+++ b/nilabels/tools/aux_methods/utils_nib.py
@@ -1,10 +1,9 @@
-import numpy as np
import nibabel as nib
+import numpy as np
def set_new_data(image, new_data, new_dtype=None, remove_nan=True):
- """
- From a nibabel image and a numpy array it creates a new image with
+ """From a nibabel image and a numpy array it creates a new image with
the same header of the image and the new_data as its data.
:param image: nibabel image
:param new_data:
@@ -16,49 +15,46 @@ def set_new_data(image, new_data, new_dtype=None, remove_nan=True):
if remove_nan:
new_data = np.nan_to_num(new_data)
# if nifty1
- if hd['sizeof_hdr'] == 348:
+ if hd["sizeof_hdr"] == 348:
new_image = nib.Nifti1Image(new_data, image.affine, header=hd)
# if nifty2
- elif hd['sizeof_hdr'] == 540:
+ elif hd["sizeof_hdr"] == 540:
new_image = nib.Nifti2Image(new_data, image.affine, header=hd)
else:
- raise IOError('Input image header problems in sizeof_hdr.')
+ raise OSError("Input image header problems in sizeof_hdr.")
if new_dtype is None:
new_image.set_data_dtype(new_data.dtype)
return new_image
- else:
- new_image.set_data_dtype(new_dtype)
- return new_image
+ new_image.set_data_dtype(new_dtype)
+ return new_image
def compare_two_nib(im1, im2):
- """
- Comparison between two nibabel imagse.
+ """Comparison between two nibabel imagse.
:param im1: one nibabel image (or str with path to a nifti image)
:param im2: another nibabel image (or another str with path to a nifti image)
:return: true false and plot to console if the images are the same or not (up to a tollerance in the data)
"""
- im1_name = 'First image'
- im2_name = 'Second image'
- msg = ''
+ im1_name = "First image"
+ im2_name = "Second image"
+ msg = ""
hd1 = im1.header
hd2 = im2.header
images_are_equal = True
# compare nifty version:
- if not hd1['sizeof_hdr'] == hd2['sizeof_hdr']:
-
- if hd1['sizeof_hdr'] == 348:
- msg += '{0} is nifti1\n{1} is nifti2.'.format(im1_name, im2_name)
+ if hd1["sizeof_hdr"] != hd2["sizeof_hdr"]:
+ if hd1["sizeof_hdr"] == 348:
+ msg += f"{im1_name} is nifti1\n{im2_name} is nifti2."
else:
- msg += '{0} is nifti2\n{1} is nifti1.'.format(im1_name, im2_name)
+ msg += f"{im1_name} is nifti2\n{im2_name} is nifti1."
images_are_equal = False
# Compare headers:
else:
for k in hd1.keys():
- if k not in ['scl_slope', 'scl_inter']:
+ if k not in ["scl_slope", "scl_inter"]:
val1, val2 = hd1[k], hd2[k]
are_different = val1 != val2
if isinstance(val1, np.ndarray):
@@ -66,35 +62,33 @@ def compare_two_nib(im1, im2):
if are_different:
images_are_equal = False
- msg += 'Header Key {0} for {1} is {2} - for {3} is {4} \n'.format(
- k, im1_name , hd1[k], im2_name, hd2[k])
+ msg += f"Header Key {k} for {im1_name} is {hd1[k]} - for {im2_name} is {hd2[k]} \n"
elif not np.isnan(hd1[k]) and np.isnan(hd2[k]):
images_are_equal = False
- msg += 'Header Key {0} for {1} is {2} - for {3} is {4} \n'.format(
- k, im1_name, hd1[k], im2_name, hd2[k])
+ msg += f"Header Key {k} for {im1_name} is {hd1[k]} - for {im2_name} is {hd2[k]} \n"
# Compare values and type:
- if not im1.get_data_dtype() == im2.get_data_dtype():
- msg += 'Dtype are different consistent {0} {1} - {2} {3} \n'.format(
- im1_name, im1.get_data_dtype(), im2_name, im2.get_data_dtype())
+ if im1.get_data_dtype() != im2.get_data_dtype():
+ msg += (
+ f"Dtype are different consistent {im1_name} {im1.get_data_dtype()} - {im2_name} {im2.get_data_dtype()} \n"
+ )
images_are_equal = False
- if not np.array_equal(im1.get_data(), im2.get_data()):
- msg += 'Data are different. \n'
+ if not np.array_equal(im1.get_fdata(), im2.get_fdata()):
+ msg += "Data are different. \n"
images_are_equal = False
- if np.array_equal(np.nan_to_num(im1.get_data()), np.nan_to_num(im2.get_data())):
- msg += '(data are different only up to nans)'
+ if np.array_equal(np.nan_to_num(im1.get_fdata()), np.nan_to_num(im2.get_fdata())):
+ msg += "(data are different only up to nans)"
if not np.array_equal(im1.affine, im2.affine):
- msg += 'Affine transformations are different. \n'
+ msg += "Affine transformations are different. \n"
images_are_equal = False
print(msg)
- print('\n -- Is it {} that images are equal!'.format(images_are_equal))
+ print(f"\n -- Is it {images_are_equal} that images are equal!")
return images_are_equal
def one_voxel_volume(im, decimals=6):
- """
- Volume of a single voxel of the given nifti image
+ """Volume of a single voxel of the given nifti image
:param im: nibabel image
:param decimals: number of decimals rouding the volume
:return: volume of a single voxel of the input image.
@@ -106,8 +100,7 @@ def one_voxel_volume(im, decimals=6):
def modify_image_data_type(im_input, new_dtype, update_descrip_field_header=None, verbose=1, remove_nan=True):
- """
- Change the data dtype of an input image with the specified new dtype.
+ """Change the data dtype of an input image with the specified new dtype.
The new data dtype must be from numpy data dtype.
:param im_input: input image
:param new_dtype: new data dtype
@@ -118,20 +111,19 @@ def modify_image_data_type(im_input, new_dtype, update_descrip_field_header=None
"""
if update_descrip_field_header is not None:
if not isinstance(update_descrip_field_header, str):
- raise IOError('update_description must be a string')
+ raise OSError("update_description must be a string")
hd = im_input.header
- hd['descrip'] = update_descrip_field_header
+ hd["descrip"] = update_descrip_field_header
im_input.update_header()
- new_im = set_new_data(im_input, im_input.get_data(), new_dtype=new_dtype, remove_nan=remove_nan)
+ new_im = set_new_data(im_input, im_input.get_fdata(), new_dtype=new_dtype, remove_nan=remove_nan)
if verbose > 0:
- print('Data type before {}'.format(im_input.get_data_dtype()))
- print('Data type after {}'.format(new_im.get_data_dtype()))
+ print(f"Data type before {im_input.get_data_dtype()}")
+ print(f"Data type after {new_im.get_data_dtype()}")
return new_im
-def modify_affine_transformation(im_input, new_aff, q_form=True, s_form=True, verbose=1, multiplication_side='left'):
- """
- Change q_form or s_form or both translational part and rotational part.
+def modify_affine_transformation(im_input, new_aff, q_form=True, s_form=True, verbose=1, multiplication_side="left"):
+ """Change q_form or s_form or both translational part and rotational part.
:param im_input: nibabel input image
:param new_aff: new affine transformation to be multiplied or replaced
:param q_form: [True] affect q_form
@@ -144,25 +136,25 @@ def modify_affine_transformation(im_input, new_aff, q_form=True, s_form=True, ve
NOTE: please see the documentation http://nipy.org/nibabel/nifti_images.html#choosing-image-affine for more on the
relationships between s_form affine, q_form affine and fall-back header affine.
"""
- if np.linalg.det(new_aff) < 0 :
- print('WARNING: affine matrix proposed has negative determinant.')
- if multiplication_side is 'left':
+ if np.linalg.det(new_aff) < 0:
+ print("WARNING: affine matrix proposed has negative determinant.")
+ if multiplication_side == "left":
new_transf = new_aff.dot(im_input.affine)
- elif multiplication_side is 'right':
+ elif multiplication_side == "right":
new_transf = im_input.affine.dot(new_aff)
- elif multiplication_side is 'replace':
+ elif multiplication_side == "replace":
new_transf = new_aff
else:
- raise IOError('multiplication_side parameter can be lef, right, or replace.')
+ raise OSError("multiplication_side parameter can be lef, right, or replace.")
# create output image on the input
- if im_input.header['sizeof_hdr'] == 348:
- new_image = nib.Nifti1Image(im_input.get_data(), new_transf, header=im_input.header)
+ if im_input.header["sizeof_hdr"] == 348:
+ new_image = nib.Nifti1Image(im_input.get_fdata(), new_transf, header=im_input.header)
# if nifty2
- elif im_input.header['sizeof_hdr'] == 540:
- new_image = nib.Nifti2Image(im_input.get_data(), new_transf, header=im_input.header)
+ elif im_input.header["sizeof_hdr"] == 540:
+ new_image = nib.Nifti2Image(im_input.get_fdata(), new_transf, header=im_input.header)
else:
- raise IOError('Input image header problems in sizeof_hdr.')
+ raise OSError("Input image header problems in sizeof_hdr.")
if s_form:
new_image.set_sform(new_transf)
@@ -177,21 +169,20 @@ def modify_affine_transformation(im_input, new_aff, q_form=True, s_form=True, ve
if verbose > 0:
# print intermediate results
- print('Affine input image:')
+ print("Affine input image:")
print(im_input.affine)
- print('Affine after update:')
+ print("Affine after update:")
print(new_image.affine)
- print('Q-form after update:')
+ print("Q-form after update:")
print(new_image.get_qform(coded=True))
- print('S-form after update:')
+ print("S-form after update:")
print(new_image.get_sform(coded=True))
return new_image
def replace_translational_part(im_input, new_translation, q_form=True, s_form=True):
- """
- Create new image with translational part replaced. Not destructive.
+ """Create new image with translational part replaced. Not destructive.
:param im_input: nibabel input image
:param new_translation: new translational part
:param q_form: new translation will be applied to the q_form
@@ -201,23 +192,25 @@ def replace_translational_part(im_input, new_translation, q_form=True, s_form=Tr
"""
new_affine = np.copy(im_input.affine)
new_affine[:3, 3] = new_translation[:3]
- im_out = modify_affine_transformation(im_input, new_aff=new_affine, q_form=q_form, s_form=s_form,
- multiplication_side='replace')
- return im_out
+ return modify_affine_transformation(
+ im_input,
+ new_aff=new_affine,
+ q_form=q_form,
+ s_form=s_form,
+ multiplication_side="replace",
+ )
def remove_nan_from_im(im_input):
- """
- Returns the same input images without nan
+ """Returns the same input images without nan
:param im_input: nibabel input image
- :return: set_new_data(im_input, np.nan_to_num(im_input.get_data()))
+ :return: set_new_data(im_input, np.nan_to_num(im_input.get_fdata()))
"""
- return set_new_data(im_input, np.nan_to_num(im_input.get_data()))
+ return set_new_data(im_input, np.nan_to_num(im_input.get_fdata()))
def images_are_overlapping(im1, im2):
- """
- Check if two images have overlapping (congruent domain) in the real space (same data and same shape)
+ """Check if two images have overlapping (congruent domain) in the real space (same data and same shape)
:param im1: nibabel image.
:param im2: nibabel image.
:return: np.array_equal(im1.shape[:3], im2.shape[:3]) * np.array_equal(im1.affine, im2.affine)
diff --git a/nilabels/tools/aux_methods/utils_path.py b/nilabels/tools/aux_methods/utils_path.py
index 1ff1b29..d68925e 100644
--- a/nilabels/tools/aux_methods/utils_path.py
+++ b/nilabels/tools/aux_methods/utils_path.py
@@ -2,8 +2,7 @@
def connect_path_tail_head(tail, head):
- """
- It is expected to find
+ """It is expected to find
1) the path to folder in tail and filename in head.
2) the full path in the head (with tail as sub-path).
3) the tail with a base path and head to have an additional path + filename.
@@ -11,17 +10,16 @@ def connect_path_tail_head(tail, head):
:param head:
:return:
"""
- if tail is None or tail == '':
+ if tail is None or tail == "":
return head
if head.startswith(tail): # Case 2
return head
- else: # case 1, 3
- return os.path.join(tail, head)
+ # case 1, 3
+ return os.path.join(tail, head)
def get_pfi_in_pfi_out(filename_in, filename_out, pfo_in, pfo_out):
- """
- Core method of every facade to connect working folder with input data files.
+ """Core method of every facade to connect working folder with input data files.
:param filename_in: filename input
:param filename_out: filename output
:param pfo_in: path to folder input
@@ -31,10 +29,9 @@ def get_pfi_in_pfi_out(filename_in, filename_out, pfo_in, pfo_out):
pfi_in = connect_path_tail_head(pfo_in, filename_in)
if filename_out is None:
pfi_out = pfi_in
+ elif pfo_out is None:
+ pfi_out = connect_path_tail_head(pfo_in, filename_out)
else:
- if pfo_out is None:
- pfi_out = connect_path_tail_head(pfo_in, filename_out)
- else:
- pfi_out = connect_path_tail_head(pfo_out, filename_out)
+ pfi_out = connect_path_tail_head(pfo_out, filename_out)
return pfi_in, pfi_out
diff --git a/nilabels/tools/aux_methods/utils_rotations.py b/nilabels/tools/aux_methods/utils_rotations.py
index 6841fb3..5c237dd 100644
--- a/nilabels/tools/aux_methods/utils_rotations.py
+++ b/nilabels/tools/aux_methods/utils_rotations.py
@@ -1,31 +1,27 @@
import numpy as np
-def get_small_orthogonal_rotation(theta, principal_axis='pitch'):
- if principal_axis == 'yaw':
- rot = np.array([[np.cos(theta), -np.sin(theta), 0, 0],
- [np.sin(theta), np.cos(theta), 0, 0],
- [0, 0, 1, 0],
- [0, 0, 0, 1]])
- elif principal_axis == 'pitch':
- rot = np.array([[1, 0, 0, 0],
- [0, np.cos(theta), -np.sin(theta), 0],
- [0, np.sin(theta), np.cos(theta), 0],
- [0, 0, 0, 1]])
- elif principal_axis == 'roll':
- rot = np.array([[np.cos(theta), 0, np.sin(theta), 0],
- [0, 1, 0, 0],
- [-np.sin(theta), 0, np.cos(theta), 0],
- [0, 0, 0, 1]])
+def get_small_orthogonal_rotation(theta, principal_axis="pitch"):
+ if principal_axis == "yaw":
+ rot = np.array(
+ [[np.cos(theta), -np.sin(theta), 0, 0], [np.sin(theta), np.cos(theta), 0, 0], [0, 0, 1, 0], [0, 0, 0, 1]],
+ )
+ elif principal_axis == "pitch":
+ rot = np.array(
+ [[1, 0, 0, 0], [0, np.cos(theta), -np.sin(theta), 0], [0, np.sin(theta), np.cos(theta), 0], [0, 0, 0, 1]],
+ )
+ elif principal_axis == "roll":
+ rot = np.array(
+ [[np.cos(theta), 0, np.sin(theta), 0], [0, 1, 0, 0], [-np.sin(theta), 0, np.cos(theta), 0], [0, 0, 0, 1]],
+ )
else:
- raise IOError('principal_axis parameter can be pitch, roll or yaw')
+ raise OSError("principal_axis parameter can be pitch, roll or yaw")
return rot # to be multiplied on the right side as im_input.affine.dot(rot)
-def get_roto_translation_matrix(theta, rotation_axis=np.array([1, 0, 0]), translation=np.array([0, 0, 0])):
- """
- Exploit the fact that every rotation is uniquely defined by an angle and a rotation direction.
+def get_roto_translation_matrix(theta, rotation_axis=np.array([1, 0, 0]), translation=np.array([0, 0, 0])):
+ """Exploit the fact that every rotation is uniquely defined by an angle and a rotation direction.
:param theta: rotation parameter
:param rotation_axis: rotation axis (3d vector)
:param translation: tranlsational part.
@@ -33,13 +29,13 @@ def get_roto_translation_matrix(theta, rotation_axis=np.array([1, 0, 0]), trans
"""
n = np.linalg.norm(rotation_axis)
if np.abs(n) < 0.001:
- raise IOError('Input rotation axis too close to zero.')
+ raise OSError("Input rotation axis too close to zero.")
rot_versor = rotation_axis / n
# Rodriguez magic formula for rotation part:
- skew_rot_versor = np.array([[0, -rot_versor[2], rot_versor[1]],
- [rot_versor[2], 0, -rot_versor[0]],
- [-rot_versor[1], rot_versor[0], 0]])
+ skew_rot_versor = np.array(
+ [[0, -rot_versor[2], rot_versor[1]], [rot_versor[2], 0, -rot_versor[0]], [-rot_versor[1], rot_versor[0], 0]],
+ )
rot_part = np.eye(3) + np.sin(theta) * skew_rot_versor + (1 - np.cos(theta)) * skew_rot_versor.dot(skew_rot_versor)
# transformations parameters
@@ -51,11 +47,11 @@ def get_roto_translation_matrix(theta, rotation_axis=np.array([1, 0, 0]), trans
def basic_90_rot_ax(m, ax=0):
- """
- Basic rotations of a 3d matrix. Ingredient of the method axial_rotations.
+ """Basic rotations of a 3d matrix. Ingredient of the method axial_rotations.
----------
- Example:
+ Example:
+ -------
cube = array([[[0, 1],
[2, 3]],
@@ -73,6 +69,7 @@ def basic_90_rot_ax(m, ax=0):
:param m: 3d matrix
:param ax: axis of rotation
:return: rotate the cube around axis ax, perpendicular to the face [[0,1],[2,3]]
+
"""
ax %= 3
@@ -80,20 +77,17 @@ def basic_90_rot_ax(m, ax=0):
return np.rot90(m[:, ::-1, :].swapaxes(0, 1)[::-1, :, :].swapaxes(0, 2), 3)
if ax == 1:
return m.swapaxes(0, 2)[::-1, :, :]
- if ax == 2:
- return np.rot90(m, 1)
+ return np.rot90(m, 1)
def axial_90_rotations(m, rot=1, ax=2):
- """
- :param m: 3d matrix
+ """:param m: 3d matrix
:param rot: number of rotations
:param ax: axis of rotation
:return: m rotate rot times around axis ax, according to convention.
"""
-
- if m.ndim is not 3:
- raise IOError('Input matrix must be a 3d volume.')
+ if m.ndim != 3:
+ raise OSError("Input matrix must be a 3d volume.")
rot %= 4
@@ -106,34 +100,37 @@ def axial_90_rotations(m, rot=1, ax=2):
return m
-def flip_data(in_data, axis_direction='x'):
- """
- Flip an array along one dimension and respect to one orthogonal axis
+def flip_data(in_data, axis_direction="x"):
+ """Flip an array along one dimension and respect to one orthogonal axis
:param in_data: input array
:param axis_direction: axis for the flipping
:return: in_data flipped respect to the axis_direction.
"""
- if not in_data.ndim == 3:
- msg = 'Input array must be 3-dimensional.'
- raise IOError(msg)
+ if in_data.ndim != 3:
+ msg = "Input array must be 3-dimensional."
+ raise OSError(msg)
- if axis_direction == 'x':
+ if axis_direction == "x":
out_data = in_data[:, ::-1, :]
- elif axis_direction == 'y':
+ elif axis_direction == "y":
out_data = in_data[:, :, ::-1]
- elif axis_direction == 'z':
+ elif axis_direction == "z":
out_data = in_data[::-1, :, :]
else:
- msg = 'axis variable must be one of the following: {}.'.format(['x', 'y', 'z'])
- raise IOError(msg)
+ msg = "axis variable must be one of the following: {}.".format(["x", "y", "z"])
+ raise OSError(msg)
return out_data
-def symmetrise_data(in_data, axis_direction='x', plane_intercept=10, side_to_copy='below',
- keep_in_data_dimensions_boundaries=True):
- """
- Symmetrise the input_array according to the axial plane
+def symmetrise_data(
+ in_data,
+ axis_direction="x",
+ plane_intercept=10,
+ side_to_copy="below",
+ keep_in_data_dimensions_boundaries=True,
+):
+ """Symmetrise the input_array according to the axial plane
axis = plane_intercept
the copied part can be 'below' or 'above' the axes, following the ordering.
@@ -144,17 +141,16 @@ def symmetrise_data(in_data, axis_direction='x', plane_intercept=10, side_to_cop
:param keep_in_data_dimensions_boundaries:
:return:
"""
-
# Sanity check:
if not in_data.ndim == 3:
- raise IOError('Input array must be 3-dimensional.')
+ raise OSError("Input array must be 3-dimensional.")
- if side_to_copy not in ['below', 'above']:
- raise IOError('side_to_copy must be one of the two {}.'.format(['below', 'above']))
+ if side_to_copy not in ["below", "above"]:
+ raise OSError("side_to_copy must be one of the two {}.".format(["below", "above"]))
- if axis_direction not in ['x', 'y', 'z']:
- raise IOError('axis variable must be one of the following: {}.'.format(['x', 'y', 'z']))
+ if axis_direction not in ["x", "y", "z"]:
+ raise OSError("axis variable must be one of the following: {}.".format(["x", "y", "z"]))
# step 1: find the block to symmetrise.
# step 2: create the symmetric and glue it to the block.
@@ -162,68 +158,64 @@ def symmetrise_data(in_data, axis_direction='x', plane_intercept=10, side_to_cop
out_data = None
- if axis_direction == 'x':
-
- if side_to_copy == 'below':
+ if axis_direction == "x":
+ if side_to_copy == "below":
s_block = in_data[:, :plane_intercept, :]
s_block_symmetric = s_block[:, ::-1, :]
out_data = np.concatenate((s_block, s_block_symmetric), axis=1)
- if side_to_copy == 'above':
+ if side_to_copy == "above":
s_block = in_data[:, plane_intercept:, :]
s_block_symmetric = s_block[:, ::-1, :]
out_data = np.concatenate((s_block_symmetric, s_block), axis=1)
- if axis_direction == 'y':
-
- if side_to_copy == 'below':
+ if axis_direction == "y":
+ if side_to_copy == "below":
s_block = in_data[:, :, :plane_intercept]
s_block_symmetric = s_block[:, :, ::-1]
out_data = np.concatenate((s_block, s_block_symmetric), axis=2)
- if side_to_copy == 'above':
+ if side_to_copy == "above":
s_block = in_data[:, :, plane_intercept:]
s_block_symmetric = s_block[:, :, ::-1]
out_data = np.concatenate((s_block_symmetric, s_block), axis=2)
- if axis_direction == 'z':
-
- if side_to_copy == 'below':
+ if axis_direction == "z":
+ if side_to_copy == "below":
s_block = in_data[:plane_intercept, :, :]
s_block_symmetric = s_block[::-1, :, :]
out_data = np.concatenate((s_block, s_block_symmetric), axis=0)
- if side_to_copy == 'above':
+ if side_to_copy == "above":
s_block = in_data[plane_intercept:, :, :]
s_block_symmetric = s_block[::-1, :, :]
out_data = np.concatenate((s_block_symmetric, s_block), axis=0)
if keep_in_data_dimensions_boundaries:
+ if side_to_copy == "below":
+ out_data = out_data[: in_data.shape[0], : in_data.shape[1], : in_data.shape[2]]
- if side_to_copy == 'below':
- out_data = out_data[:in_data.shape[0], :in_data.shape[1], :in_data.shape[2]]
-
- if side_to_copy == 'above':
- out_data = out_data[out_data.shape[0] - in_data.shape[0]:,
- out_data.shape[1] - in_data.shape[1]:,
- out_data.shape[2] - in_data.shape[2]:]
+ if side_to_copy == "above":
+ out_data = out_data[
+ out_data.shape[0] - in_data.shape[0] :,
+ out_data.shape[1] - in_data.shape[1] :,
+ out_data.shape[2] - in_data.shape[2] :,
+ ]
return out_data
def reorient_b_vect(bvect_array, transform):
- """
- Reorient b-vectors of a DWI scan.
+ """Reorient b-vectors of a DWI scan.
:param bvect_array: array of b-vectors column convention n x 3.
:param transform: transformation to be applied to each b-vector.
:return:
"""
- return np.einsum('...kl,...l->...k', bvect_array, transform).T
+ return np.einsum("...kl,...l->...k", bvect_array, transform).T
-def reorient_b_vect_from_files(pfi_input, pfi_output, transform, fmt='%.14f'):
- """
- Reorient b-vectors of a DWI scan from textfiles.
+def reorient_b_vect_from_files(pfi_input, pfi_output, transform, fmt="%.14f"):
+ """Reorient b-vectors of a DWI scan from textfiles.
:param pfi_input: input to txt data with b-vectors column convention 'x y z \\'.
:param pfi_output: output b-vectors after transformation in a new .txt file.
:param transform: transformation to be applied to each b-vector.
@@ -236,8 +228,7 @@ def reorient_b_vect_from_files(pfi_input, pfi_output, transform, fmt='%.14f'):
def matrix_vector_field_product(j_input, v_input):
- """
- :param j_input: matrix m x n x (4 or 9) as for example a jacobian column major
+ r""":param j_input: matrix m x n x (4 or 9) as for example a jacobian column major
:param v_input: matrix m x n x (2 or 3) to be multiplied by the matrix point-wise.
:return: m x n x (2 or 3) whose each element is the result of the product of the
matrix (i,j,:) multiplied by the corresponding element in the vector v (i,j,:).
@@ -261,13 +252,13 @@ def matrix_vector_field_product(j_input, v_input):
"""
if not len(j_input.shape) == len(v_input.shape):
- raise IOError
+ raise OSError
if not j_input.shape[:-1] == v_input.shape[:-1]:
- raise IOError
+ raise OSError
d = v_input.shape[-1]
vol = list(v_input.shape[:d])
extra_ones = len(v_input.shape) - (len(vol) + 1)
temp = j_input.reshape(vol + [1] * extra_ones + [d, d]) # transform in squared block with additional ones
- return np.einsum('...kl,...l->...k', temp, v_input)
+ return np.einsum("...kl,...l->...k", temp, v_input)
diff --git a/nilabels/tools/caliber/distances.py b/nilabels/tools/caliber/distances.py
index 1e38ba3..817adcc 100644
--- a/nilabels/tools/caliber/distances.py
+++ b/nilabels/tools/caliber/distances.py
@@ -1,15 +1,14 @@
import numpy as np
-import pandas as pa
+import pandas as pd
from scipy import ndimage as nd
from nilabels.tools.detections.contours import get_internal_contour_with_erosion_at_label
-
# --- Auxiliaries
+
def centroid_array(arr, labels):
- """
- Auxiliary of centroid, for arrays in array coordinates.
+ """Auxiliary of centroid, for arrays in array coordinates.
:param arr: numpy array of any dimension > 1 .
:param labels: list of labels
:return: list of centre of masses for the selected values in the array.
@@ -26,14 +25,13 @@ def centroid_array(arr, labels):
def centroid(im, labels, return_mm3=True):
- """
- Centroid (center of mass, barycenter) of a list of labels.
+ """Centroid (center of mass, barycenter) of a list of labels.
:param im: nifti image from nibabel.
:param labels: list of labels, e.g. [3] or [2, 3, 45]
:param return_mm3: if true the answer is in mm if false in voxel indexes.
:return: list of centroids, one for each label in the input order.
"""
- centers_of_mass = centroid_array(im.get_data(), labels)
+ centers_of_mass = centroid_array(im.get_fdata(), labels)
ans = []
if return_mm3:
for cm in centers_of_mass:
@@ -51,8 +49,7 @@ def centroid(im, labels, return_mm3=True):
def covariance_matrices(im, labels, return_mm3=True):
- """
- Considers the label as a point distribution in the space, and returns the covariance matrix of the points
+ """Considers the label as a point distribution in the space, and returns the covariance matrix of the points
distributions.
:param im: input nibabel image
:param labels: list of labels input.
@@ -61,7 +58,7 @@ def covariance_matrices(im, labels, return_mm3=True):
"""
cov_matrices = [np.zeros([3, 3])] * len(labels)
for l_id, l in enumerate(labels):
- coords = np.where(im.get_data() == l) # returns [X_vector, Y_vector, Z_vector]
+ coords = np.where(im.get_fdata() == l) # returns [X_vector, Y_vector, Z_vector]
if np.count_nonzero(coords) > 0:
cov_matrices[l_id] = np.cov(coords)
else:
@@ -73,8 +70,7 @@ def covariance_matrices(im, labels, return_mm3=True):
def covariance_distance_between_matrices(m1, m2, mul_factor=1):
- """
- Covariance distance between matrices m1 and m2, defined as
+ """Covariance distance between matrices m1 and m2, defined as
d = factor * (1 - (trace(m1 * m2)) / (norm_fro(m1) + norm_fro(m2)))
:param m1: matrix
:param m2: matrix
@@ -84,62 +80,65 @@ def covariance_distance_between_matrices(m1, m2, mul_factor=1):
if np.nan in m1 or np.nan in m2:
cd = np.nan
else:
- cd = mul_factor * (1 - (np.trace(m1.dot(m2)) / (np.linalg.norm(m1, ord='fro') * np.linalg.norm(m2, ord='fro'))))
+ cd = mul_factor * (1 - (np.trace(m1.dot(m2)) / (np.linalg.norm(m1, ord="fro") * np.linalg.norm(m2, ord="fro"))))
return cd
+
# --- global distances: (segm, segm) |-> real
def global_dice_score(im_segm1, im_segm2):
- """
- Global dice score as in Munoz-Moreno et al. 2013
+ """Global dice score as in Munoz-Moreno et al. 2013
:param im_segm1:
:param im_segm2:
:return:
"""
- all_labels1 = set(im_segm1.get_data().astype(np.int).flat) - {0}
- all_labels2 = set(im_segm1.get_data().astype(np.int).flat) - {0}
- sum_intersections = np.sum([np.count_nonzero((im_segm1.get_data() == l) * (im_segm2.get_data() == l))
- for l in set.union(all_labels1, all_labels2)])
- return 2 * sum_intersections / float(np.count_nonzero(im_segm1.get_data()) + np.count_nonzero(im_segm2.get_data()))
+ all_labels1 = set(im_segm1.get_fdata().astype(int).flat) - {0}
+ all_labels2 = set(im_segm1.get_fdata().astype(int).flat) - {0}
+ sum_intersections = np.sum(
+ [
+ np.count_nonzero((im_segm1.get_fdata() == l) * (im_segm2.get_fdata() == l))
+ for l in set.union(all_labels1, all_labels2)
+ ],
+ )
+ return (
+ 2 * sum_intersections / float(np.count_nonzero(im_segm1.get_fdata()) + np.count_nonzero(im_segm2.get_fdata()))
+ )
def global_outline_error(im_segm1, im_segm2):
- """
- Volume of the binarised image differences over the average binarised volume of the two images.
+ """Volume of the binarised image differences over the average binarised volume of the two images.
:param im_segm1:
:param im_segm2:
:return:
"""
- num_voxels_1, num_voxels_2 = np.count_nonzero(im_segm1.get_data()), np.count_nonzero(im_segm2.get_data())
- num_voxels_diff = np.count_nonzero(im_segm1.get_data() - im_segm2.get_data())
- return num_voxels_diff / (.5 * (num_voxels_1 + num_voxels_2))
+ num_voxels_1, num_voxels_2 = np.count_nonzero(im_segm1.get_fdata()), np.count_nonzero(im_segm2.get_fdata())
+ num_voxels_diff = np.count_nonzero(im_segm1.get_fdata() - im_segm2.get_fdata())
+ return num_voxels_diff / (0.5 * (num_voxels_1 + num_voxels_2))
# --- Single labels distances (segm, segm, label) |-> real
def dice_score_one_label(im_segm1, im_segm2, lab):
- """
- Dice score for a single label. The input images must have the same grid shape (but can have different affine part).
+ """Dice score for a single label. The input images must have the same grid shape
+ (but can have different affine part).
:param im_segm1: nibabel image representing a segmentation
:param im_segm2: as im_segm1
:param lab: a label.
:return: dice score distance for the given label. If the label is not present, it returns a nan.
"""
- place1 = im_segm1.get_data() == lab
- place2 = im_segm2.get_data() == lab
+ place1 = im_segm1.get_fdata() == lab
+ place2 = im_segm2.get_fdata() == lab
non_zero_place1 = np.count_nonzero(place1)
non_zero_place2 = np.count_nonzero(place2)
if non_zero_place1 + non_zero_place2 == 0:
return np.nan
- else:
- return 2 * np.count_nonzero(place1 * place2) / float(non_zero_place1 + non_zero_place2)
+ return 2 * np.count_nonzero(place1 * place2) / float(non_zero_place1 + non_zero_place2)
def d_H(im1, im2, lab, return_mm3):
- """
- Asymmetric component of the Hausdorff distance.
+ """Asymmetric component of the Hausdorff distance.
:param im1: first image (nibabel format)
:param im2: second image (nibabel format)
:param lab: label in the image
@@ -147,8 +146,8 @@ def d_H(im1, im2, lab, return_mm3):
:return: max(d(x, contourY)), x: point belonging to the first contour,
contourY: contour of the second segmentation.
"""
- arr1 = im1.get_data() == lab
- arr2 = im2.get_data() == lab
+ arr1 = im1.get_fdata() == lab
+ arr2 = im2.get_fdata() == lab
if np.count_nonzero(arr1) == 0 or np.count_nonzero(arr2) == 0:
return np.nan
if return_mm3:
@@ -162,9 +161,8 @@ def hausdorff_distance_one_label(im_segm1, im_segm2, lab, return_mm3):
return np.max([d_H(im_segm1, im_segm2, lab, return_mm3), d_H(im_segm2, im_segm1, lab, return_mm3)])
-def symmetric_contour_distance_one_label(im1, im2, lab, return_mm3, formula='normalised'):
- """
- Generalised normalised symmetric contour distance.
+def symmetric_contour_distance_one_label(im1, im2, lab, return_mm3, formula="normalised"):
+ r"""Generalised normalised symmetric contour distance.
On the sets {d(x, contourY)) | x in contourX} and {d(y, contourX)) | y in contourY}, several statistics
can be computed. Mean, median and standard deviation can be useful, as well as a more robust normalisation.
Formula can be
@@ -173,14 +171,15 @@ def symmetric_contour_distance_one_label(im1, im2, lab, return_mm3, formula='nor
:param lab:
:param return_mm3:
:param formula: 'normalised', 'averaged', 'median', 'std'
- 'normalised' = (\sum_{x in contourX} d(x, contourY)) + \sum_{y in contourY} d(y, contourX))) / (|contourX| + |contourY|)
+ 'normalised' = (\sum_{x in contourX} d(x, contourY))
+ + \sum_{y in contourY} d(y, contourX))) / (|contourX| + |contourY|)
'averaged' = 0.5 (mean({d(x, contourY)) | x in contourX}) + mean({d(y, contourX)) | y in contourY}))
'median' = 0.5 (median({d(x, contourY)) | x in contourX}) + median({d(y, contourX)) | y in contourY}))
'std' = 0.5 \sqrt(std({d(x, contourY)) | x in contourX})^2 + std({d(y, contourX)) | y in contourY})^2)
:return:
"""
- arr1 = im1.get_data() == lab
- arr2 = im2.get_data() == lab
+ arr1 = im1.get_fdata() == lab
+ arr2 = im2.get_fdata() == lab
if np.count_nonzero(arr1) == 0 or np.count_nonzero(arr2) == 0:
return np.nan
@@ -201,26 +200,28 @@ def symmetric_contour_distance_one_label(im1, im2, lab, return_mm3, formula='nor
dist_border1_array2 = dist_border1_array2[dist_border1_array2 > 0]
dist_border2_array1 = dist_border2_array1[dist_border2_array1 > 0]
- if formula == 'normalised':
- return (np.sum(dist_border1_array2) + np.sum(dist_border2_array1)) / float(np.count_nonzero(arr1_contour) + np.count_nonzero(arr2_contour))
- elif formula == 'averaged':
- return .5 * (np.mean(dist_border1_array2) + np.mean(dist_border2_array1))
- elif formula == 'median':
- return .5 * (np.median(dist_border1_array2) + np.median(dist_border2_array1))
- elif formula == 'std':
- return np.sqrt(.5 * (np.std(dist_border1_array2)**2 + np.std(dist_border2_array1)**2))
- elif formula == 'average_std':
- return .5 * (np.mean(dist_border1_array2) + np.mean(dist_border2_array1)), \
- np.sqrt(.5 * (np.std(dist_border1_array2) ** 2 + np.std(dist_border2_array1) ** 2))
- else:
- raise IOError('adf')
+ if formula == "normalised":
+ return (np.sum(dist_border1_array2) + np.sum(dist_border2_array1)) / float(
+ np.count_nonzero(arr1_contour) + np.count_nonzero(arr2_contour),
+ )
+ if formula == "averaged":
+ return 0.5 * (np.mean(dist_border1_array2) + np.mean(dist_border2_array1))
+ if formula == "median":
+ return 0.5 * (np.median(dist_border1_array2) + np.median(dist_border2_array1))
+ if formula == "std":
+ return np.sqrt(0.5 * (np.std(dist_border1_array2) ** 2 + np.std(dist_border2_array1) ** 2))
+ if formula == "average_std":
+ return 0.5 * (np.mean(dist_border1_array2) + np.mean(dist_border2_array1)), np.sqrt(
+ 0.5 * (np.std(dist_border1_array2) ** 2 + np.std(dist_border2_array1) ** 2),
+ )
+ raise OSError("adf")
# --- distances - (segm, segm) |-> pandas.Series (indexed by labels)
+
def dice_score(im_segm1, im_segm2, labels_list, labels_names, verbose=1):
- """
- Dice score between paired labels of segmentations.
+ """Dice score between paired labels of segmentations.
:param im_segm1: nibabel image with labels
:param im_segm2: nibabel image with labels
:param labels_list:
@@ -233,14 +234,13 @@ def dice_score(im_segm1, im_segm2, labels_list, labels_names, verbose=1):
d = dice_score_one_label(im_segm1, im_segm2, l)
scores.append(d)
if verbose > 0:
- print(' Dice scores label {0} : {1} '.format(l, d))
+ print(f" Dice scores label {l} : {d} ")
- return pa.Series(scores, index=labels_names)
+ return pd.Series(scores, index=labels_names)
def covariance_distance(im_segm1, im_segm2, labels_list, labels_names, return_mm3=True, verbose=1, factor=100):
- """
- Considers the label as a point distribution in the space, and returns the covariance matrix of the points
+ """Considers the label as a point distribution in the space, and returns the covariance matrix of the points
distributions.
:return:
See: Herdin 2005, Correlation matrix distance, a meaningful measure for evaluation of non-stationary MIMO channels
@@ -253,14 +253,13 @@ def covariance_distance(im_segm1, im_segm2, labels_list, labels_names, return_mm
d = covariance_distance_between_matrices(a1, a2, mul_factor=factor)
cov_dist.append(d)
if verbose > 0:
- print(' Covariance distance label {0} : {1} '.format(l, d))
+ print(f" Covariance distance label {l} : {d} ")
- return pa.Series(np.array(cov_dist), index=labels_names)
+ return pd.Series(np.array(cov_dist), index=labels_names)
def hausdorff_distance(im_segm1, im_segm2, labels_list, labels_names, return_mm3=True, verbose=1):
- """
- From 2 segmentations sampled in overlapping grids (with affine in starndard form) it returns the hausdoroff
+ """From 2 segmentations sampled in overlapping grids (with affine in starndard form) it returns the hausdoroff
distance for each label in the labels list and list names it returns the pandas series with the corresponding
distances for each label.
:param im_segm1:
@@ -276,52 +275,86 @@ def hausdorff_distance(im_segm1, im_segm2, labels_list, labels_names, return_mm3
d = hausdorff_distance_one_label(im_segm1, im_segm2, l, return_mm3)
hausd_dist.append(d)
if verbose > 0:
- print(' Hausdoroff distance label {0} : {1} '.format(l, d))
+ print(f" Hausdoroff distance label {l} : {d} ")
- return pa.Series(np.array(hausd_dist), index=labels_names)
+ return pd.Series(np.array(hausd_dist), index=labels_names)
-def symmetric_contour_distance(im_segm1, im_segm2, labels_list, labels_names, return_mm3=True, verbose=1,
- formula='normalised'):
+def symmetric_contour_distance(
+ im_segm1,
+ im_segm2,
+ labels_list,
+ labels_names,
+ return_mm3=True,
+ verbose=1,
+ formula="normalised",
+):
nscd_dist = []
for l in labels_list:
d = symmetric_contour_distance_one_label(im_segm1, im_segm2, l, return_mm3, formula)
nscd_dist.append(d)
if verbose > 0:
- print(' {0}-SCD {1} : {2} '.format(formula, l, d))
+ print(f" {formula}-SCD {l} : {d} ")
- return pa.Series(np.array(nscd_dist), index=labels_names)
+ return pd.Series(np.array(nscd_dist), index=labels_names)
# --- variants over symmetric contour distance:
def normalised_symmetric_contour_distance(im_segm1, im_segm2, labels_list, labels_names, return_mm3=True, verbose=1):
- return symmetric_contour_distance(im_segm1, im_segm2, labels_list, labels_names,
- return_mm3=return_mm3, verbose=verbose, formula='normalised')
+ return symmetric_contour_distance(
+ im_segm1,
+ im_segm2,
+ labels_list,
+ labels_names,
+ return_mm3=return_mm3,
+ verbose=verbose,
+ formula="normalised",
+ )
def averaged_symmetric_contour_distance(im_segm1, im_segm2, labels_list, labels_names, return_mm3=True, verbose=1):
- return symmetric_contour_distance(im_segm1, im_segm2, labels_list, labels_names,
- return_mm3=return_mm3, verbose=verbose, formula='averaged')
+ return symmetric_contour_distance(
+ im_segm1,
+ im_segm2,
+ labels_list,
+ labels_names,
+ return_mm3=return_mm3,
+ verbose=verbose,
+ formula="averaged",
+ )
def median_symmetric_contour_distance(im_segm1, im_segm2, labels_list, labels_names, return_mm3=True, verbose=1):
- return symmetric_contour_distance(im_segm1, im_segm2, labels_list, labels_names,
- return_mm3=return_mm3, verbose=verbose, formula='median')
+ return symmetric_contour_distance(
+ im_segm1,
+ im_segm2,
+ labels_list,
+ labels_names,
+ return_mm3=return_mm3,
+ verbose=verbose,
+ formula="median",
+ )
def std_symmetric_contour_distance(im_segm1, im_segm2, labels_list, labels_names, return_mm3=True, verbose=1):
- return symmetric_contour_distance(im_segm1, im_segm2, labels_list, labels_names,
- return_mm3=return_mm3, verbose=verbose, formula='std')
+ return symmetric_contour_distance(
+ im_segm1,
+ im_segm2,
+ labels_list,
+ labels_names,
+ return_mm3=return_mm3,
+ verbose=verbose,
+ formula="std",
+ )
# --- extra:
def box_sides_length(im, labels_list, labels_names, return_mm3=True):
- """
- Length of the rectangular hull surrounding the labels in the given list.
+ """Length of the rectangular hull surrounding the labels in the given list.
The rectangle is parallel to the matrix coordinate system.
:param im: sampled on an orthogonal grid.
:param labels_list: list of labels
@@ -329,8 +362,8 @@ def box_sides_length(im, labels_list, labels_names, return_mm3=True):
:param return_mm3: if True the answer is provided in the real space coordinates.
:return: output pandas series. One row for each label.
"""
- def box_sides_length_l(arr, lab, scaling_factors):
+ def box_sides_length_l(arr, lab, scaling_factors):
if lab not in arr:
return np.nan
coordinates = np.where(arr == lab) # returns [X_vector, Y_vector, Z_vector]
@@ -338,5 +371,5 @@ def box_sides_length_l(arr, lab, scaling_factors):
coordinates = [d * dd for d, dd in zip(scaling_factors, coordinates)]
return [np.abs(np.max(coordinates[k]) - np.min(coordinates[k])) for k in range(len(coordinates))]
- boxes_values = [box_sides_length_l(im.get_data(), l, np.diag(im.affine)[:-1]) for l in labels_list]
- return pa.Series(boxes_values, index=labels_names)
+ boxes_values = [box_sides_length_l(im.get_fdata(), l, np.diag(im.affine)[:-1]) for l in labels_list]
+ return pd.Series(boxes_values, index=labels_names)
diff --git a/nilabels/tools/caliber/volumes_and_values.py b/nilabels/tools/caliber/volumes_and_values.py
index 1099d3f..5ba0b58 100644
--- a/nilabels/tools/caliber/volumes_and_values.py
+++ b/nilabels/tools/caliber/volumes_and_values.py
@@ -1,35 +1,33 @@
-"""
-This module is divided into two parts.
+"""This module is divided into two parts.
First one -> essential functions, input are nibabel objects, output are reals or arrays.
The first part refers to the number of voxels.
Second one -> it uses the first part, to plot volumes, normalisation outputs values in pandas arrays or dataframes.
"""
+
import numpy as np
-import pandas as pa
+import pandas as pd
from nilabels.tools.aux_methods.utils_nib import one_voxel_volume
def get_total_num_nonzero_voxels(im_segm, list_labels_to_exclude=None):
- """
- :param im_segm:
+ """:param im_segm:
:param list_labels_to_exclude:
:return:
"""
- seg = np.copy(im_segm.get_data())
+ seg = np.copy(im_segm.get_fdata())
if list_labels_to_exclude is not None:
for label_k in list_labels_to_exclude:
places = seg != label_k
seg = seg * places
num_voxels = np.count_nonzero(seg)
else:
- num_voxels = int(np.count_nonzero(im_segm.get_data()))
+ num_voxels = int(np.count_nonzero(im_segm.get_fdata()))
return num_voxels
def get_num_voxels_from_labels_list(im_segm, labels_list):
- """
- :param im_segm: image segmentation
+ """:param im_segm: image segmentation
:param labels_list: integer, list of labels [l1, l2, ..., ln], or list of list of labels if labels needs to be
considered together.
e.g. labels_list = [1,2,[3,4]] -> values below label 1, values below label 2, values below label 3 and 4.
@@ -39,22 +37,21 @@ def get_num_voxels_from_labels_list(im_segm, labels_list):
for k, label_k in enumerate(labels_list):
if isinstance(label_k, int):
- all_places = im_segm.get_data() == label_k
+ all_places = im_segm.get_fdata() == label_k
num_voxels_per_label[k] = np.count_nonzero(np.nan_to_num(all_places))
elif isinstance(label_k, list):
- all_places = np.zeros_like(im_segm.get_data(), dtype=np.bool)
+ all_places = np.zeros_like(im_segm.get_fdata(), dtype=bool)
for label_k_j in label_k:
- all_places += im_segm.get_data() == label_k_j
+ all_places += im_segm.get_fdata() == label_k_j
num_voxels_per_label[k] = np.count_nonzero(np.nan_to_num(all_places))
else:
- raise IOError('Labels list must be like [1,2,[3,4]], where [3, 4] are considered as a single label.')
+ raise OSError("Labels list must be like [1,2,[3,4]], where [3, 4] are considered as a single label.")
return num_voxels_per_label
def get_values_below_labels_list(im_segm, im_anat, labels_list):
- """
- :param im_segm: image segmentation
+ """:param im_segm: image segmentation
:param im_anat: anatomical image, corresponding to the segmentation.
:param labels_list: integer, list of labels [l1, l2, ..., ln], or list of list of labels if labels needs to be
considered together.
@@ -67,23 +64,22 @@ def get_values_below_labels_list(im_segm, im_anat, labels_list):
for label_k in labels_list:
if isinstance(label_k, int):
- coords = np.where(im_segm.get_data() == label_k)
- values_below_each_label.append(im_anat.get_data()[coords].flatten())
+ coords = np.where(im_segm.get_fdata() == label_k)
+ values_below_each_label.append(im_anat.get_fdata()[coords].flatten())
elif isinstance(label_k, list):
vals = np.array([])
for label_k_j in label_k:
- coords = np.where(im_segm.get_data() == label_k_j)
- vals = np.concatenate((vals, im_anat.get_data()[coords].flatten()), axis=0)
+ coords = np.where(im_segm.get_fdata() == label_k_j)
+ vals = np.concatenate((vals, im_anat.get_fdata()[coords].flatten()), axis=0)
values_below_each_label.append(vals)
else:
- raise IOError('Labels list must be like [1,2,[3,4]], where [3, 4] are considered as a single label.')
+ raise OSError("Labels list must be like [1,2,[3,4]], where [3, 4] are considered as a single label.")
return values_below_each_label
def get_volumes_per_label(im_segm, labels, labels_names, tot_volume_prior=None, verbose=0):
- """
- Get a separate volume for each label in a data-frame
+ """Get a separate volume for each label in a data-frame
:param im_segm: nibabel segmentation
:param labels: labels you want to measure, or 'all' if you want them all or 'tot' to have the total of the non zero
labels.
@@ -96,58 +92,63 @@ def get_volumes_per_label(im_segm, labels, labels_names, tot_volume_prior=None,
vol_non_zero_voxels_mm3 = num_non_zero_voxels * one_voxel_volume(im_segm)
if tot_volume_prior is None:
tot_volume_prior = vol_non_zero_voxels_mm3
- if labels_names not in [None, 'all', 'tot']:
- if len(labels) != len(labels_names):
- raise IOError('Inconsistent labels - labels_names input.')
+ if labels_names not in [None, "all", "tot"] and len(labels) != len(labels_names):
+ raise OSError("Inconsistent labels - labels_names input.")
- if labels_names == 'all':
- labels_names = ['reg {}'.format(l) for l in labels]
+ if labels_names == "all":
+ labels_names = [f"reg {l}" for l in labels]
- if labels_names == 'tot':
- labels_names = ['tot']
+ if labels_names == "tot":
+ labels_names = ["tot"]
- non_zero_voxels = np.count_nonzero(im_segm.get_data())
+ non_zero_voxels = np.count_nonzero(im_segm.get_fdata())
volumes = non_zero_voxels * one_voxel_volume(im_segm)
vol_over_tot = volumes / float(tot_volume_prior)
- data_frame = pa.DataFrame({'Num voxels': pa.Series([non_zero_voxels], index=labels_names),
- 'Volume': pa.Series([volumes], index=labels_names),
- 'Vol over Tot': pa.Series([vol_over_tot], index=labels_names)})
- return data_frame
+ return pd.DataFrame(
+ {
+ "Num voxels": pd.Series([non_zero_voxels], index=labels_names),
+ "Volume": pd.Series([volumes], index=labels_names),
+ "Vol over Tot": pd.Series([vol_over_tot], index=labels_names),
+ },
+ )
- else:
- non_zero_voxels_list = []
- volumes_list = []
- vol_over_tot_list = []
+ non_zero_voxels_list = []
+ volumes_list = []
+ vol_over_tot_list = []
- for label_k in labels:
- all_places = np.zeros_like(im_segm.get_data(), dtype=np.bool)
- if isinstance(label_k, int):
- all_places += im_segm.get_data() == label_k
- else:
- for label_k_j in label_k:
- all_places += im_segm.get_data() == label_k_j
+ for label_k in labels:
+ all_places = np.zeros_like(im_segm.get_fdata(), dtype=bool)
+ if isinstance(label_k, int):
+ all_places += im_segm.get_fdata() == label_k
+ else:
+ for label_k_j in label_k:
+ all_places += im_segm.get_fdata() == label_k_j
- flat_volume_voxel = np.nan_to_num((all_places.astype(np.float64)).flatten())
+ flat_volume_voxel = np.nan_to_num((all_places.astype(np.float64)).flatten())
- non_zero_voxels = np.count_nonzero(flat_volume_voxel)
- volumes = non_zero_voxels * one_voxel_volume(im_segm)
+ non_zero_voxels = np.count_nonzero(flat_volume_voxel)
+ volumes = non_zero_voxels * one_voxel_volume(im_segm)
- vol_over_tot = volumes / float(tot_volume_prior)
+ vol_over_tot = volumes / float(tot_volume_prior)
- non_zero_voxels_list.append(non_zero_voxels)
- volumes_list.append(volumes)
- vol_over_tot_list.append(vol_over_tot)
+ non_zero_voxels_list.append(non_zero_voxels)
+ volumes_list.append(volumes)
+ vol_over_tot_list.append(vol_over_tot)
- data_frame = pa.DataFrame({'Label' : pa.Series(labels, index=labels_names),
- 'Num voxels' : pa.Series(non_zero_voxels_list, index=labels_names),
- 'Volume' : pa.Series(volumes_list, index=labels_names),
- 'Vol over Tot' : pa.Series(vol_over_tot_list, index=labels_names)})
+ data_frame = pd.DataFrame(
+ {
+ "Label": pd.Series(labels, index=labels_names),
+ "Num voxels": pd.Series(non_zero_voxels_list, index=labels_names),
+ "Volume": pd.Series(volumes_list, index=labels_names),
+ "Vol over Tot": pd.Series(vol_over_tot_list, index=labels_names),
+ },
+ )
- data_frame = data_frame.rename_axis('Region')
- data_frame = data_frame.reset_index()
+ data_frame = data_frame.rename_axis("Region")
+ data_frame = data_frame.reset_index()
- if verbose > 0:
- print(data_frame)
+ if verbose > 0:
+ print(data_frame)
- return data_frame
+ return data_frame
diff --git a/nilabels/tools/cleaning/labels_cleaner.py b/nilabels/tools/cleaning/labels_cleaner.py
index 7c34fc8..3a02add 100644
--- a/nilabels/tools/cleaning/labels_cleaner.py
+++ b/nilabels/tools/cleaning/labels_cleaner.py
@@ -5,8 +5,7 @@
def multi_lab_segmentation_dilate_1_above_selected_label(arr_segm, selected_label=-1, labels_to_dilate=(), verbose=2):
- """
- The orders of labels to dilate counts.
+ """The orders of labels to dilate counts.
:param arr_segm:
:param selected_label:
:param labels_to_dilate: if None all labels are dilated, in ascending order (algorithm is NOT order invariant).
@@ -14,30 +13,29 @@ def multi_lab_segmentation_dilate_1_above_selected_label(arr_segm, selected_labe
:return:
"""
answer = np.copy(arr_segm)
- if labels_to_dilate is ():
- labels_to_dilate = sorted(list(set(arr_segm.flat) - {selected_label}))
+ if labels_to_dilate == ():
+ labels_to_dilate = sorted(set(arr_segm.flat) - {selected_label})
num_labels_dilated = 0
for l in labels_to_dilate:
if verbose > 1:
- print('Dilating label {} over hole-label {}'.format(l, selected_label))
- selected_labels_mask = np.zeros_like(answer, dtype=np.bool)
+ print(f"Dilating label {l} over hole-label {selected_label}")
+ selected_labels_mask = np.zeros_like(answer, dtype=bool)
selected_labels_mask[answer == selected_label] = 1
- bin_label_l = np.zeros_like(answer, dtype=np.bool)
+ bin_label_l = np.zeros_like(answer, dtype=bool)
bin_label_l[answer == l] = 1
dilated_bin_label_l = ndimage.morphology.binary_dilation(bin_label_l)
dilation_l_over_selected_label = dilated_bin_label_l * selected_labels_mask
answer[dilation_l_over_selected_label > 0] = l
num_labels_dilated += 1
if verbose > 0:
- print('Number of labels_dilated: {}\n'.format(num_labels_dilated))
+ print(f"Number of labels_dilated: {num_labels_dilated}\n")
return answer
def holes_filler(arr_segm_with_holes, holes_label=-1, labels_sequence=(), verbose=1):
- """
- Given a segmentation with holes (holes are specified by a special labels called holes_label)
+ """Given a segmentation with holes (holes are specified by a special labels called holes_label)
the holes are filled with the closest labels around.
It applies multi_lab_segmentation_dilate_1_above_selected_label until all the holes
are filled.
@@ -52,22 +50,22 @@ def holes_filler(arr_segm_with_holes, holes_label=-1, labels_sequence=(), verbos
arr_segm_no_holes = np.copy(arr_segm_with_holes)
if verbose:
- print('Filling holes in the segmentation:')
+ print("Filling holes in the segmentation:")
while holes_label in arr_segm_no_holes:
- arr_segm_no_holes = multi_lab_segmentation_dilate_1_above_selected_label(arr_segm_no_holes,
- selected_label=holes_label, labels_to_dilate=labels_sequence)
+ arr_segm_no_holes = multi_lab_segmentation_dilate_1_above_selected_label(
+ arr_segm_no_holes, selected_label=holes_label, labels_to_dilate=labels_sequence,
+ )
num_rounds += 1
if verbose:
- print('Number of dilations required to remove the holes: {}'.format(num_rounds))
+ print(f"Number of dilations required to remove the holes: {num_rounds}")
return arr_segm_no_holes
def clean_semgentation(arr_segm, labels_to_clean=(), label_for_holes=-1, verbose=1):
- """
- Given an array representing a binary segmentation, the connected components of the segmentations.
+ """Given an array representing a binary segmentation, the connected components of the segmentations.
If an hole could be filled by 2 different labels, wins the label with lower value.
This should be improved with a probabilistic framework [future work].
Only the largest connected component of each label will remain in the final
@@ -84,13 +82,13 @@ def clean_semgentation(arr_segm, labels_to_clean=(), label_for_holes=-1, verbose
:return:
"""
if labels_to_clean == ():
- labels_to_clean = sorted(list(set(arr_segm.flat)))
+ labels_to_clean = sorted(set(arr_segm.flat))
labels_to_clean = [[z, 1] for z in labels_to_clean]
segm_with_holes = np.copy(arr_segm)
for lab, num_components in labels_to_clean:
if verbose:
- print('Cleaning label {}, keeping {} components'.format(lab, num_components))
+ print(f"Cleaning label {lab}, keeping {num_components} components")
islands = island_for_label(arr_segm, lab, m=num_components, special_label=label_for_holes)
segm_with_holes[islands == label_for_holes] = label_for_holes
diff --git a/nilabels/tools/detections/check_imperfections.py b/nilabels/tools/detections/check_imperfections.py
index d88d483..66f1041 100644
--- a/nilabels/tools/detections/check_imperfections.py
+++ b/nilabels/tools/detections/check_imperfections.py
@@ -1,14 +1,13 @@
import numpy as np
-import pandas as pa
+import pandas as pd
from skimage import measure
-from nilabels.tools.aux_methods.utils_nib import one_voxel_volume
from nilabels.tools.aux_methods.label_descriptor_manager import LabelsDescriptorManager
+from nilabels.tools.aux_methods.utils_nib import one_voxel_volume
def check_missing_labels(im_segm, labels_descriptor, pfi_where_log=None):
- """
- :param im_segm: nibabel image of a segmentation.
+ """:param im_segm: nibabel image of a segmentation.
:param labels_descriptor: instance of LabelsDescriptorManager
:param pfi_where_log: path to file where to save a log of the output.
:return: correspondences between labels in the segmentation and labels in the label descriptors,
@@ -18,14 +17,14 @@ def check_missing_labels(im_segm, labels_descriptor, pfi_where_log=None):
labels_dict = labels_descriptor.get_dict_itk_snap()
labels_list_from_descriptor = labels_dict.keys()
- labels_in_the_image = set(im_segm.get_data().astype(np.int).flatten())
+ labels_in_the_image = set(im_segm.get_fdata().astype(int).flatten())
intersection = labels_in_the_image & set(labels_list_from_descriptor)
in_descriptor_not_delineated = set(labels_list_from_descriptor) - intersection
delineated_not_in_descriptor = labels_in_the_image - intersection
- msg = 'Labels in the descriptor not delineated: \n{}\n'.format(in_descriptor_not_delineated)
- msg += 'Labels delineated not in the descriptor: \n{}'.format(delineated_not_in_descriptor)
+ msg = f"Labels in the descriptor not delineated: \n{in_descriptor_not_delineated}\n"
+ msg += f"Labels delineated not in the descriptor: \n{delineated_not_in_descriptor}"
print(msg)
if pfi_where_log is not None:
@@ -34,38 +33,40 @@ def check_missing_labels(im_segm, labels_descriptor, pfi_where_log=None):
num_voxels_per_label = []
num_connected_components_per_label = []
for label_k in labels_list_from_descriptor:
- all_places = im_segm.get_data() == label_k
+ all_places = im_segm.get_fdata() == label_k
- cc_l = len(set(list(measure.label(all_places, background=0).flatten()))) - 1
+ cc_l = len(set(measure.label(all_places, background=0).flatten())) - 1
num_connected_components_per_label.append(cc_l)
len_non_zero_places = len(all_places[np.where(all_places > 1e-6)])
if len_non_zero_places == 0:
- msg_l = '\nLabel {0} present in label descriptor and not delineated in the ' \
- 'given segmentation.'.format(label_k)
+ msg_l = (
+ f"\nLabel {label_k} present in label descriptor and not delineated in the " "given segmentation."
+ )
msg += msg_l
print(msg_l)
num_voxels_per_label.append(len_non_zero_places)
- se_voxels = pa.Series(num_voxels_per_label, index=labels_names)
- se_volume = pa.Series(one_voxel_volume(im_segm) * np.array(num_voxels_per_label), index=labels_names)
+ se_voxels = pd.Series(num_voxels_per_label, index=labels_names)
+ se_volume = pd.Series(one_voxel_volume(im_segm) * np.array(num_voxels_per_label), index=labels_names)
- df = pa.DataFrame({'Num voxels' :se_voxels, 'Volumes' : se_volume,
- 'Connected components': num_connected_components_per_label})
+ df = pd.DataFrame(
+ {"Num voxels": se_voxels, "Volumes": se_volume, "Connected components": num_connected_components_per_label},
+ )
- df.index = df.index.map('{:<30}'.format)
- df['Num voxels'] = df['Num voxels'].map('{:<10}'.format)
- df['Volumes'] = df['Volumes'].map('{:<10}'.format)
- df['Connected components'] = df['Connected components'].map('{:<10}'.format)
+ df.index = df.index.map("{:<30}".format)
+ df["Num voxels"] = df["Num voxels"].map("{:<10}".format)
+ df["Volumes"] = df["Volumes"].map("{:<10}".format)
+ df["Connected components"] = df["Connected components"].map("{:<10}".format)
f = open(pfi_where_log, "w+")
df.to_string(f)
f.close()
f = open(pfi_where_log, "a+")
- f.write('\n\n' + msg)
+ f.write("\n\n" + msg)
f.close()
- print('Log status saved in {}'.format(pfi_where_log))
+ print(f"Log status saved in {pfi_where_log}")
return in_descriptor_not_delineated, delineated_not_in_descriptor
diff --git a/nilabels/tools/detections/contours.py b/nilabels/tools/detections/contours.py
index c99b162..7688e48 100644
--- a/nilabels/tools/detections/contours.py
+++ b/nilabels/tools/detections/contours.py
@@ -5,8 +5,7 @@
def contour_from_array_at_label(im_arr, lab, thr=0.3, omit_axis=None, verbose=0):
- """
- Get the contour of a single label
+ """Get the contour of a single label
:param im_arr: input array with segmentation
:param lab: considered label
:param thr: threshold (default 0.3) increase to increase the contour thickness.
@@ -16,26 +15,25 @@ def contour_from_array_at_label(im_arr, lab, thr=0.3, omit_axis=None, verbose=0)
:return: boolean mask with the array labels.
"""
if verbose > 0:
- print('Getting contour for label {}'.format(lab))
+ print(f"Getting contour for label {lab}")
array_label_l = im_arr == lab
assert isinstance(array_label_l, np.ndarray)
- gra = np.gradient(array_label_l.astype(np.bool).astype(np.float64))
+ gra = np.gradient(array_label_l.astype(bool).astype(np.float64))
if omit_axis is None:
thresholded_gra = np.sqrt(gra[0] ** 2 + gra[1] ** 2 + gra[2] ** 2) > thr
- elif omit_axis == 'x':
+ elif omit_axis == "x":
thresholded_gra = np.sqrt(gra[1] ** 2 + gra[2] ** 2) > thr
- elif omit_axis == 'y':
+ elif omit_axis == "y":
thresholded_gra = np.sqrt(gra[0] ** 2 + gra[2] ** 2) > thr
- elif omit_axis == 'z':
+ elif omit_axis == "z":
thresholded_gra = np.sqrt(gra[0] ** 2 + gra[1] ** 2) > thr
else:
- raise IOError
+ raise OSError
return thresholded_gra
def contour_from_segmentation(im_segm, omit_axis=None, verbose=0):
- """
- From an input nibabel image segmentation, returns the contour of each segmented region with the original
+ """From an input nibabel image segmentation, returns the contour of each segmented region with the original
label.
:param im_segm:
:param omit_axis: a directional axis preference for the contour creation, to avoid "walls" when scrolling
@@ -43,18 +41,17 @@ def contour_from_segmentation(im_segm, omit_axis=None, verbose=0):
:param verbose: 0 no, 1 yes.
:return: return the contour of the provided segmentation
"""
- list_labels = sorted(list(set(im_segm.get_data().flat)))[1:]
- output_arr = np.zeros_like(im_segm.get_data(), dtype=im_segm.get_data_dtype())
+ list_labels = sorted(set(im_segm.get_fdata().flat))[1:]
+ output_arr = np.zeros_like(im_segm.get_fdata(), dtype=im_segm.get_data_dtype())
for la in list_labels:
- output_arr += contour_from_array_at_label(im_segm.get_data(), la, omit_axis=omit_axis, verbose=verbose)
+ output_arr += contour_from_array_at_label(im_segm.get_fdata(), la, omit_axis=omit_axis, verbose=verbose)
- return set_new_data(im_segm, output_arr.astype(np.bool) * im_segm.get_data(), new_dtype=im_segm.get_data_dtype())
+ return set_new_data(im_segm, output_arr.astype(bool) * im_segm.get_fdata(), new_dtype=im_segm.get_data_dtype())
def get_xyz_borders_of_a_label(segm_arr, label):
- """
- :param segm_arr: array representing a segmentation
+ """:param segm_arr: array representing a segmentation
:param label: a single integer label
:return: box coordinates containing the given label in the segmentation, None if the label is not present.
"""
@@ -68,13 +65,13 @@ def get_xyz_borders_of_a_label(segm_arr, label):
def get_internal_contour_with_erosion_at_label(segm_arr, lab, thickness=1):
- """
- Get the internal contour for a given thickness.
+ """Get the internal contour for a given thickness.
:param segm_arr: input segmentation where to extract the contour
:param lab: label to extract the contour
:param thickness: final thickness of the segmentation
:return: image with only the contour of the given input image.
"""
im_lab = segm_arr == lab
- return (im_lab ^ nd.morphology.binary_erosion(im_lab, iterations=thickness).astype(np.bool)).astype(segm_arr.dtype) * lab
-
+ return (im_lab ^ nd.morphology.binary_erosion(im_lab, iterations=thickness).astype(bool)).astype(
+ segm_arr.dtype,
+ ) * lab
diff --git a/nilabels/tools/detections/get_segmentation.py b/nilabels/tools/detections/get_segmentation.py
index ce6ee73..feba72a 100644
--- a/nilabels/tools/detections/get_segmentation.py
+++ b/nilabels/tools/detections/get_segmentation.py
@@ -1,9 +1,8 @@
import numpy as np
-import matplotlib.mlab as mlab
+from matplotlib import mlab
from matplotlib import pyplot as plt
-
-from sklearn.mixture import GaussianMixture
from scipy.signal import medfilt
+from sklearn.mixture import GaussianMixture
try:
from skimage import filters
@@ -14,8 +13,7 @@
def intensity_segmentation(in_array, num_levels=5):
- """
- Simplest way of getting an intensity based segmentation.
+ """Simplest way of getting an intensity based segmentation.
:param in_array: image data in a numpy array.
:param num_levels: maximum allowed 65535 - 1.
:return: segmentation of the result in levels levels based on the intensities of the in_data.
@@ -24,39 +22,44 @@ def intensity_segmentation(in_array, num_levels=5):
min_data = np.min(in_array)
max_data = np.max(in_array)
h = (max_data - min_data) / float(int(num_levels))
- for k in range(0, num_levels):
+ for k in range(num_levels):
places = (min_data + k * h <= in_array) * (in_array < min_data + (k + 1) * h)
np.place(segm, places, k)
places = in_array == max_data
- np.place(segm, places, num_levels-1)
+ np.place(segm, places, num_levels - 1)
return segm
-def otsu_threshold(in_array, side='above', return_as_mask=True):
- """
- Segmentation of an array with Otsu thresholding parameters from skimage filters.
+def otsu_threshold(in_array, side="above", return_as_mask=True):
+ """Segmentation of an array with Otsu thresholding parameters from skimage filters.
:param in_array: input array representing an rgb image.
:param side: must be 'above' or 'below', representing the side of the image thresholded after Otsu response.
:param return_as_mask: the output can be a boolean mask if True.
:return: thresholded input image according to Otsu and input parameters.
"""
otsu_thr = filters.threshold_otsu(in_array)
- if side == 'above':
+ if side == "above":
new_data = in_array * (in_array >= otsu_thr)
- elif side == 'below':
+ elif side == "below":
new_data = in_array * (in_array < otsu_thr)
else:
- raise IOError("Parameter side must be 'above' or 'below'.")
+ raise OSError("Parameter side must be 'above' or 'below'.")
if return_as_mask:
- new_data = new_data.astype(np.bool)
+ new_data = new_data.astype(bool)
return new_data
-def MoG_array(in_array, K=None, mask_array=None, pre_process_median_filter=False,
- output_gmm_class=False, pre_process_only_interquartile=False,
- see_histogram=None, reorder_mus=True):
- """
- Mixture of gaussians for medical images. A simple wrap of
+def MoG_array(
+ in_array,
+ K=None,
+ mask_array=None,
+ pre_process_median_filter=False,
+ output_gmm_class=False,
+ pre_process_only_interquartile=False,
+ see_histogram=None,
+ reorder_mus=True,
+):
+ """Mixture of gaussians for medical images. A simple wrap of
sklearn.mixture.GaussianMixture to get a mog-based segmentation of an input
nibabel image.
:param in_array: input array format to be segmented with a MOG method.
@@ -70,7 +73,7 @@ def MoG_array(in_array, K=None, mask_array=None, pre_process_median_filter=False
:return: [c, p] crisp and probabilistic segmentation OR gmm, instance of the class sklearn.mixture.GaussianMixture.
"""
if pre_process_median_filter:
- print('Pre-process with a median filter.')
+ print("Pre-process with a median filter.")
data = medfilt(in_array)
else:
data = in_array
@@ -81,50 +84,50 @@ def MoG_array(in_array, K=None, mask_array=None, pre_process_median_filter=False
data = mask_data * data
if pre_process_only_interquartile:
- print('Get only interquartile data.')
+ print("Get only interquartile data.")
non_zero_data = data[np.where(np.nan_to_num(data) > 1e-6)]
low_p = np.percentile(non_zero_data, 25)
high_p = np.percentile(non_zero_data, 75)
data = (data > low_p) * (data < high_p) * data
if K is None:
- print('Estimating numbers of components with BIC criterion... may take some minutes')
+ print("Estimating numbers of components with BIC criterion... may take some minutes")
n_components = range(3, 15)
models = [GaussianMixture(n_components=k, random_state=0).fit(data) for k in n_components]
K = np.min([m.bic(data) for m in models])
- print('Estimated number of classes according to BIC: {}'.format(K))
+ print(f"Estimated number of classes according to BIC: {K}")
gmm = GaussianMixture(n_components=K).fit(data)
if output_gmm_class:
return gmm
- else:
- crisp = gmm.predict(data).reshape(in_array.shape)
- prob = gmm.predict_proba(data).reshape(list(in_array.shape) + [K])
-
- if reorder_mus:
- mu = gmm.means_.reshape(-1)
- p = list(np.argsort(mu))
-
- old_labels = list(range(K))
- new_labels = [p.index(l) for l in old_labels] # the inverse of p
-
- crisp = np.copy(relabeller(crisp, old_labels, new_labels))
- prob = np.stack([prob[..., t] for t in new_labels], axis=3)
-
- if see_histogram is not None and see_histogram is not False:
- fig = plt.figure()
- ax = fig.add_subplot(111)
- ax.set_aspect(1)
- ax.hist(crisp.flatten(), bins=50, normed=True)
- lx = ax.get_xlim()
- x = np.arange(lx[0], lx[1], (lx[1] - lx[0]) / 1000.)
- for m, s in zip(gmm.means_, gmm.precisions_.reshape(-1)):
- ax.plot(x, mlab.normpdf(x, m, s))
-
- if isinstance(see_histogram, str):
- plt.savefig(see_histogram)
- else:
- plt.show()
-
- return crisp, prob
+
+ crisp = gmm.predict(data).reshape(in_array.shape)
+ prob = gmm.predict_proba(data).reshape(list(in_array.shape) + [K])
+
+ if reorder_mus:
+ mu = gmm.means_.reshape(-1)
+ p = list(np.argsort(mu))
+
+ old_labels = list(range(K))
+ new_labels = [p.index(l) for l in old_labels] # the inverse of p
+
+ crisp = np.copy(relabeller(crisp, old_labels, new_labels))
+ prob = np.stack([prob[..., t] for t in new_labels], axis=3)
+
+ if see_histogram is not None and see_histogram is not False:
+ fig = plt.figure()
+ ax = fig.add_subplot(111)
+ ax.set_aspect(1)
+ ax.hist(crisp.flatten(), bins=50, normed=True)
+ lx = ax.get_xlim()
+ x = np.arange(lx[0], lx[1], (lx[1] - lx[0]) / 1000.0)
+ for m, s in zip(gmm.means_, gmm.precisions_.reshape(-1)):
+ ax.plot(x, mlab.normpdf(x, m, s))
+
+ if isinstance(see_histogram, str):
+ plt.savefig(see_histogram)
+ else:
+ plt.show()
+
+ return crisp, prob
diff --git a/nilabels/tools/detections/island_detection.py b/nilabels/tools/detections/island_detection.py
index ef7efc0..bb8735e 100644
--- a/nilabels/tools/detections/island_detection.py
+++ b/nilabels/tools/detections/island_detection.py
@@ -1,3 +1,5 @@
+import logging
+
import numpy as np
from scipy import ndimage
@@ -5,8 +7,7 @@
def island_for_label(array_segm, label, m=0, special_label=-1):
- """
- As ndimage.label, with output ordered by the size of the connected component.
+ """As ndimage.label, with output ordered by the size of the connected component.
:param array_segm:
:param label:
:param m: integer. If m = 0 the n connected components will be numbered from 1 (biggest) to n (smallest).
@@ -20,9 +21,9 @@ def island_for_label(array_segm, label, m=0, special_label=-1):
If m = 0 it returns the components labelled from 1 to 4, where 1 is the biggest.
if m = 2 the first two largest components are numbered 1 and 2, and the remaining 2 are labelled with special_label.
"""
-
if label not in array_segm:
- print('Label {} not in the provided array.'.format(label))
+ msg = f"Label {label} not in the provided array."
+ logging.info(msg)
return array_segm
binary_segm_comp, num_comp = ndimage.label(array_segm == label)
diff --git a/nilabels/tools/image_colors_manipulations/cutter.py b/nilabels/tools/image_colors_manipulations/cutter.py
index 496f6fb..639f5c7 100644
--- a/nilabels/tools/image_colors_manipulations/cutter.py
+++ b/nilabels/tools/image_colors_manipulations/cutter.py
@@ -4,8 +4,7 @@
def cut_4d_volume_with_a_1_slice_mask(data_4d, data_mask):
- """
- Fist slice maks is applied to all the timepoints of the volume.
+ """Fist slice maks is applied to all the timepoints of the volume.
:param data_4d:
:param data_mask:
:return:
@@ -23,23 +22,20 @@ def cut_4d_volume_with_a_1_slice_mask(data_4d, data_mask):
def cut_4d_volume_with_a_1_slice_mask_nib(input_4d_nib, input_mask_nib):
- """
- Fist slice maks is applied to all the timepoints of the nibabel image.
+ """Fist slice maks is applied to all the timepoints of the nibabel image.
:param input_4d_nib: input 4d nifty image
:param input_mask_nib: input mask
:return:
"""
-
- data_4d = input_4d_nib.get_data()
- data_mask = input_mask_nib.get_data()
+ data_4d = input_4d_nib.get_fdata()
+ data_mask = input_mask_nib.get_fdata()
ans = cut_4d_volume_with_a_1_slice_mask(data_4d, data_mask)
return set_new_data(input_4d_nib, ans)
def apply_a_mask_nib(im_input, im_mask):
- """
- Set to zero all the values outside the mask.
+ """Set to zero all the values outside the mask.
From nibabel input and output.
Adaptative - if the mask is 3D and the image is 4D, will create a temporary mask,
generate the stack of masks, and apply the stacks to the image.
@@ -50,16 +46,15 @@ def apply_a_mask_nib(im_input, im_mask):
assert len(im_mask.shape) == 3
# TODO correct this: merge the cut_4d_volume_with_a_1_slice_mask here
- if not im_mask.shape == im_input.shape[:3]:
- msg = 'Provided mask and image does not have compatible dimension: {0} and {1}'.format(
- im_input.shape, im_mask.shape)
- raise IOError(msg)
+ if im_mask.shape != im_input.shape[:3]:
+ msg = f"Provided mask and image does not have compatible dimension: {im_input.shape} and {im_mask.shape}"
+ raise OSError(msg)
if len(im_input.shape) == 3:
- new_data = im_input.get_data() * im_mask.get_data().astype(np.bool)
+ new_data = im_input.get_fdata() * im_mask.get_fdata().astype(bool)
else:
- new_data = np.zeros_like(im_input.get_data())
+ new_data = np.zeros_like(im_input.get_fdata())
for t in range(im_input.shape[3]):
- new_data[..., t] = im_input.get_data()[..., t] * im_mask.get_data().astype(np.bool)
+ new_data[..., t] = im_input.get_fdata()[..., t] * im_mask.get_fdata().astype(bool)
return set_new_data(image=im_input, new_data=new_data)
diff --git a/nilabels/tools/image_colors_manipulations/normaliser.py b/nilabels/tools/image_colors_manipulations/normaliser.py
index 8eb7974..bc186cb 100644
--- a/nilabels/tools/image_colors_manipulations/normaliser.py
+++ b/nilabels/tools/image_colors_manipulations/normaliser.py
@@ -1,15 +1,10 @@
import numpy as np
-try:
- from skimage import filters
-except ImportError:
- from skimage import filter as filters
from nilabels.tools.aux_methods.utils_nib import set_new_data
def normalise_below_labels(im_input, im_segm, labels_list=None, stats=np.median, exclude_first_label=True):
- """
- Normalise the intensities of im_input for the stats obtained of the values found under input labels.
+ """Normalise the intensities of im_input for the stats obtained of the values found under input labels.
:param im_input: nibabel image
:param im_segm: nibabel segmentation
:param labels_list: list of labels you want to normalise below
@@ -20,26 +15,29 @@ def normalise_below_labels(im_input, im_segm, labels_list=None, stats=np.median,
if labels_list is not None:
if exclude_first_label:
labels_list = labels_list[1:]
- mask_data = np.zeros_like(im_segm.get_data(), dtype=np.bool)
+ mask_data = np.zeros_like(im_segm.get_fdata(), dtype=bool)
for label_k in labels_list:
- mask_data += im_segm.get_data() == label_k
+ mask_data += im_segm.get_fdata() == label_k
else:
- mask_data = np.zeros_like(im_segm.get_data(), dtype=np.bool)
- mask_data[im_segm.get_data() > 0] = 1
+ mask_data = np.zeros_like(im_segm.get_fdata(), dtype=bool)
+ mask_data[im_segm.get_fdata() > 0] = 1
- masked_im_data = np.nan_to_num((mask_data.astype(np.float64) * im_input.get_data().astype(np.float64)).flatten())
+ masked_im_data = np.nan_to_num((mask_data.astype(np.float64) * im_input.get_fdata().astype(np.float64)).flatten())
non_zero_masked_im_data = masked_im_data[np.where(masked_im_data > 1e-6)]
s = stats(non_zero_masked_im_data)
assert isinstance(s, float)
- output_im = set_new_data(im_input, (1 / float(s)) * im_input.get_data())
-
- return output_im
-
-
-def intensities_normalisation_linear(im_input, im_segm, im_mask_foreground=None,
- toll=1e-12, percentile_range=(1, 99), output_range=(0.1, 10)):
- """
- Normalise the values below the binarised segmentation so that the normalised image
+ return set_new_data(im_input, (1 / float(s)) * im_input.get_fdata())
+
+
+def intensities_normalisation_linear(
+ im_input,
+ im_segm,
+ im_mask_foreground=None,
+ toll=1e-12,
+ percentile_range=(1, 99),
+ output_range=(0.1, 10),
+):
+ """Normalise the values below the binarised segmentation so that the normalised image
will be between 0 and 1, based on a linear transformation whose parameters are learned
from the values below the segmentation.
E.G.
@@ -57,10 +55,10 @@ def intensities_normalisation_linear(im_input, im_segm, im_mask_foreground=None,
f(x) = 0 in the background.
:return:
"""
- mask_data = np.zeros_like(im_segm.get_data(), dtype=np.bool)
- mask_data[im_segm.get_data() > 0] = 1
+ mask_data = np.zeros_like(im_segm.get_fdata(), dtype=bool)
+ mask_data[im_segm.get_fdata() > 0] = 1
- non_zero_below_mask = im_input.get_data()[np.where(mask_data > toll)].flatten()
+ non_zero_below_mask = im_input.get_fdata()[np.where(mask_data > toll)].flatten()
min_intensities = np.percentile(non_zero_below_mask, percentile_range[0])
max_intensities = np.percentile(non_zero_below_mask, percentile_range[1])
@@ -69,31 +67,30 @@ def intensities_normalisation_linear(im_input, im_segm, im_mask_foreground=None,
b = output_range[0] - a * min_intensities
if im_mask_foreground is None:
- im_mask_foreground_data = np.ones_like(im_input.get_data())
+ im_mask_foreground_data = np.ones_like(im_input.get_fdata())
else:
- im_mask_foreground_data = im_mask_foreground.get_data()
+ im_mask_foreground_data = im_mask_foreground.get_fdata()
- return set_new_data(im_input, im_mask_foreground_data * (a * im_input.get_data() + b))
+ return set_new_data(im_input, im_mask_foreground_data * (a * im_input.get_fdata() + b))
def mahalanobis_distance_map(im, im_mask=None, trim=False):
- """
- From an image to its Mahalanobis distance map
+ """From an image to its Mahalanobis distance map
:param im: input image acquired with some modality.
:param im_mask: considering only the data below the given mask.
:param trim: if mask is provided the output image is masked with zeros values outside the mask.
:return: nibabel image same shape as im, with the corresponding Mahalanobis map
"""
if im_mask is None:
- mu = np.mean(im.get_data().flatten())
- sigma2 = np.std(im.get_data().flatten())
- return set_new_data(im, np.sqrt((im.get_data() - mu) * sigma2 * (im.get_data() - mu)))
- else:
- np.testing.assert_array_equal(im.affine, im_mask.affine)
- np.testing.assert_array_equal(im.shape, im_mask.shape)
- mu = np.mean(im.get_data().flatten() * im_mask.get_data().flatten())
- sigma2 = np.std(im.get_data().flatten() * im_mask.get_data().flatten())
- new_data = np.sqrt((im.get_data() - mu) * sigma2**(-1) * (im.get_data() - mu))
- if trim:
- new_data = new_data * im_mask.get_data().astype(np.bool)
- return set_new_data(im, new_data)
+ mu = np.mean(im.get_fdata().flatten())
+ sigma2 = np.std(im.get_fdata().flatten())
+ return set_new_data(im, np.sqrt((im.get_fdata() - mu) * sigma2 * (im.get_fdata() - mu)))
+
+ np.testing.assert_array_equal(im.affine, im_mask.affine)
+ np.testing.assert_array_equal(im.shape, im_mask.shape)
+ mu = np.mean(im.get_fdata().flatten() * im_mask.get_fdata().flatten())
+ sigma2 = np.std(im.get_fdata().flatten() * im_mask.get_fdata().flatten())
+ new_data = np.sqrt((im.get_fdata() - mu) * sigma2 ** (-1) * (im.get_fdata() - mu))
+ if trim:
+ new_data = new_data * im_mask.get_fdata().astype(bool)
+ return set_new_data(im, new_data)
diff --git a/nilabels/tools/image_colors_manipulations/relabeller.py b/nilabels/tools/image_colors_manipulations/relabeller.py
index 78468b1..73533f0 100644
--- a/nilabels/tools/image_colors_manipulations/relabeller.py
+++ b/nilabels/tools/image_colors_manipulations/relabeller.py
@@ -1,26 +1,26 @@
import copy
+import logging
+
import numpy as np
from nilabels.tools.aux_methods.sanity_checks import is_valid_permutation
def relabeller(in_data, list_old_labels, list_new_labels, verbose=True):
- """
- :param in_data: array corresponding to an image segmentation.
+ """:param in_data: array corresponding to an image segmentation.
:param list_old_labels: list or tuple of labels
:param list_new_labels: list or tuple of labels of the same len as list_old_labels
:param verbose:
:return: array where all the labels in list_new_labels are substituted with list_new_label in the same order.
"""
-
if isinstance(list_new_labels, int):
- list_new_labels = [list_new_labels, ]
+ list_new_labels = [list_new_labels]
if isinstance(list_old_labels, int):
- list_old_labels = [list_old_labels, ]
+ list_old_labels = [list_old_labels]
# sanity check: old and new must have the same number of elements
if not len(list_old_labels) == len(list_new_labels):
- raise IOError('Labels lists old and new do not have the same length.')
+ raise OSError("Labels lists old and new do not have the same length.")
new_data = copy.deepcopy(in_data)
@@ -29,79 +29,81 @@ def relabeller(in_data, list_old_labels, list_new_labels, verbose=True):
if np.any(places):
np.place(new_data, places, list_new_labels[k])
if verbose:
- print('Label {0} substituted with label {1}'.format(list_old_labels[k], list_new_labels[k]))
- else:
- if verbose:
- print('Label {0} not present in the array'.format(list_old_labels[k]))
+ msg = f"Label {list_old_labels[k]} substituted with label {list_new_labels[k]}"
+ logging.info(msg)
+ elif verbose:
+ msg = f"Label {list_old_labels[k]} not present in the array"
+ logging.info(msg)
return new_data
def permute_labels(in_data, permutation):
- """
- Permute the values of the labels in an int image.
+ """Permute the values of the labels in an int image.
:param in_data: array corresponding to an image segmentation.
:param permutation:
:return:
"""
if not is_valid_permutation(permutation):
- raise IOError('Input permutation not valid.')
+ raise OSError("Input permutation not valid.")
return relabeller(in_data, permutation[0], permutation[1])
def erase_labels(in_data, labels_to_erase):
- """
- :param in_data: array corresponding to an image segmentation.
+ """:param in_data: array corresponding to an image segmentation.
:param labels_to_erase: list or tuple of labels
:return: all the labels in the list labels_to_erase will be assigned to zero.
"""
if isinstance(labels_to_erase, int):
- labels_to_erase = [labels_to_erase, ]
- return relabeller(in_data, list_old_labels=labels_to_erase,
- list_new_labels=[0, ] * len(labels_to_erase))
+ labels_to_erase = [labels_to_erase]
+ return relabeller(in_data, list_old_labels=labels_to_erase, list_new_labels=[0] * len(labels_to_erase))
def assign_all_other_labels_the_same_value(in_data, labels_to_keep, same_value_label=255):
- """
- All the labels that are not in the list labels_to_keep will be given the value same_value_label
+ """All the labels that are not in the list labels_to_keep will be given the value same_value_label
:param in_data: array corresponding to an image segmentation.
:param labels_to_keep: list or tuple of value in in_data
:param same_value_label: a single label value.
:return: segmentation of the same size where all the labels not in the list label_to_keep will be assigned to the
value same_value_label.
"""
- list_labels = sorted(list(set(in_data.flat)))
+ list_labels = sorted(set(in_data.flat))
if isinstance(labels_to_keep, int):
- labels_to_keep = [labels_to_keep, ]
+ labels_to_keep = [labels_to_keep]
labels_that_will_have_the_same_value = list(set(list_labels) - set(labels_to_keep) - {0})
- return relabeller(in_data, list_old_labels=labels_that_will_have_the_same_value,
- list_new_labels=[same_value_label, ] * len(labels_that_will_have_the_same_value))
+ return relabeller(
+ in_data,
+ list_old_labels=labels_that_will_have_the_same_value,
+ list_new_labels=[same_value_label] * len(labels_that_will_have_the_same_value),
+ )
def keep_only_one_label(in_data, label_to_keep):
- """
- From a segmentation keeps only the values in the list labels_to_keep.
+ """From a segmentation keeps only the values in the list labels_to_keep.
:param in_data: a segmentation (only positive labels allowed).
:param label_to_keep: the single label that will be kept.
:return:
"""
-
- list_labels = sorted(list(set(in_data.flat)))
+ list_labels = sorted(set(in_data.flat))
if label_to_keep not in list_labels:
- print('the label {} you want to keep is not present in the segmentation'.format(label_to_keep))
+ msg = f"the label {label_to_keep} you want to keep is not present in the segmentation"
+ logging.info(msg)
return in_data
labels_not_to_keep = list(set(list_labels) - {label_to_keep})
- return relabeller(in_data, list_old_labels=labels_not_to_keep, list_new_labels=[0, ]*len(labels_not_to_keep),
- verbose=False)
+ return relabeller(
+ in_data,
+ list_old_labels=labels_not_to_keep,
+ list_new_labels=[0] * len(labels_not_to_keep),
+ verbose=False,
+ )
def relabel_half_side_one_label(in_data, label_old, label_new, side_to_modify, axis, plane_intercept):
- """
- :param in_data: input data array (must be 3d)
+ """:param in_data: input data array (must be 3d)
:param label_old: single label to be replaced
:param label_new: single label to replace with
:param side_to_modify: can be the string 'above' or 'below'
@@ -109,35 +111,34 @@ def relabel_half_side_one_label(in_data, label_old, label_new, side_to_modify, a
:param plane_intercept: voxel along the selected direction plane where to consider the symmetry.
:return:
"""
- if not in_data.ndim == 3:
- msg = 'Input array must be 3-dimensional.'
- raise IOError(msg)
+ if in_data.ndim != 3:
+ msg = "Input array must be 3-dimensional."
+ raise OSError(msg)
- if side_to_modify not in ['below', 'above']:
- msg = 'side_to_copy must be one of the two {}.'.format(['below', 'above'])
- raise IOError(msg)
+ if side_to_modify not in ["below", "above"]:
+ msg = "side_to_copy must be one of the two {}.".format(["below", "above"])
+ raise OSError(msg)
- if axis not in ['x', 'y', 'z']:
- msg = 'axis variable must be one of the following: {}.'.format(['x', 'y', 'z'])
- raise IOError(msg)
+ if axis not in ["x", "y", "z"]:
+ msg = "axis variable must be one of the following: {}.".format(["x", "y", "z"])
+ raise OSError(msg)
positions = in_data == label_old
halfed_positions = np.zeros_like(positions)
- if axis == 'x':
- if side_to_modify == 'above':
+ if axis == "x":
+ if side_to_modify == "above":
halfed_positions[plane_intercept:, :, :] = positions[plane_intercept:, :, :]
- if side_to_modify == 'below':
+ if side_to_modify == "below":
halfed_positions[:plane_intercept, :, :] = positions[:plane_intercept, :, :]
- if axis == 'y':
- if side_to_modify == 'above':
+ if axis == "y":
+ if side_to_modify == "above":
halfed_positions[:, plane_intercept:, :] = positions[:, plane_intercept:, :]
- if side_to_modify == 'below':
+ if side_to_modify == "below":
halfed_positions[:, plane_intercept, :] = positions[:, plane_intercept, :]
- if axis == 'z':
- if side_to_modify == 'above':
- halfed_positions[:, :, plane_intercept:] = positions[ :, :, plane_intercept:]
- if side_to_modify == 'below':
+ if axis == "z":
+ if side_to_modify == "above":
+ halfed_positions[:, :, plane_intercept:] = positions[:, :, plane_intercept:]
+ if side_to_modify == "below":
halfed_positions[:, :, :plane_intercept] = positions[:, :, :plane_intercept]
- new_data = in_data * np.invert(halfed_positions) + label_new * halfed_positions.astype(np.int)
- return new_data
+ return in_data * np.invert(halfed_positions) + label_new * halfed_positions.astype(int)
diff --git a/nilabels/tools/image_colors_manipulations/segmentation_to_rgb.py b/nilabels/tools/image_colors_manipulations/segmentation_to_rgb.py
index a2fb5df..b29c3e4 100644
--- a/nilabels/tools/image_colors_manipulations/segmentation_to_rgb.py
+++ b/nilabels/tools/image_colors_manipulations/segmentation_to_rgb.py
@@ -4,30 +4,24 @@
def get_rgb_image_from_segmentation_and_label_descriptor(im_segm, ldm, invert_black_white=False, dtype_output=np.int32):
- """
- From the labels descriptor and a nibabel segmentation image.
+ """From the labels descriptor and a nibabel segmentation image.
:param im_segm: nibabel segmentation whose labels corresponds to the input labels descriptor.
:param ldm: instance of class label descriptor manager.
:param dtype_output: data type of the output image.
:param invert_black_white: to swap black with white (improving background visualisation).
:return: a 4d image, where at each voxel there is the [r, g, b] vector in the fourth dimension.
"""
- labels_in_image = list(np.sort(list(set(im_segm.get_data().flatten()))))
if not len(im_segm.shape) == 3:
- raise IOError('input segmentation must be 3D.')
+ raise OSError("input segmentation must be 3D.")
rgb_image_arr = np.ones(list(im_segm.shape) + [3])
- for l in ldm.dict_label_descriptor.keys():
- if l not in labels_in_image:
- msg = 'get_corresponding_rgb_image: Label {} present in the label descriptor and not in ' \
- 'selected image'.format(l)
- print(msg)
- pl = im_segm.get_data() == l
+ for l in ldm.dict_label_descriptor:
+ pl = im_segm.get_fdata() == l
rgb_image_arr[pl, :] = ldm.dict_label_descriptor[l][0]
if invert_black_white:
- pl = im_segm.get_data() == 0
+ pl = im_segm.get_fdata() == 0
rgb_image_arr[pl, :] = np.array([255, 255, 255])
return set_new_data(im_segm, rgb_image_arr, new_dtype=dtype_output)
diff --git a/nilabels/tools/image_shape_manipulations/apply_passepartout.py b/nilabels/tools/image_shape_manipulations/apply_passepartout.py
index 78631ac..e1083bd 100644
--- a/nilabels/tools/image_shape_manipulations/apply_passepartout.py
+++ b/nilabels/tools/image_shape_manipulations/apply_passepartout.py
@@ -1,22 +1,20 @@
-from nilabels.tools.detections.contours import get_xyz_borders_of_a_label, set_new_data
from nilabels.tools.aux_methods.utils_nib import images_are_overlapping
+from nilabels.tools.detections.contours import get_xyz_borders_of_a_label, set_new_data
def crop_with_passepartout(im_input, passepartout_values):
- """
- :param im_input:
+ """:param im_input:
:param passepartout_values: in the form [x_min, -x_max, y_min, -y_max, z_min, -z_max]. where -x_max is the
thickness of the border from the border.
:return:
"""
x_min, x_max, y_min, y_max, z_min, z_max = passepartout_values
- cropped_data = im_input.get_data()[x_min:-x_max, y_min:-y_max, z_min:-z_max]
+ cropped_data = im_input.get_fdata()[x_min:-x_max, y_min:-y_max, z_min:-z_max]
return set_new_data(im_input, cropped_data)
def crop_with_passepartout_based_on_label_segmentation(im_input_to_crop, im_segm, margins, label):
- """
- :param im_input_to_crop:
+ """:param im_input_to_crop:
:param im_segm:
:param margins: in the form [x,y,z] space in voxel in each direction left as passepartout
around the selected label of the segmentation.
@@ -27,10 +25,9 @@ def crop_with_passepartout_based_on_label_segmentation(im_input_to_crop, im_segm
"""
assert images_are_overlapping(im_input_to_crop, im_segm)
- v = get_xyz_borders_of_a_label(im_segm.get_data(), label)
-
+ v = get_xyz_borders_of_a_label(im_segm.get_fdata(), label)
x_min, x_max = v[0] - margins[0], v[1] + margins[0] + 1
y_min, y_max = v[2] - margins[1], v[3] + margins[1] + 1
z_min, z_max = v[4] - margins[2], v[5] + margins[2] + 1
- print([x_min, x_max, y_min, y_max, z_min, z_max])
+
return crop_with_passepartout(im_input_to_crop, [x_min, -x_max, y_min, -y_max, z_min, -z_max])
diff --git a/nilabels/tools/image_shape_manipulations/merger.py b/nilabels/tools/image_shape_manipulations/merger.py
index 7be7e66..6d41e4a 100644
--- a/nilabels/tools/image_shape_manipulations/merger.py
+++ b/nilabels/tools/image_shape_manipulations/merger.py
@@ -4,16 +4,16 @@
def merge_labels_from_4d(in_data, keep_original_values=True):
- """
- Can be the inverse function of split label with default parameters.
+ """Can be the inverse function of split label with default parameters.
The labels are assuming to have no overlaps.
:param in_data: 4d volume
- :param keep_original_values: merge the labels with their values, otherwise it uses the values of the slice numbering (including zero label!).
+ :param keep_original_values: merge the labels with their values, otherwise it uses the values of the slice
+ numbering (including zero label!).
:return:
"""
- if not in_data.ndim == 4:
- msg = 'Input array must be 4-dimensional.'
- raise IOError(msg)
+ if in_data.ndim != 4:
+ msg = "Input array must be 4-dimensional."
+ raise OSError(msg)
in_data_shape = in_data.shape
out_data = np.zeros(in_data_shape[:3], dtype=in_data.dtype)
@@ -23,36 +23,33 @@ def merge_labels_from_4d(in_data, keep_original_values=True):
if keep_original_values:
out_data = out_data + slice_t
else:
- out_data = out_data + ((t + 1) * slice_t.astype(np.bool)).astype(in_data.dtype)
+ out_data = out_data + ((t + 1) * slice_t.astype(bool)).astype(in_data.dtype)
return out_data
def stack_images(list_images):
- """
- From a list of images of the same shape, the stack of these images in the new dimension.
+ """From a list of images of the same shape, the stack of these images in the new dimension.
:param list_images:
:return: stack image of the input list
"""
- msg = 'input images shapes are not all of the same dimension'
+ msg = "input images shapes are not all of the same dimension"
assert False not in [list_images[0].shape == im.shape for im in list_images[1:]], msg
- new_data = np.stack([nib_image.get_data() for nib_image in list_images] , axis=len(list_images[0].shape))
- stack_im = set_new_data(list_images[0], new_data)
- return stack_im
+ new_data = np.stack([nib_image.get_fdata() for nib_image in list_images], axis=len(list_images[0].shape))
+ return set_new_data(list_images[0], new_data)
def reproduce_slice_fourth_dimension(nib_image, num_slices=10, repetition_axis=3):
im_sh = nib_image.shape
if not (len(im_sh) == 2 or len(im_sh) == 3):
- raise IOError('Methods can be used only for 2 or 3 dim images. No conflicts with existing multi, slices')
+ raise OSError("Methods can be used only for 2 or 3 dim images. No conflicts with existing multi, slices")
- new_data = np.stack([nib_image.get_data(), ] * num_slices, axis=repetition_axis)
- output_im = set_new_data(nib_image, new_data)
- return output_im
+ new_data = np.stack([nib_image.get_fdata()] * num_slices, axis=repetition_axis)
+ return set_new_data(nib_image, new_data)
def grafting(im_hosting, im_patch, im_patch_mask=None):
- """
- Takes an hosting image, an image patch and a patch mask (optional) of the same dimension and in the same real space.
+ """Takes an hosting image, an image patch and a patch mask (optional) of the same dimension and
+ in the same real space.
It crops the patch (or patch mask if present) on the hosting image, and substitute the value from the patch.
:param im_hosting: Mould or holder of the patch
:param im_patch: patch to add.
@@ -62,36 +59,35 @@ def grafting(im_hosting, im_patch, im_patch_mask=None):
np.testing.assert_array_equal(im_hosting.affine, im_patch.affine)
if im_patch_mask is None:
- patch_region = im_patch.get_data().astype(np.bool)
+ patch_region = im_patch.get_fdata().astype(bool)
else:
np.testing.assert_array_equal(im_hosting.affine, im_patch_mask.affine)
np.testing.assert_array_equal(im_hosting.shape, im_patch_mask.shape)
- patch_region = im_patch_mask.get_data().astype(np.bool)
+ patch_region = im_patch_mask.get_fdata().astype(bool)
patch_inverted = np.invert(patch_region)
- new_data = im_hosting.get_data() * patch_inverted + im_patch.get_data() * patch_region
+ new_data = im_hosting.get_fdata() * patch_inverted + im_patch.get_fdata() * patch_region
return set_new_data(im_hosting, new_data)
def from_segmentations_stack_to_probabilistic_segmentation(arr_labels_stack):
- """
- A probabilistic atlas has at each time-point a different label (a mapping is provided as well in
+ """A probabilistic atlas has at each time-point a different label (a mapping is provided as well in
the conversion with correspondence time-point<->label number). Each time point has the normalised average
of each label.
- :param arr_labels_stack: stack of 1D arrays (segmentations x num_voxels) containing a different discrete segmentation.
+ :param arr_labels_stack: stack of 1D arrays (segmentations x num_voxels) containing a different discrete
+ segmentation.
Values of labels needs to be consecutive, or there will be empty images in the result.
:return:
N number of voxels, J number of segmentations, K number of labels.
"""
J, K = arr_labels_stack.shape[0], np.max(arr_labels_stack) + 1
- return 1/float(J) * np.stack([np.sum(arr_labels_stack == k, axis=0).astype(np.float64) for k in range(K)], axis=0)
+ return 1 / float(J) * np.stack([np.sum(arr_labels_stack == k, axis=0).astype(np.float64) for k in range(K)], axis=0)
def substitute_volume_at_timepoint(im_input_4d, im_input_3d, timepoint):
- """
- Substitute the im_input_3d image at the time point timepoint of the im_input_4d.
+ """Substitute the im_input_3d image at the time point timepoint of the im_input_4d.
:param im_input_4d: 4d image
:param im_input_3d: 3d image whose shape is compatible with the fist 3 dimensions of im_input_4d
:param timepoint: a timepoint in the 4th dimension of the im_input_4d
@@ -100,9 +96,8 @@ def substitute_volume_at_timepoint(im_input_4d, im_input_3d, timepoint):
"""
if len(im_input_4d.shape) == 3 and timepoint == 0:
return im_input_3d
- elif len(im_input_4d.shape) == 4 and timepoint < im_input_4d.shape[-1]:
- new_data = im_input_4d.get_data()[:]
- new_data[..., timepoint] = im_input_3d.get_data()[:]
+ if len(im_input_4d.shape) == 4 and timepoint < im_input_4d.shape[-1]:
+ new_data = im_input_4d.get_fdata()[:]
+ new_data[..., timepoint] = im_input_3d.get_fdata()[:]
return set_new_data(im_input_4d, new_data)
- else:
- raise IOError('Incompatible shape input volume.')
+ raise OSError("Incompatible shape input volume.")
diff --git a/nilabels/tools/image_shape_manipulations/splitter.py b/nilabels/tools/image_shape_manipulations/splitter.py
index cb00a3b..0a7661f 100644
--- a/nilabels/tools/image_shape_manipulations/splitter.py
+++ b/nilabels/tools/image_shape_manipulations/splitter.py
@@ -2,8 +2,7 @@
def split_labels_to_4d(in_data, list_labels=(), keep_original_values=True):
- """
- Split labels of a 3d segmentation in a 4d segmentation,
+ """Split labels of a 3d segmentation in a 4d segmentation,
one label for each slice in ascending order.
Labels can be relabelled in consecutive order or can keep the
original labels value.
@@ -13,7 +12,7 @@ def split_labels_to_4d(in_data, list_labels=(), keep_original_values=True):
value for all the
:return:
"""
- msg = 'Input array must be 3-dimensional.'
+ msg = "Input array must be 3-dimensional."
assert in_data.ndim == 3, msg
out_data = np.zeros(list(in_data.shape) + [len(list_labels)], dtype=in_data.dtype)
diff --git a/nilabels/tools/visualiser/graphs_and_stats.py b/nilabels/tools/visualiser/graphs_and_stats.py
index 7a1fc6e..4bad86f 100644
--- a/nilabels/tools/visualiser/graphs_and_stats.py
+++ b/nilabels/tools/visualiser/graphs_and_stats.py
@@ -1,20 +1,24 @@
-import numpy as np
-import matplotlib
-from matplotlib.patches import Circle, Wedge, Polygon
-from matplotlib.collections import PatchCollection
import matplotlib as mpl
import matplotlib.pyplot as plt
+import numpy as np
import pandas as pd
-import matplotlib.ticker as ticker
-
-
-def bulls_eye(ax, data, cmap=None, norm=None, raidal_subdivisions=(2, 8, 8, 11),
- centered=(True, False, False, True), add_nomenclatures=True,
- nomenclature_white=False,
- cell_resolution=128,
- pfi_where_to_save=None, colors_bound='-k'):
- """
- Clockwise, from smaller radius to bigger radius.
+from matplotlib import ticker
+
+
+def bulls_eye(
+ ax,
+ data,
+ cmap=None,
+ norm=None,
+ raidal_subdivisions=(2, 8, 8, 11),
+ centered=(True, False, False, True),
+ add_nomenclatures=True,
+ nomenclature_white=False,
+ cell_resolution=128,
+ pfi_where_to_save=None,
+ colors_bound="-k",
+):
+ """Clockwise, from smaller radius to bigger radius.
:param ax:
:param data:
:param cmap:
@@ -37,14 +41,14 @@ def bulls_eye(ax, data, cmap=None, norm=None, raidal_subdivisions=(2, 8, 8, 11),
if norm is None:
norm = mpl.colors.Normalize(vmin=data.min(), vmax=data.max())
- theta = np.linspace(0, 2*np.pi, 768)
- r = np.linspace(0, 1, len(raidal_subdivisions)+1)
+ theta = np.linspace(0, 2 * np.pi, 768)
+ r = np.linspace(0, 1, len(raidal_subdivisions) + 1)
nomenclatures = []
if isinstance(add_nomenclatures, bool):
if add_nomenclatures:
- nomenclatures = range(1, sum(raidal_subdivisions)+1)
- elif isinstance(add_nomenclatures, list) or isinstance(add_nomenclatures, tuple):
+ nomenclatures = range(1, sum(raidal_subdivisions) + 1)
+ elif isinstance(add_nomenclatures, (list, tuple)):
assert len(add_nomenclatures) == sum(raidal_subdivisions)
nomenclatures = add_nomenclatures[:]
add_nomenclatures = True
@@ -53,56 +57,65 @@ def bulls_eye(ax, data, cmap=None, norm=None, raidal_subdivisions=(2, 8, 8, 11),
line_width_circular = line_width
for i in range(r.shape[0]):
if i == range(r.shape[0])[-1]:
- line_width_circular = int(line_width / 2.)
+ line_width_circular = int(line_width / 2.0)
ax.plot(theta, np.repeat(r[i], theta.shape), colors_bound, lw=line_width_circular)
# iterate over cells divided by radial subdivision
for rs_id, rs in enumerate(raidal_subdivisions):
for i in range(rs):
cell_id = sum(raidal_subdivisions[:rs_id]) + i
- theta_i = - i * 2 * np.pi / rs + np.pi / 2
+ theta_i = -i * 2 * np.pi / rs + np.pi / 2
if not centered[rs_id]:
theta_i += (2 * np.pi / rs) / 2
theta_i_plus_one = theta_i - 2 * np.pi / rs # clockwise
# Create colour fillings for each cell:
theta_interval = np.linspace(theta_i, theta_i_plus_one, cell_resolution)
- r_interval = np.array([r[rs_id], r[rs_id+1]])
- angle = np.repeat(theta_interval[:, np.newaxis], 2, axis=1)
+ r_interval = np.array([r[rs_id], r[rs_id + 1]])
+ angle = np.repeat(theta_interval[:, np.newaxis], 2, axis=1)
radius = np.repeat(r_interval[:, np.newaxis], cell_resolution, axis=1).T
z = np.ones((cell_resolution, 2)) * data[cell_id]
ax.pcolormesh(angle, radius, z, cmap=cmap, norm=norm)
# Create radial bounds
- if rs > 1:
- ax.plot([theta_i, theta_i], [r[rs_id], r[rs_id+1]], colors_bound, lw=line_width)
+ if rs > 1:
+ ax.plot([theta_i, theta_i], [r[rs_id], r[rs_id + 1]], colors_bound, lw=line_width)
# Add centered nomenclatures if needed, with selected colour
if add_nomenclatures:
if rs == 1 and rs_id == 0:
cell_center = (0, 0)
else:
- cell_center = ((theta_i + theta_i_plus_one) / 2., r[rs_id] + .5 * r[1] )
+ cell_center = ((theta_i + theta_i_plus_one) / 2.0, r[rs_id] + 0.5 * r[1])
- color_nomenclature = 'black'
- if nomenclature_white == True:
- color_nomenclature = 'w'
+ color_nomenclature = "black"
+ if nomenclature_white:
+ color_nomenclature = "w"
elif isinstance(nomenclature_white, str):
- sign, perc = nomenclature_white[0], int(nomenclature_white[1:]) / 100.
+ sign, perc = nomenclature_white[0], int(nomenclature_white[1:]) / 100.0
data_interval = np.max(data) - np.min(data)
- if sign == '>':
- if data[cell_id] > perc * data_interval + np.min(data):
- color_nomenclature = 'w'
- if sign == '<':
- if data[cell_id] < perc * data_interval + np.min(data):
- color_nomenclature = 'w'
-
- if isinstance(nomenclatures[0], (int, long, float, complex)):
- ax.annotate(r"${:.3g}$".format(nomenclatures[cell_id]), xy=cell_center,
- xytext=(cell_center[0], cell_center[1]), color=color_nomenclature,
- horizontalalignment='center', verticalalignment='center', size=8)
+ if sign == ">" and data[cell_id] > perc * data_interval + np.min(data):
+ color_nomenclature = "w"
+ if sign == "<" and data[cell_id] < perc * data_interval + np.min(data):
+ color_nomenclature = "w"
+
+ if isinstance(nomenclatures[0], (int, int, float, complex)):
+ ax.annotate(
+ rf"${nomenclatures[cell_id]:.3g}$",
+ xy=cell_center,
+ xytext=(cell_center[0], cell_center[1]),
+ color=color_nomenclature,
+ horizontalalignment="center",
+ verticalalignment="center",
+ size=8,
+ )
else:
- ax.annotate(nomenclatures[cell_id], xy=cell_center,
- xytext=(cell_center[0], cell_center[1]),
- horizontalalignment='center', verticalalignment='center', size=12)
+ ax.annotate(
+ nomenclatures[cell_id],
+ xy=cell_center,
+ xytext=(cell_center[0], cell_center[1]),
+ horizontalalignment="center",
+ verticalalignment="center",
+ size=12,
+ )
ax.grid(False)
ax.set_ylim([0, 1])
@@ -110,14 +123,25 @@ def bulls_eye(ax, data, cmap=None, norm=None, raidal_subdivisions=(2, 8, 8, 11),
ax.set_xticklabels([])
if pfi_where_to_save is not None:
- plt.savefig(pfi_where_to_save, format='pdf', dpi=200)
-
-
-def multi_bull_eyes(multi_data, cbar=None, cmaps=None, normalisations=None,
- global_title=None, canvas_title='title', titles=None, units=None, raidal_subdivisions=(2, 8, 8, 11),
- centered=(True, False, False, True), add_nomenclatures=(True, True, True, True),
- nomenclature_white=(False, False, False, False),
- pfi_where_to_save=None, show=True):
+ plt.savefig(pfi_where_to_save, format="pdf", dpi=200)
+
+
+def multi_bull_eyes(
+ multi_data,
+ cbar=None,
+ cmaps=None,
+ normalisations=None,
+ global_title=None,
+ canvas_title="title",
+ titles=None,
+ units=None,
+ raidal_subdivisions=(2, 8, 8, 11),
+ centered=(True, False, False, True),
+ add_nomenclatures=(True, True, True, True),
+ nomenclature_white=(False, False, False, False),
+ pfi_where_to_save=None,
+ show=True,
+):
plt.clf()
n_fig = len(multi_data)
if cbar is None:
@@ -125,20 +149,21 @@ def multi_bull_eyes(multi_data, cbar=None, cmaps=None, normalisations=None,
if cmaps is None:
cmaps = [mpl.cm.viridis] * n_fig
if normalisations is None:
- normalisations = [mpl.colors.Normalize(vmin=np.min(multi_data[i]), vmax=np.max(multi_data[i]))
- for i in range(n_fig)]
+ normalisations = [
+ mpl.colors.Normalize(vmin=np.min(multi_data[i]), vmax=np.max(multi_data[i])) for i in range(n_fig)
+ ]
if titles is None:
- titles = ['Title {}'.format(i) for i in range(n_fig)]
+ titles = [f"Title {i}" for i in range(n_fig)]
h_space = 0.1 / n_fig
- h_dim_fig = .85
- w_dim_fig = .85 / n_fig
+ h_dim_fig = 0.85
+ w_dim_fig = 0.85 / n_fig
- def fmt(x, pos):
+ def fmt(x, pos): # noqa: ARG001
# a, b = '{:.2e}'.format(x).split('e')
# b = int(b)
# return r'${} \times 10^{{{}}}$'.format(a, b)
- return r"${:.4g}$".format(x)
+ return rf"${x:.4g}$"
# Make a figure and axes with dimensions as desired.
fig = plt.figure(figsize=(3 * n_fig, 4))
@@ -149,43 +174,67 @@ def fmt(x, pos):
for n in range(n_fig):
origin_fig = (h_space * (n + 1) + w_dim_fig * n, 0.14)
ax = fig.add_axes([origin_fig[0], origin_fig[1], w_dim_fig, h_dim_fig], polar=True)
- bulls_eye(ax, multi_data[n], cmap=cmaps[n], norm=normalisations[n], raidal_subdivisions=raidal_subdivisions,
- centered=centered, add_nomenclatures=add_nomenclatures[n], nomenclature_white=nomenclature_white[n])
+ bulls_eye(
+ ax,
+ multi_data[n],
+ cmap=cmaps[n],
+ norm=normalisations[n],
+ raidal_subdivisions=raidal_subdivisions,
+ centered=centered,
+ add_nomenclatures=add_nomenclatures[n],
+ nomenclature_white=nomenclature_white[n],
+ )
ax.set_title(titles[n], size=10)
if cbar[n]:
- origin_cbar = (h_space * (n + 1) + w_dim_fig * n, .15)
- axl = fig.add_axes([origin_cbar[0], origin_cbar[1], w_dim_fig, .05])
- cb1 = mpl.colorbar.ColorbarBase(axl, cmap=cmaps[n], norm=normalisations[n], orientation='horizontal',
- format=ticker.FuncFormatter(fmt))
+ origin_cbar = (h_space * (n + 1) + w_dim_fig * n, 0.15)
+ axl = fig.add_axes([origin_cbar[0], origin_cbar[1], w_dim_fig, 0.05])
+ cb1 = mpl.colorbar.ColorbarBase(
+ axl,
+ cmap=cmaps[n],
+ norm=normalisations[n],
+ orientation="horizontal",
+ format=ticker.FuncFormatter(fmt),
+ )
cb1.ax.tick_params(labelsize=8)
if units is not None:
cb1.set_label(units[n])
if pfi_where_to_save is not None:
- plt.savefig(pfi_where_to_save, format='pdf', dpi=330)
+ plt.savefig(pfi_where_to_save, format="pdf", dpi=330)
if show:
plt.show()
-def confusion_matrix(confusion_data_frame, annotation_data_frame=None, fig_size=(4,4), title='Title', cmap=plt.cm.jet,
- pfi_where_to_save=None, show_fig=True, axis_position=None, margin=None, top_adjust=None):
-
+def confusion_matrix(
+ confusion_data_frame,
+ annotation_data_frame=None,
+ fig_size=(4, 4),
+ title="Title",
+ cmap=plt.cm.jet,
+ pfi_where_to_save=None,
+ show_fig=True,
+ axis_position=None,
+ margin=None,
+ top_adjust=None,
+):
fig = plt.figure(figsize=fig_size)
plt.clf()
ax = fig.add_subplot(111)
ax.set_aspect(1)
- res = ax.imshow(confusion_data_frame.as_matrix(), cmap=cmap,
- interpolation='nearest', origin='lower')
+ res = ax.imshow(confusion_data_frame.as_matrix(), cmap=cmap, interpolation="nearest", origin="lower")
rows, cols = confusion_data_frame.shape
if annotation_data_frame is not None:
for x in range(rows):
for y in range(cols):
- ax.annotate(str(annotation_data_frame.as_matrix()[x, y]), xy=(y, x),
- horizontalalignment='center',
- verticalalignment='center')
+ ax.annotate(
+ str(annotation_data_frame.as_matrix()[x, y]),
+ xy=(y, x),
+ horizontalalignment="center",
+ verticalalignment="center",
+ )
fig.colorbar(res)
@@ -193,10 +242,10 @@ def confusion_matrix(confusion_data_frame, annotation_data_frame=None, fig_size=
cols_index_list = list(confusion_data_frame.columns)
ax.set_xticks(range(cols))
- ax.set_xticklabels(cols_index_list, rotation=45, ha='center')
+ ax.set_xticklabels(cols_index_list, rotation=45, ha="center")
ax.set_yticks(range(rows))
ax.set_yticklabels(rows_index_list)
- fig.text(.5, .05, title, ha='center')
+ fig.text(0.5, 0.05, title, ha="center")
ax.invert_yaxis()
ax.xaxis.tick_top()
@@ -211,12 +260,12 @@ def confusion_matrix(confusion_data_frame, annotation_data_frame=None, fig_size=
fig.tight_layout()
if pfi_where_to_save is not None:
- plt.savefig(pfi_where_to_save, format='pdf', dpi=200)
+ plt.savefig(pfi_where_to_save, format="pdf", dpi=200)
if show_fig:
plt.show()
-if __name__ == '__main__':
+if __name__ == "__main__":
# TODO move this part in examples.
# Very dummy data:
@@ -224,49 +273,47 @@ def confusion_matrix(confusion_data_frame, annotation_data_frame=None, fig_size=
# TEST bull-eye three-fold
if True:
-
- fig, ax = plt.subplots(figsize=(12, 8), nrows=1, ncols=3,
- subplot_kw=dict(projection='polar'))
- fig.canvas.set_window_title('Left Ventricle Bulls Eyes (AHA)')
+ fig, ax = plt.subplots(figsize=(12, 8), nrows=1, ncols=3, subplot_kw={"projection": "polar"})
+ fig.canvas.set_window_title("Left Ventricle Bulls Eyes (AHA)")
# First one:
cmap = mpl.cm.viridis
norm = mpl.colors.Normalize(vmin=1, vmax=29)
bulls_eye(ax[0], data, cmap=cmap, norm=norm)
- ax[0].set_title('Bulls Eye (AHA)')
+ ax[0].set_title("Bulls Eye (AHA)")
axl = fig.add_axes([0.14, 0.15, 0.2, 0.05])
- cb1 = mpl.colorbar.ColorbarBase(axl, cmap=cmap, norm=norm, orientation='horizontal')
- cb1.set_label('Some Units')
+ cb1 = mpl.colorbar.ColorbarBase(axl, cmap=cmap, norm=norm, orientation="horizontal")
+ cb1.set_label("Some Units")
# Second one
cmap2 = mpl.cm.cool
norm2 = mpl.colors.Normalize(vmin=1, vmax=29)
bulls_eye(ax[1], data, cmap=cmap2, norm=norm2)
- ax[1].set_title('Bulls Eye (AHA)')
+ ax[1].set_title("Bulls Eye (AHA)")
axl2 = fig.add_axes([0.41, 0.15, 0.2, 0.05])
- cb2 = mpl.colorbar.ColorbarBase(axl2, cmap=cmap2, norm=norm2, orientation='horizontal')
- cb2.set_label('Some other units')
+ cb2 = mpl.colorbar.ColorbarBase(axl2, cmap=cmap2, norm=norm2, orientation="horizontal")
+ cb2.set_label("Some other units")
# Third one
cmap3 = mpl.cm.winter
norm3 = mpl.colors.Normalize(vmin=1, vmax=29)
bulls_eye(ax[2], data, cmap=cmap3, norm=norm3)
- ax[2].set_title('Bulls Eye third')
+ ax[2].set_title("Bulls Eye third")
axl3 = fig.add_axes([0.69, 0.15, 0.2, 0.05])
- cb3 = mpl.colorbar.ColorbarBase(axl3, cmap=cmap3, norm=norm3, orientation='horizontal')
- cb3.set_label('Some more units')
+ cb3 = mpl.colorbar.ColorbarBase(axl3, cmap=cmap3, norm=norm3, orientation="horizontal")
+ cb3.set_label("Some more units")
plt.show()
if False:
fig = plt.figure(figsize=(5, 7))
- fig.canvas.set_window_title('Bulls Eyes - segmentation assessment')
+ fig.canvas.set_window_title("Bulls Eyes - segmentation assessment")
# First and only:
cmap = mpl.cm.viridis
@@ -274,32 +321,31 @@ def confusion_matrix(confusion_data_frame, annotation_data_frame=None, fig_size=
ax = fig.add_axes([0.1, 0.2, 0.8, 0.7], polar=True)
bulls_eye(ax, data, cmap=cmap, norm=norm)
- ax.set_title('Bulls Eye (AHA)')
+ ax.set_title("Bulls Eye (AHA)")
axl = fig.add_axes([0.1, 0.15, 0.8, 0.05])
- cb1 = mpl.colorbar.ColorbarBase(axl, cmap=cmap, norm=norm, orientation='horizontal')
- cb1.set_label('Some Units')
+ cb1 = mpl.colorbar.ColorbarBase(axl, cmap=cmap, norm=norm, orientation="horizontal")
+ cb1.set_label("Some Units")
plt.show()
if False:
+ multi_data = [range(1, 17), list(0.000000001 * np.array(range(1, 17))), list(0.001 * np.array(range(1, 17)))]
- # multi_data = [data for _ in range(3)]
- # print(multi_data)
- # multi_bull_eyes(multi_data)
- #
- # plt.show(block=False)
-
- multi_data = [range(1, 17), list(0.000000001 * np.array(range(1,17))), list( 0.001 * np.array(range(1,17)))]
- print(multi_data)
- multi_bull_eyes(multi_data, raidal_subdivisions=(3,3,4,6),
- centered=(True, True, True, True), add_nomenclatures=[True]*3)
+ multi_bull_eyes(
+ multi_data,
+ raidal_subdivisions=(3, 3, 4, 6),
+ centered=(True, True, True, True),
+ add_nomenclatures=[True] * 3,
+ )
plt.show(block=True)
- if False :
- d = {'one': pd.Series([1., 2., 3.], index=['a', 'b', 'c']),
- 'two': pd.Series([1.5, 2.5, 3.5, 4.5], index=['a', 'b', 'c', 'd'])}
- df = pd.DataFrame(d)
+ if False:
+ d = {
+ "one": pd.Series([1.0, 2.0, 3.0], index=["a", "b", "c"]),
+ "two": pd.Series([1.5, 2.5, 3.5, 4.5], index=["a", "b", "c", "d"]),
+ }
+ df_for_matrix = pd.DataFrame(d)
- confusion_matrix(df, annotation_data_frame=df, title='a title', fig_size=(5.5, 8))
+ confusion_matrix(df_for_matrix, annotation_data_frame=df_for_matrix, title="a title", fig_size=(5.5, 8))
diff --git a/nilabels/tools/visualiser/see_volume.py b/nilabels/tools/visualiser/see_volume.py
index 3c63c85..872554c 100644
--- a/nilabels/tools/visualiser/see_volume.py
+++ b/nilabels/tools/visualiser/see_volume.py
@@ -1,17 +1,17 @@
import os
from os.path import join as jph
-import numpy as np
-import nibabel as nib
-from matplotlib import rc
import matplotlib.pyplot as plt
+import nibabel as nib
+import numpy as np
+from matplotlib import rc
from nilabels.tools.aux_methods.utils import print_and_run
-def see_array(in_array, pfo_tmp='./z_tmp', in_array_segm=None, pfi_label_descriptor=None, block=False):
- """
- Itk-snap based quick array visualiser.
+def see_array(in_array, pfo_tmp="./z_tmp", in_array_segm=None, pfi_label_descriptor=None, block=False):
+ """Itk-snap based quick array visualiser.
+
:param in_array: numpy array or list of numpy array same dimension (GIGO).
:param pfo_tmp: path to file temporary folder.
:param in_array_segm: if there is a single array representing a segmentation (in this case all images must
@@ -25,48 +25,53 @@ def see_array(in_array, pfo_tmp='./z_tmp', in_array_segm=None, pfi_label_descrip
sh = in_array[0].shape
for arr in in_array[1:]:
assert sh == arr.shape
- print_and_run('mkdir {}'.format(pfo_tmp))
- cmd = 'itksnap -g '
+ print_and_run(f"mkdir {pfo_tmp}")
+ cmd = "itksnap -g "
for arr_id, arr in enumerate(in_array):
im = nib.Nifti1Image(arr, affine=np.eye(4))
- pfi_im = jph(pfo_tmp, 'im_{}.nii.gz'.format(arr_id))
+ pfi_im = jph(pfo_tmp, f"im_{arr_id}.nii.gz")
nib.save(im, pfi_im)
if arr_id == 1:
- cmd += ' -o {} '.format(pfi_im)
+ cmd += f" -o {pfi_im} "
else:
- cmd += ' {} '.format(pfi_im)
+ cmd += f" {pfi_im} "
elif isinstance(in_array, np.ndarray):
- print_and_run('mkdir {}'.format(pfo_tmp))
+ print_and_run(f"mkdir {pfo_tmp}")
im = nib.Nifti1Image(in_array, affine=np.eye(4))
- pfi_im = jph(pfo_tmp, 'im_0.nii.gz')
+ pfi_im = jph(pfo_tmp, "im_0.nii.gz")
nib.save(im, pfi_im)
- cmd = 'itksnap -g {}'.format(pfi_im)
+ cmd = f"itksnap -g {pfi_im}"
else:
- raise IOError
+ raise OSError
if in_array_segm is not None:
im_segm = nib.Nifti1Image(in_array_segm, affine=np.eye(4))
- pfi_im_segm = jph(pfo_tmp, 'im_segm_0.nii.gz')
+ pfi_im_segm = jph(pfo_tmp, "im_segm_0.nii.gz")
nib.save(im_segm, pfi_im_segm)
- cmd += ' -s {} '.format(pfi_im_segm)
- if pfi_label_descriptor:
- if os.path.exists(pfi_label_descriptor):
- cmd += ' -l {} '.format(pfi_im_segm)
+ cmd += f" -s {pfi_im_segm} "
+ if pfi_label_descriptor and os.path.exists(pfi_label_descriptor):
+ cmd += f" -l {pfi_im_segm} "
print_and_run(cmd)
if block:
- _ = raw_input("Press any key to continue.")
-
-
-def see_image_slice_with_a_grid(pfi_image, fig_num=1, axis_quote=('y', 230), vmin=None, vmax=None, cmap='gray',
- pfi_where_to_save=None):
- rc('text', usetex=True)
+ _ = input("Press any key to continue.")
+
+
+def see_image_slice_with_a_grid(
+ pfi_image,
+ fig_num=1,
+ axis_quote=("y", 230),
+ vmin=None,
+ vmax=None,
+ cmap="gray",
+ pfi_where_to_save=None,
+):
+ rc("text", usetex=True)
fig = plt.figure(fig_num, figsize=(6, 6))
- fig.canvas.set_window_title('canvas {}'.format(fig_num))
+ fig.canvas.set_window_title(f"canvas {fig_num}")
ax = fig.add_subplot(111)
im = nib.load(pfi_image)
- if axis_quote[0] == 'x':
-
- data = im.get_data()[axis_quote[1], :, :].T
+ if axis_quote[0] == "x":
+ data = im.get_fdata()[axis_quote[1], :, :].T
shape = data.shape
voxel_origin = np.array([axis_quote[1], 0, 0, 1])
@@ -85,10 +90,9 @@ def see_image_slice_with_a_grid(pfi_image, fig_num=1, axis_quote=('y', 230), vmi
vertical_max = pt_y[2]
extent = [horizontal_min, horizontal_max, vertical_min, vertical_max]
- print(extent)
- elif axis_quote[0] == 'y':
- data = im.get_data()[:, axis_quote[1], :].T
+ elif axis_quote[0] == "y":
+ data = im.get_fdata()[:, axis_quote[1], :].T
shape = data.shape
voxel_origin = np.array([0, axis_quote[1], 0, 1])
@@ -108,9 +112,8 @@ def see_image_slice_with_a_grid(pfi_image, fig_num=1, axis_quote=('y', 230), vmi
extent = [horizontal_min, horizontal_max, vertical_min, vertical_max]
- elif axis_quote[0] == 'z':
-
- data = im.get_data()[:, :, axis_quote[1]].T
+ elif axis_quote[0] == "z":
+ data = im.get_fdata()[:, :, axis_quote[1]].T
shape = data.shape
voxel_origin = np.array([0, 0, axis_quote[1], 1])
@@ -131,23 +134,12 @@ def see_image_slice_with_a_grid(pfi_image, fig_num=1, axis_quote=('y', 230), vmi
extent = [horizontal_min, horizontal_max, vertical_min, vertical_max]
else:
- raise IOError
-
- print(voxel_origin)
- print(voxel_x)
- print(voxel_y)
- print(pt_origin)
- print(pt_x)
- print(pt_y)
- print(extent)
- res = ax.imshow(data,
- extent=extent,
- origin='lower',
- interpolation='nearest',
- cmap=cmap, vmin=vmin, vmax=vmax)
-
- ax.grid(color='grey', linestyle='-', linewidth=0.5)
- ax.set_aspect('equal')
+ raise OSError
+
+ ax.imshow(data, extent=extent, origin="lower", interpolation="nearest", cmap=cmap, vmin=vmin, vmax=vmax)
+
+ ax.grid(color="grey", linestyle="-", linewidth=0.5)
+ ax.set_aspect("equal")
for tick in ax.xaxis.get_major_ticks():
tick.label.set_fontsize(8)
@@ -155,10 +147,4 @@ def see_image_slice_with_a_grid(pfi_image, fig_num=1, axis_quote=('y', 230), vmi
tick.label.set_fontsize(8)
if pfi_where_to_save is not None:
- plt.savefig(pfi_where_to_save, format='pdf', dpi=200)
-
-
-
-
-
-
+ plt.savefig(pfi_where_to_save, format="pdf", dpi=200)
diff --git a/nilabels/tools/visualiser/volume_manipulations_for_visualisation.py b/nilabels/tools/visualiser/volume_manipulations_for_visualisation.py
index cbe5198..77a660c 100644
--- a/nilabels/tools/visualiser/volume_manipulations_for_visualisation.py
+++ b/nilabels/tools/visualiser/volume_manipulations_for_visualisation.py
@@ -3,9 +3,8 @@
from nilabels.tools.aux_methods.utils_nib import set_new_data
-def exploded_segmentation(im_segm, direction, intercepts, offset, dtype=np.int):
- """
- Damien Hirst-like sectioning of an anatomical segmentation.
+def exploded_segmentation(im_segm, direction, intercepts, offset, dtype=int):
+ """Damien Hirst-like sectioning of an anatomical segmentation.
:param im_segm: nibabel image segmentation
:param direction: sectioning direction, can be sagittal, axial or coronal
(conventional names for images oriented to standard (diagonal affine transformation))
@@ -14,28 +13,25 @@ def exploded_segmentation(im_segm, direction, intercepts, offset, dtype=np.int):
:param offset: voxel to leave empty between one slice and the other
:return: nibabel image output as sectioning of the input one.
"""
- if direction.lower() == 'axial':
+ if direction.lower() == "axial":
block = np.zeros([im_segm.shape[0], im_segm.shape[1], offset]).astype(dtype)
stack = []
for j in range(1, len(intercepts)):
- stack += [im_segm.get_data()[:, :, intercepts[j-1]:intercepts[j]].astype(dtype)] + [block]
+ stack += [im_segm.get_fdata()[:, :, intercepts[j - 1] : intercepts[j]].astype(dtype)] + [block]
return set_new_data(im_segm, np.concatenate(stack, axis=2))
- elif direction.lower() == 'sagittal':
+ if direction.lower() == "sagittal":
block = np.zeros([offset, im_segm.shape[1], im_segm.shape[2]]).astype(dtype)
stack = []
for j in range(1, len(intercepts)):
- stack += [im_segm.get_data()[intercepts[j - 1]:intercepts[j], :, :].astype(dtype)] + [block]
+ stack += [im_segm.get_fdata()[intercepts[j - 1] : intercepts[j], :, :].astype(dtype)] + [block]
return set_new_data(im_segm, np.concatenate(stack, axis=0))
- elif direction.lower() == 'coronal':
+ if direction.lower() == "coronal":
block = np.zeros([im_segm.shape[0], offset, im_segm.shape[2]]).astype(dtype)
stack = []
for j in range(1, len(intercepts)):
- stack += [im_segm.get_data()[:, intercepts[j - 1]:intercepts[j], :].astype(dtype)] + [block]
- for st in stack:
- print(st.shape)
+ stack += [im_segm.get_fdata()[:, intercepts[j - 1] : intercepts[j], :].astype(dtype)] + [block]
return set_new_data(im_segm, np.concatenate(stack, axis=1))
- else:
- raise IOError
+ raise OSError
diff --git a/poetry.lock b/poetry.lock
new file mode 100644
index 0000000..9f52a7e
--- /dev/null
+++ b/poetry.lock
@@ -0,0 +1,1097 @@
+# This file is automatically @generated by Poetry 1.4.2 and should not be changed by hand.
+
+[[package]]
+name = "colorama"
+version = "0.4.6"
+description = "Cross-platform colored terminal text."
+category = "dev"
+optional = false
+python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,!=3.6.*,>=2.7"
+files = [
+ {file = "colorama-0.4.6-py2.py3-none-any.whl", hash = "sha256:4f1d9991f5acc0ca119f9d443620b77f9d6b33703e51011c16baf57afb285fc6"},
+ {file = "colorama-0.4.6.tar.gz", hash = "sha256:08695f5cb7ed6e0531a20572697297273c47b8cae5a63ffc6d6ed5c201be6e44"},
+]
+
+[[package]]
+name = "contourpy"
+version = "1.2.1"
+description = "Python library for calculating contours of 2D quadrilateral grids"
+category = "main"
+optional = false
+python-versions = ">=3.9"
+files = [
+ {file = "contourpy-1.2.1-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:bd7c23df857d488f418439686d3b10ae2fbf9bc256cd045b37a8c16575ea1040"},
+ {file = "contourpy-1.2.1-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:5b9eb0ca724a241683c9685a484da9d35c872fd42756574a7cfbf58af26677fd"},
+ {file = "contourpy-1.2.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4c75507d0a55378240f781599c30e7776674dbaf883a46d1c90f37e563453480"},
+ {file = "contourpy-1.2.1-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:11959f0ce4a6f7b76ec578576a0b61a28bdc0696194b6347ba3f1c53827178b9"},
+ {file = "contourpy-1.2.1-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:eb3315a8a236ee19b6df481fc5f997436e8ade24a9f03dfdc6bd490fea20c6da"},
+ {file = "contourpy-1.2.1-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:39f3ecaf76cd98e802f094e0d4fbc6dc9c45a8d0c4d185f0f6c2234e14e5f75b"},
+ {file = "contourpy-1.2.1-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:94b34f32646ca0414237168d68a9157cb3889f06b096612afdd296003fdd32fd"},
+ {file = "contourpy-1.2.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:457499c79fa84593f22454bbd27670227874cd2ff5d6c84e60575c8b50a69619"},
+ {file = "contourpy-1.2.1-cp310-cp310-win32.whl", hash = "sha256:ac58bdee53cbeba2ecad824fa8159493f0bf3b8ea4e93feb06c9a465d6c87da8"},
+ {file = "contourpy-1.2.1-cp310-cp310-win_amd64.whl", hash = "sha256:9cffe0f850e89d7c0012a1fb8730f75edd4320a0a731ed0c183904fe6ecfc3a9"},
+ {file = "contourpy-1.2.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:6022cecf8f44e36af10bd9118ca71f371078b4c168b6e0fab43d4a889985dbb5"},
+ {file = "contourpy-1.2.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:ef5adb9a3b1d0c645ff694f9bca7702ec2c70f4d734f9922ea34de02294fdf72"},
+ {file = "contourpy-1.2.1-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6150ffa5c767bc6332df27157d95442c379b7dce3a38dff89c0f39b63275696f"},
+ {file = "contourpy-1.2.1-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:4c863140fafc615c14a4bf4efd0f4425c02230eb8ef02784c9a156461e62c965"},
+ {file = "contourpy-1.2.1-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:00e5388f71c1a0610e6fe56b5c44ab7ba14165cdd6d695429c5cd94021e390b2"},
+ {file = "contourpy-1.2.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d4492d82b3bc7fbb7e3610747b159869468079fe149ec5c4d771fa1f614a14df"},
+ {file = "contourpy-1.2.1-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:49e70d111fee47284d9dd867c9bb9a7058a3c617274900780c43e38d90fe1205"},
+ {file = "contourpy-1.2.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:b59c0ffceff8d4d3996a45f2bb6f4c207f94684a96bf3d9728dbb77428dd8cb8"},
+ {file = "contourpy-1.2.1-cp311-cp311-win32.whl", hash = "sha256:7b4182299f251060996af5249c286bae9361fa8c6a9cda5efc29fe8bfd6062ec"},
+ {file = "contourpy-1.2.1-cp311-cp311-win_amd64.whl", hash = "sha256:2855c8b0b55958265e8b5888d6a615ba02883b225f2227461aa9127c578a4922"},
+ {file = "contourpy-1.2.1-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:62828cada4a2b850dbef89c81f5a33741898b305db244904de418cc957ff05dc"},
+ {file = "contourpy-1.2.1-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:309be79c0a354afff9ff7da4aaed7c3257e77edf6c1b448a779329431ee79d7e"},
+ {file = "contourpy-1.2.1-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2e785e0f2ef0d567099b9ff92cbfb958d71c2d5b9259981cd9bee81bd194c9a4"},
+ {file = "contourpy-1.2.1-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:1cac0a8f71a041aa587410424ad46dfa6a11f6149ceb219ce7dd48f6b02b87a7"},
+ {file = "contourpy-1.2.1-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:af3f4485884750dddd9c25cb7e3915d83c2db92488b38ccb77dd594eac84c4a0"},
+ {file = "contourpy-1.2.1-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9ce6889abac9a42afd07a562c2d6d4b2b7134f83f18571d859b25624a331c90b"},
+ {file = "contourpy-1.2.1-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:a1eea9aecf761c661d096d39ed9026574de8adb2ae1c5bd7b33558af884fb2ce"},
+ {file = "contourpy-1.2.1-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:187fa1d4c6acc06adb0fae5544c59898ad781409e61a926ac7e84b8f276dcef4"},
+ {file = "contourpy-1.2.1-cp312-cp312-win32.whl", hash = "sha256:c2528d60e398c7c4c799d56f907664673a807635b857df18f7ae64d3e6ce2d9f"},
+ {file = "contourpy-1.2.1-cp312-cp312-win_amd64.whl", hash = "sha256:1a07fc092a4088ee952ddae19a2b2a85757b923217b7eed584fdf25f53a6e7ce"},
+ {file = "contourpy-1.2.1-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:bb6834cbd983b19f06908b45bfc2dad6ac9479ae04abe923a275b5f48f1a186b"},
+ {file = "contourpy-1.2.1-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:1d59e739ab0e3520e62a26c60707cc3ab0365d2f8fecea74bfe4de72dc56388f"},
+ {file = "contourpy-1.2.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bd3db01f59fdcbce5b22afad19e390260d6d0222f35a1023d9adc5690a889364"},
+ {file = "contourpy-1.2.1-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a12a813949e5066148712a0626895c26b2578874e4cc63160bb007e6df3436fe"},
+ {file = "contourpy-1.2.1-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:fe0ccca550bb8e5abc22f530ec0466136379c01321fd94f30a22231e8a48d985"},
+ {file = "contourpy-1.2.1-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e1d59258c3c67c865435d8fbeb35f8c59b8bef3d6f46c1f29f6123556af28445"},
+ {file = "contourpy-1.2.1-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:f32c38afb74bd98ce26de7cc74a67b40afb7b05aae7b42924ea990d51e4dac02"},
+ {file = "contourpy-1.2.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:d31a63bc6e6d87f77d71e1abbd7387ab817a66733734883d1fc0021ed9bfa083"},
+ {file = "contourpy-1.2.1-cp39-cp39-win32.whl", hash = "sha256:ddcb8581510311e13421b1f544403c16e901c4e8f09083c881fab2be80ee31ba"},
+ {file = "contourpy-1.2.1-cp39-cp39-win_amd64.whl", hash = "sha256:10a37ae557aabf2509c79715cd20b62e4c7c28b8cd62dd7d99e5ed3ce28c3fd9"},
+ {file = "contourpy-1.2.1-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:a31f94983fecbac95e58388210427d68cd30fe8a36927980fab9c20062645609"},
+ {file = "contourpy-1.2.1-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ef2b055471c0eb466033760a521efb9d8a32b99ab907fc8358481a1dd29e3bd3"},
+ {file = "contourpy-1.2.1-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:b33d2bc4f69caedcd0a275329eb2198f560b325605810895627be5d4b876bf7f"},
+ {file = "contourpy-1.2.1.tar.gz", hash = "sha256:4d8908b3bee1c889e547867ca4cdc54e5ab6be6d3e078556814a22457f49423c"},
+]
+
+[package.dependencies]
+numpy = ">=1.20"
+
+[package.extras]
+bokeh = ["bokeh", "selenium"]
+docs = ["furo", "sphinx (>=7.2)", "sphinx-copybutton"]
+mypy = ["contourpy[bokeh,docs]", "docutils-stubs", "mypy (==1.8.0)", "types-Pillow"]
+test = ["Pillow", "contourpy[test-no-images]", "matplotlib"]
+test-no-images = ["pytest", "pytest-cov", "pytest-xdist", "wurlitzer"]
+
+[[package]]
+name = "cycler"
+version = "0.12.1"
+description = "Composable style cycles"
+category = "main"
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "cycler-0.12.1-py3-none-any.whl", hash = "sha256:85cef7cff222d8644161529808465972e51340599459b8ac3ccbac5a854e0d30"},
+ {file = "cycler-0.12.1.tar.gz", hash = "sha256:88bb128f02ba341da8ef447245a9e138fae777f6a23943da4540077d3601eb1c"},
+]
+
+[package.extras]
+docs = ["ipython", "matplotlib", "numpydoc", "sphinx"]
+tests = ["pytest", "pytest-cov", "pytest-xdist"]
+
+[[package]]
+name = "exceptiongroup"
+version = "1.2.0"
+description = "Backport of PEP 654 (exception groups)"
+category = "dev"
+optional = false
+python-versions = ">=3.7"
+files = [
+ {file = "exceptiongroup-1.2.0-py3-none-any.whl", hash = "sha256:4bfd3996ac73b41e9b9628b04e079f193850720ea5945fc96a08633c66912f14"},
+ {file = "exceptiongroup-1.2.0.tar.gz", hash = "sha256:91f5c769735f051a4290d52edd0858999b57e5876e9f85937691bd4c9fa3ed68"},
+]
+
+[package.extras]
+test = ["pytest (>=6)"]
+
+[[package]]
+name = "fonttools"
+version = "4.51.0"
+description = "Tools to manipulate font files"
+category = "main"
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "fonttools-4.51.0-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:84d7751f4468dd8cdd03ddada18b8b0857a5beec80bce9f435742abc9a851a74"},
+ {file = "fonttools-4.51.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:8b4850fa2ef2cfbc1d1f689bc159ef0f45d8d83298c1425838095bf53ef46308"},
+ {file = "fonttools-4.51.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b5b48a1121117047d82695d276c2af2ee3a24ffe0f502ed581acc2673ecf1037"},
+ {file = "fonttools-4.51.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:180194c7fe60c989bb627d7ed5011f2bef1c4d36ecf3ec64daec8302f1ae0716"},
+ {file = "fonttools-4.51.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:96a48e137c36be55e68845fc4284533bda2980f8d6f835e26bca79d7e2006438"},
+ {file = "fonttools-4.51.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:806e7912c32a657fa39d2d6eb1d3012d35f841387c8fc6cf349ed70b7c340039"},
+ {file = "fonttools-4.51.0-cp310-cp310-win32.whl", hash = "sha256:32b17504696f605e9e960647c5f64b35704782a502cc26a37b800b4d69ff3c77"},
+ {file = "fonttools-4.51.0-cp310-cp310-win_amd64.whl", hash = "sha256:c7e91abdfae1b5c9e3a543f48ce96013f9a08c6c9668f1e6be0beabf0a569c1b"},
+ {file = "fonttools-4.51.0-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:a8feca65bab31479d795b0d16c9a9852902e3a3c0630678efb0b2b7941ea9c74"},
+ {file = "fonttools-4.51.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:8ac27f436e8af7779f0bb4d5425aa3535270494d3bc5459ed27de3f03151e4c2"},
+ {file = "fonttools-4.51.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0e19bd9e9964a09cd2433a4b100ca7f34e34731e0758e13ba9a1ed6e5468cc0f"},
+ {file = "fonttools-4.51.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b2b92381f37b39ba2fc98c3a45a9d6383bfc9916a87d66ccb6553f7bdd129097"},
+ {file = "fonttools-4.51.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:5f6bc991d1610f5c3bbe997b0233cbc234b8e82fa99fc0b2932dc1ca5e5afec0"},
+ {file = "fonttools-4.51.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:9696fe9f3f0c32e9a321d5268208a7cc9205a52f99b89479d1b035ed54c923f1"},
+ {file = "fonttools-4.51.0-cp311-cp311-win32.whl", hash = "sha256:3bee3f3bd9fa1d5ee616ccfd13b27ca605c2b4270e45715bd2883e9504735034"},
+ {file = "fonttools-4.51.0-cp311-cp311-win_amd64.whl", hash = "sha256:0f08c901d3866a8905363619e3741c33f0a83a680d92a9f0e575985c2634fcc1"},
+ {file = "fonttools-4.51.0-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:4060acc2bfa2d8e98117828a238889f13b6f69d59f4f2d5857eece5277b829ba"},
+ {file = "fonttools-4.51.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:1250e818b5f8a679ad79660855528120a8f0288f8f30ec88b83db51515411fcc"},
+ {file = "fonttools-4.51.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:76f1777d8b3386479ffb4a282e74318e730014d86ce60f016908d9801af9ca2a"},
+ {file = "fonttools-4.51.0-cp312-cp312-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8b5ad456813d93b9c4b7ee55302208db2b45324315129d85275c01f5cb7e61a2"},
+ {file = "fonttools-4.51.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:68b3fb7775a923be73e739f92f7e8a72725fd333eab24834041365d2278c3671"},
+ {file = "fonttools-4.51.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:8e2f1a4499e3b5ee82c19b5ee57f0294673125c65b0a1ff3764ea1f9db2f9ef5"},
+ {file = "fonttools-4.51.0-cp312-cp312-win32.whl", hash = "sha256:278e50f6b003c6aed19bae2242b364e575bcb16304b53f2b64f6551b9c000e15"},
+ {file = "fonttools-4.51.0-cp312-cp312-win_amd64.whl", hash = "sha256:b3c61423f22165541b9403ee39874dcae84cd57a9078b82e1dce8cb06b07fa2e"},
+ {file = "fonttools-4.51.0-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:1621ee57da887c17312acc4b0e7ac30d3a4fb0fec6174b2e3754a74c26bbed1e"},
+ {file = "fonttools-4.51.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:e9d9298be7a05bb4801f558522adbe2feea1b0b103d5294ebf24a92dd49b78e5"},
+ {file = "fonttools-4.51.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ee1af4be1c5afe4c96ca23badd368d8dc75f611887fb0c0dac9f71ee5d6f110e"},
+ {file = "fonttools-4.51.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c18b49adc721a7d0b8dfe7c3130c89b8704baf599fb396396d07d4aa69b824a1"},
+ {file = "fonttools-4.51.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:de7c29bdbdd35811f14493ffd2534b88f0ce1b9065316433b22d63ca1cd21f14"},
+ {file = "fonttools-4.51.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:cadf4e12a608ef1d13e039864f484c8a968840afa0258b0b843a0556497ea9ed"},
+ {file = "fonttools-4.51.0-cp38-cp38-win32.whl", hash = "sha256:aefa011207ed36cd280babfaa8510b8176f1a77261833e895a9d96e57e44802f"},
+ {file = "fonttools-4.51.0-cp38-cp38-win_amd64.whl", hash = "sha256:865a58b6e60b0938874af0968cd0553bcd88e0b2cb6e588727117bd099eef836"},
+ {file = "fonttools-4.51.0-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:60a3409c9112aec02d5fb546f557bca6efa773dcb32ac147c6baf5f742e6258b"},
+ {file = "fonttools-4.51.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:f7e89853d8bea103c8e3514b9f9dc86b5b4120afb4583b57eb10dfa5afbe0936"},
+ {file = "fonttools-4.51.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:56fc244f2585d6c00b9bcc59e6593e646cf095a96fe68d62cd4da53dd1287b55"},
+ {file = "fonttools-4.51.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0d145976194a5242fdd22df18a1b451481a88071feadf251221af110ca8f00ce"},
+ {file = "fonttools-4.51.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:c5b8cab0c137ca229433570151b5c1fc6af212680b58b15abd797dcdd9dd5051"},
+ {file = "fonttools-4.51.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:54dcf21a2f2d06ded676e3c3f9f74b2bafded3a8ff12f0983160b13e9f2fb4a7"},
+ {file = "fonttools-4.51.0-cp39-cp39-win32.whl", hash = "sha256:0118ef998a0699a96c7b28457f15546815015a2710a1b23a7bf6c1be60c01636"},
+ {file = "fonttools-4.51.0-cp39-cp39-win_amd64.whl", hash = "sha256:599bdb75e220241cedc6faebfafedd7670335d2e29620d207dd0378a4e9ccc5a"},
+ {file = "fonttools-4.51.0-py3-none-any.whl", hash = "sha256:15c94eeef6b095831067f72c825eb0e2d48bb4cea0647c1b05c981ecba2bf39f"},
+ {file = "fonttools-4.51.0.tar.gz", hash = "sha256:dc0673361331566d7a663d7ce0f6fdcbfbdc1f59c6e3ed1165ad7202ca183c68"},
+]
+
+[package.extras]
+all = ["brotli (>=1.0.1)", "brotlicffi (>=0.8.0)", "fs (>=2.2.0,<3)", "lxml (>=4.0)", "lz4 (>=1.7.4.2)", "matplotlib", "munkres", "pycairo", "scipy", "skia-pathops (>=0.5.0)", "sympy", "uharfbuzz (>=0.23.0)", "unicodedata2 (>=15.1.0)", "xattr", "zopfli (>=0.1.4)"]
+graphite = ["lz4 (>=1.7.4.2)"]
+interpolatable = ["munkres", "pycairo", "scipy"]
+lxml = ["lxml (>=4.0)"]
+pathops = ["skia-pathops (>=0.5.0)"]
+plot = ["matplotlib"]
+repacker = ["uharfbuzz (>=0.23.0)"]
+symfont = ["sympy"]
+type1 = ["xattr"]
+ufo = ["fs (>=2.2.0,<3)"]
+unicode = ["unicodedata2 (>=15.1.0)"]
+woff = ["brotli (>=1.0.1)", "brotlicffi (>=0.8.0)", "zopfli (>=0.1.4)"]
+
+[[package]]
+name = "imageio"
+version = "2.34.0"
+description = "Library for reading and writing a wide range of image, video, scientific, and volumetric data formats."
+category = "main"
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "imageio-2.34.0-py3-none-any.whl", hash = "sha256:08082bf47ccb54843d9c73fe9fc8f3a88c72452ab676b58aca74f36167e8ccba"},
+ {file = "imageio-2.34.0.tar.gz", hash = "sha256:ae9732e10acf807a22c389aef193f42215718e16bd06eed0c5bb57e1034a4d53"},
+]
+
+[package.dependencies]
+numpy = "*"
+pillow = ">=8.3.2"
+
+[package.extras]
+all-plugins = ["astropy", "av", "imageio-ffmpeg", "pillow-heif", "psutil", "tifffile"]
+all-plugins-pypy = ["av", "imageio-ffmpeg", "pillow-heif", "psutil", "tifffile"]
+build = ["wheel"]
+dev = ["black", "flake8", "fsspec[github]", "pytest", "pytest-cov"]
+docs = ["numpydoc", "pydata-sphinx-theme", "sphinx (<6)"]
+ffmpeg = ["imageio-ffmpeg", "psutil"]
+fits = ["astropy"]
+full = ["astropy", "av", "black", "flake8", "fsspec[github]", "gdal", "imageio-ffmpeg", "itk", "numpydoc", "pillow-heif", "psutil", "pydata-sphinx-theme", "pytest", "pytest-cov", "sphinx (<6)", "tifffile", "wheel"]
+gdal = ["gdal"]
+itk = ["itk"]
+linting = ["black", "flake8"]
+pillow-heif = ["pillow-heif"]
+pyav = ["av"]
+test = ["fsspec[github]", "pytest", "pytest-cov"]
+tifffile = ["tifffile"]
+
+[[package]]
+name = "importlib-resources"
+version = "6.4.0"
+description = "Read resources from Python packages"
+category = "main"
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "importlib_resources-6.4.0-py3-none-any.whl", hash = "sha256:50d10f043df931902d4194ea07ec57960f66a80449ff867bfe782b4c486ba78c"},
+ {file = "importlib_resources-6.4.0.tar.gz", hash = "sha256:cdb2b453b8046ca4e3798eb1d84f3cce1446a0e8e7b5ef4efb600f19fc398145"},
+]
+
+[package.dependencies]
+zipp = {version = ">=3.1.0", markers = "python_version < \"3.10\""}
+
+[package.extras]
+docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "rst.linker (>=1.9)", "sphinx (<7.2.5)", "sphinx (>=3.5)", "sphinx-lint"]
+testing = ["jaraco.test (>=5.4)", "pytest (>=6)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-mypy", "pytest-ruff (>=0.2.1)", "zipp (>=3.17)"]
+
+[[package]]
+name = "iniconfig"
+version = "2.0.0"
+description = "brain-dead simple config-ini parsing"
+category = "dev"
+optional = false
+python-versions = ">=3.7"
+files = [
+ {file = "iniconfig-2.0.0-py3-none-any.whl", hash = "sha256:b6a85871a79d2e3b22d2d1b94ac2824226a63c6b741c88f7ae975f18b6778374"},
+ {file = "iniconfig-2.0.0.tar.gz", hash = "sha256:2d91e135bf72d31a410b17c16da610a82cb55f6b0477d1a902134b24a455b8b3"},
+]
+
+[[package]]
+name = "joblib"
+version = "1.4.0"
+description = "Lightweight pipelining with Python functions"
+category = "main"
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "joblib-1.4.0-py3-none-any.whl", hash = "sha256:42942470d4062537be4d54c83511186da1fc14ba354961a2114da91efa9a4ed7"},
+ {file = "joblib-1.4.0.tar.gz", hash = "sha256:1eb0dc091919cd384490de890cb5dfd538410a6d4b3b54eef09fb8c50b409b1c"},
+]
+
+[[package]]
+name = "kiwisolver"
+version = "1.4.5"
+description = "A fast implementation of the Cassowary constraint solver"
+category = "main"
+optional = false
+python-versions = ">=3.7"
+files = [
+ {file = "kiwisolver-1.4.5-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:05703cf211d585109fcd72207a31bb170a0f22144d68298dc5e61b3c946518af"},
+ {file = "kiwisolver-1.4.5-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:146d14bebb7f1dc4d5fbf74f8a6cb15ac42baadee8912eb84ac0b3b2a3dc6ac3"},
+ {file = "kiwisolver-1.4.5-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:6ef7afcd2d281494c0a9101d5c571970708ad911d028137cd558f02b851c08b4"},
+ {file = "kiwisolver-1.4.5-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:9eaa8b117dc8337728e834b9c6e2611f10c79e38f65157c4c38e9400286f5cb1"},
+ {file = "kiwisolver-1.4.5-cp310-cp310-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:ec20916e7b4cbfb1f12380e46486ec4bcbaa91a9c448b97023fde0d5bbf9e4ff"},
+ {file = "kiwisolver-1.4.5-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:39b42c68602539407884cf70d6a480a469b93b81b7701378ba5e2328660c847a"},
+ {file = "kiwisolver-1.4.5-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:aa12042de0171fad672b6c59df69106d20d5596e4f87b5e8f76df757a7c399aa"},
+ {file = "kiwisolver-1.4.5-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:2a40773c71d7ccdd3798f6489aaac9eee213d566850a9533f8d26332d626b82c"},
+ {file = "kiwisolver-1.4.5-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:19df6e621f6d8b4b9c4d45f40a66839294ff2bb235e64d2178f7522d9170ac5b"},
+ {file = "kiwisolver-1.4.5-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:83d78376d0d4fd884e2c114d0621624b73d2aba4e2788182d286309ebdeed770"},
+ {file = "kiwisolver-1.4.5-cp310-cp310-musllinux_1_1_ppc64le.whl", hash = "sha256:e391b1f0a8a5a10ab3b9bb6afcfd74f2175f24f8975fb87ecae700d1503cdee0"},
+ {file = "kiwisolver-1.4.5-cp310-cp310-musllinux_1_1_s390x.whl", hash = "sha256:852542f9481f4a62dbb5dd99e8ab7aedfeb8fb6342349a181d4036877410f525"},
+ {file = "kiwisolver-1.4.5-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:59edc41b24031bc25108e210c0def6f6c2191210492a972d585a06ff246bb79b"},
+ {file = "kiwisolver-1.4.5-cp310-cp310-win32.whl", hash = "sha256:a6aa6315319a052b4ee378aa171959c898a6183f15c1e541821c5c59beaa0238"},
+ {file = "kiwisolver-1.4.5-cp310-cp310-win_amd64.whl", hash = "sha256:d0ef46024e6a3d79c01ff13801cb19d0cad7fd859b15037aec74315540acc276"},
+ {file = "kiwisolver-1.4.5-cp311-cp311-macosx_10_9_universal2.whl", hash = "sha256:11863aa14a51fd6ec28688d76f1735f8f69ab1fabf388851a595d0721af042f5"},
+ {file = "kiwisolver-1.4.5-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:8ab3919a9997ab7ef2fbbed0cc99bb28d3c13e6d4b1ad36e97e482558a91be90"},
+ {file = "kiwisolver-1.4.5-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:fcc700eadbbccbf6bc1bcb9dbe0786b4b1cb91ca0dcda336eef5c2beed37b797"},
+ {file = "kiwisolver-1.4.5-cp311-cp311-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:dfdd7c0b105af050eb3d64997809dc21da247cf44e63dc73ff0fd20b96be55a9"},
+ {file = "kiwisolver-1.4.5-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:76c6a5964640638cdeaa0c359382e5703e9293030fe730018ca06bc2010c4437"},
+ {file = "kiwisolver-1.4.5-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:bbea0db94288e29afcc4c28afbf3a7ccaf2d7e027489c449cf7e8f83c6346eb9"},
+ {file = "kiwisolver-1.4.5-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:ceec1a6bc6cab1d6ff5d06592a91a692f90ec7505d6463a88a52cc0eb58545da"},
+ {file = "kiwisolver-1.4.5-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:040c1aebeda72197ef477a906782b5ab0d387642e93bda547336b8957c61022e"},
+ {file = "kiwisolver-1.4.5-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:f91de7223d4c7b793867797bacd1ee53bfe7359bd70d27b7b58a04efbb9436c8"},
+ {file = "kiwisolver-1.4.5-cp311-cp311-musllinux_1_1_i686.whl", hash = "sha256:faae4860798c31530dd184046a900e652c95513796ef51a12bc086710c2eec4d"},
+ {file = "kiwisolver-1.4.5-cp311-cp311-musllinux_1_1_ppc64le.whl", hash = "sha256:b0157420efcb803e71d1b28e2c287518b8808b7cf1ab8af36718fd0a2c453eb0"},
+ {file = "kiwisolver-1.4.5-cp311-cp311-musllinux_1_1_s390x.whl", hash = "sha256:06f54715b7737c2fecdbf140d1afb11a33d59508a47bf11bb38ecf21dc9ab79f"},
+ {file = "kiwisolver-1.4.5-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:fdb7adb641a0d13bdcd4ef48e062363d8a9ad4a182ac7647ec88f695e719ae9f"},
+ {file = "kiwisolver-1.4.5-cp311-cp311-win32.whl", hash = "sha256:bb86433b1cfe686da83ce32a9d3a8dd308e85c76b60896d58f082136f10bffac"},
+ {file = "kiwisolver-1.4.5-cp311-cp311-win_amd64.whl", hash = "sha256:6c08e1312a9cf1074d17b17728d3dfce2a5125b2d791527f33ffbe805200a355"},
+ {file = "kiwisolver-1.4.5-cp312-cp312-macosx_10_9_universal2.whl", hash = "sha256:32d5cf40c4f7c7b3ca500f8985eb3fb3a7dfc023215e876f207956b5ea26632a"},
+ {file = "kiwisolver-1.4.5-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:f846c260f483d1fd217fe5ed7c173fb109efa6b1fc8381c8b7552c5781756192"},
+ {file = "kiwisolver-1.4.5-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:5ff5cf3571589b6d13bfbfd6bcd7a3f659e42f96b5fd1c4830c4cf21d4f5ef45"},
+ {file = "kiwisolver-1.4.5-cp312-cp312-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:7269d9e5f1084a653d575c7ec012ff57f0c042258bf5db0954bf551c158466e7"},
+ {file = "kiwisolver-1.4.5-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:da802a19d6e15dffe4b0c24b38b3af68e6c1a68e6e1d8f30148c83864f3881db"},
+ {file = "kiwisolver-1.4.5-cp312-cp312-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3aba7311af82e335dd1e36ffff68aaca609ca6290c2cb6d821a39aa075d8e3ff"},
+ {file = "kiwisolver-1.4.5-cp312-cp312-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:763773d53f07244148ccac5b084da5adb90bfaee39c197554f01b286cf869228"},
+ {file = "kiwisolver-1.4.5-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2270953c0d8cdab5d422bee7d2007f043473f9d2999631c86a223c9db56cbd16"},
+ {file = "kiwisolver-1.4.5-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:d099e745a512f7e3bbe7249ca835f4d357c586d78d79ae8f1dcd4d8adeb9bda9"},
+ {file = "kiwisolver-1.4.5-cp312-cp312-musllinux_1_1_i686.whl", hash = "sha256:74db36e14a7d1ce0986fa104f7d5637aea5c82ca6326ed0ec5694280942d1162"},
+ {file = "kiwisolver-1.4.5-cp312-cp312-musllinux_1_1_ppc64le.whl", hash = "sha256:7e5bab140c309cb3a6ce373a9e71eb7e4873c70c2dda01df6820474f9889d6d4"},
+ {file = "kiwisolver-1.4.5-cp312-cp312-musllinux_1_1_s390x.whl", hash = "sha256:0f114aa76dc1b8f636d077979c0ac22e7cd8f3493abbab152f20eb8d3cda71f3"},
+ {file = "kiwisolver-1.4.5-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:88a2df29d4724b9237fc0c6eaf2a1adae0cdc0b3e9f4d8e7dc54b16812d2d81a"},
+ {file = "kiwisolver-1.4.5-cp312-cp312-win32.whl", hash = "sha256:72d40b33e834371fd330fb1472ca19d9b8327acb79a5821d4008391db8e29f20"},
+ {file = "kiwisolver-1.4.5-cp312-cp312-win_amd64.whl", hash = "sha256:2c5674c4e74d939b9d91dda0fae10597ac7521768fec9e399c70a1f27e2ea2d9"},
+ {file = "kiwisolver-1.4.5-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:3a2b053a0ab7a3960c98725cfb0bf5b48ba82f64ec95fe06f1d06c99b552e130"},
+ {file = "kiwisolver-1.4.5-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3cd32d6c13807e5c66a7cbb79f90b553642f296ae4518a60d8d76243b0ad2898"},
+ {file = "kiwisolver-1.4.5-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:59ec7b7c7e1a61061850d53aaf8e93db63dce0c936db1fda2658b70e4a1be709"},
+ {file = "kiwisolver-1.4.5-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:da4cfb373035def307905d05041c1d06d8936452fe89d464743ae7fb8371078b"},
+ {file = "kiwisolver-1.4.5-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:2400873bccc260b6ae184b2b8a4fec0e4082d30648eadb7c3d9a13405d861e89"},
+ {file = "kiwisolver-1.4.5-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:1b04139c4236a0f3aff534479b58f6f849a8b351e1314826c2d230849ed48985"},
+ {file = "kiwisolver-1.4.5-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:4e66e81a5779b65ac21764c295087de82235597a2293d18d943f8e9e32746265"},
+ {file = "kiwisolver-1.4.5-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:7931d8f1f67c4be9ba1dd9c451fb0eeca1a25b89e4d3f89e828fe12a519b782a"},
+ {file = "kiwisolver-1.4.5-cp37-cp37m-musllinux_1_1_ppc64le.whl", hash = "sha256:b3f7e75f3015df442238cca659f8baa5f42ce2a8582727981cbfa15fee0ee205"},
+ {file = "kiwisolver-1.4.5-cp37-cp37m-musllinux_1_1_s390x.whl", hash = "sha256:bbf1d63eef84b2e8c89011b7f2235b1e0bf7dacc11cac9431fc6468e99ac77fb"},
+ {file = "kiwisolver-1.4.5-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:4c380469bd3f970ef677bf2bcba2b6b0b4d5c75e7a020fb863ef75084efad66f"},
+ {file = "kiwisolver-1.4.5-cp37-cp37m-win32.whl", hash = "sha256:9408acf3270c4b6baad483865191e3e582b638b1654a007c62e3efe96f09a9a3"},
+ {file = "kiwisolver-1.4.5-cp37-cp37m-win_amd64.whl", hash = "sha256:5b94529f9b2591b7af5f3e0e730a4e0a41ea174af35a4fd067775f9bdfeee01a"},
+ {file = "kiwisolver-1.4.5-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:11c7de8f692fc99816e8ac50d1d1aef4f75126eefc33ac79aac02c099fd3db71"},
+ {file = "kiwisolver-1.4.5-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:53abb58632235cd154176ced1ae8f0d29a6657aa1aa9decf50b899b755bc2b93"},
+ {file = "kiwisolver-1.4.5-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:88b9f257ca61b838b6f8094a62418421f87ac2a1069f7e896c36a7d86b5d4c29"},
+ {file = "kiwisolver-1.4.5-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:3195782b26fc03aa9c6913d5bad5aeb864bdc372924c093b0f1cebad603dd712"},
+ {file = "kiwisolver-1.4.5-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:fc579bf0f502e54926519451b920e875f433aceb4624a3646b3252b5caa9e0b6"},
+ {file = "kiwisolver-1.4.5-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:5a580c91d686376f0f7c295357595c5a026e6cbc3d77b7c36e290201e7c11ecb"},
+ {file = "kiwisolver-1.4.5-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:cfe6ab8da05c01ba6fbea630377b5da2cd9bcbc6338510116b01c1bc939a2c18"},
+ {file = "kiwisolver-1.4.5-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:d2e5a98f0ec99beb3c10e13b387f8db39106d53993f498b295f0c914328b1333"},
+ {file = "kiwisolver-1.4.5-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:a51a263952b1429e429ff236d2f5a21c5125437861baeed77f5e1cc2d2c7c6da"},
+ {file = "kiwisolver-1.4.5-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:3edd2fa14e68c9be82c5b16689e8d63d89fe927e56debd6e1dbce7a26a17f81b"},
+ {file = "kiwisolver-1.4.5-cp38-cp38-musllinux_1_1_ppc64le.whl", hash = "sha256:74d1b44c6cfc897df648cc9fdaa09bc3e7679926e6f96df05775d4fb3946571c"},
+ {file = "kiwisolver-1.4.5-cp38-cp38-musllinux_1_1_s390x.whl", hash = "sha256:76d9289ed3f7501012e05abb8358bbb129149dbd173f1f57a1bf1c22d19ab7cc"},
+ {file = "kiwisolver-1.4.5-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:92dea1ffe3714fa8eb6a314d2b3c773208d865a0e0d35e713ec54eea08a66250"},
+ {file = "kiwisolver-1.4.5-cp38-cp38-win32.whl", hash = "sha256:5c90ae8c8d32e472be041e76f9d2f2dbff4d0b0be8bd4041770eddb18cf49a4e"},
+ {file = "kiwisolver-1.4.5-cp38-cp38-win_amd64.whl", hash = "sha256:c7940c1dc63eb37a67721b10d703247552416f719c4188c54e04334321351ced"},
+ {file = "kiwisolver-1.4.5-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:9407b6a5f0d675e8a827ad8742e1d6b49d9c1a1da5d952a67d50ef5f4170b18d"},
+ {file = "kiwisolver-1.4.5-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:15568384086b6df3c65353820a4473575dbad192e35010f622c6ce3eebd57af9"},
+ {file = "kiwisolver-1.4.5-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:0dc9db8e79f0036e8173c466d21ef18e1befc02de8bf8aa8dc0813a6dc8a7046"},
+ {file = "kiwisolver-1.4.5-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:cdc8a402aaee9a798b50d8b827d7ecf75edc5fb35ea0f91f213ff927c15f4ff0"},
+ {file = "kiwisolver-1.4.5-cp39-cp39-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:6c3bd3cde54cafb87d74d8db50b909705c62b17c2099b8f2e25b461882e544ff"},
+ {file = "kiwisolver-1.4.5-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:955e8513d07a283056b1396e9a57ceddbd272d9252c14f154d450d227606eb54"},
+ {file = "kiwisolver-1.4.5-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:346f5343b9e3f00b8db8ba359350eb124b98c99efd0b408728ac6ebf38173958"},
+ {file = "kiwisolver-1.4.5-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:b9098e0049e88c6a24ff64545cdfc50807818ba6c1b739cae221bbbcbc58aad3"},
+ {file = "kiwisolver-1.4.5-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:00bd361b903dc4bbf4eb165f24d1acbee754fce22ded24c3d56eec268658a5cf"},
+ {file = "kiwisolver-1.4.5-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:7b8b454bac16428b22560d0a1cf0a09875339cab69df61d7805bf48919415901"},
+ {file = "kiwisolver-1.4.5-cp39-cp39-musllinux_1_1_ppc64le.whl", hash = "sha256:f1d072c2eb0ad60d4c183f3fb44ac6f73fb7a8f16a2694a91f988275cbf352f9"},
+ {file = "kiwisolver-1.4.5-cp39-cp39-musllinux_1_1_s390x.whl", hash = "sha256:31a82d498054cac9f6d0b53d02bb85811185bcb477d4b60144f915f3b3126342"},
+ {file = "kiwisolver-1.4.5-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:6512cb89e334e4700febbffaaa52761b65b4f5a3cf33f960213d5656cea36a77"},
+ {file = "kiwisolver-1.4.5-cp39-cp39-win32.whl", hash = "sha256:9db8ea4c388fdb0f780fe91346fd438657ea602d58348753d9fb265ce1bca67f"},
+ {file = "kiwisolver-1.4.5-cp39-cp39-win_amd64.whl", hash = "sha256:59415f46a37f7f2efeec758353dd2eae1b07640d8ca0f0c42548ec4125492635"},
+ {file = "kiwisolver-1.4.5-pp37-pypy37_pp73-macosx_10_9_x86_64.whl", hash = "sha256:5c7b3b3a728dc6faf3fc372ef24f21d1e3cee2ac3e9596691d746e5a536de920"},
+ {file = "kiwisolver-1.4.5-pp37-pypy37_pp73-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:620ced262a86244e2be10a676b646f29c34537d0d9cc8eb26c08f53d98013390"},
+ {file = "kiwisolver-1.4.5-pp37-pypy37_pp73-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:378a214a1e3bbf5ac4a8708304318b4f890da88c9e6a07699c4ae7174c09a68d"},
+ {file = "kiwisolver-1.4.5-pp37-pypy37_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:aaf7be1207676ac608a50cd08f102f6742dbfc70e8d60c4db1c6897f62f71523"},
+ {file = "kiwisolver-1.4.5-pp37-pypy37_pp73-win_amd64.whl", hash = "sha256:ba55dce0a9b8ff59495ddd050a0225d58bd0983d09f87cfe2b6aec4f2c1234e4"},
+ {file = "kiwisolver-1.4.5-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:fd32ea360bcbb92d28933fc05ed09bffcb1704ba3fc7942e81db0fd4f81a7892"},
+ {file = "kiwisolver-1.4.5-pp38-pypy38_pp73-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:5e7139af55d1688f8b960ee9ad5adafc4ac17c1c473fe07133ac092310d76544"},
+ {file = "kiwisolver-1.4.5-pp38-pypy38_pp73-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:dced8146011d2bc2e883f9bd68618b8247387f4bbec46d7392b3c3b032640126"},
+ {file = "kiwisolver-1.4.5-pp38-pypy38_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c9bf3325c47b11b2e51bca0824ea217c7cd84491d8ac4eefd1e409705ef092bd"},
+ {file = "kiwisolver-1.4.5-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:5794cf59533bc3f1b1c821f7206a3617999db9fbefc345360aafe2e067514929"},
+ {file = "kiwisolver-1.4.5-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:e368f200bbc2e4f905b8e71eb38b3c04333bddaa6a2464a6355487b02bb7fb09"},
+ {file = "kiwisolver-1.4.5-pp39-pypy39_pp73-manylinux_2_12_i686.manylinux2010_i686.manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:e5d706eba36b4c4d5bc6c6377bb6568098765e990cfc21ee16d13963fab7b3e7"},
+ {file = "kiwisolver-1.4.5-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:85267bd1aa8880a9c88a8cb71e18d3d64d2751a790e6ca6c27b8ccc724bcd5ad"},
+ {file = "kiwisolver-1.4.5-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:210ef2c3a1f03272649aff1ef992df2e724748918c4bc2d5a90352849eb40bea"},
+ {file = "kiwisolver-1.4.5-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:11d011a7574eb3b82bcc9c1a1d35c1d7075677fdd15de527d91b46bd35e935ee"},
+ {file = "kiwisolver-1.4.5.tar.gz", hash = "sha256:e57e563a57fb22a142da34f38acc2fc1a5c864bc29ca1517a88abc963e60d6ec"},
+]
+
+[[package]]
+name = "lazy-loader"
+version = "0.4"
+description = "Makes it easy to load subpackages and functions on demand."
+category = "main"
+optional = false
+python-versions = ">=3.7"
+files = [
+ {file = "lazy_loader-0.4-py3-none-any.whl", hash = "sha256:342aa8e14d543a154047afb4ba8ef17f5563baad3fc610d7b15b213b0f119efc"},
+ {file = "lazy_loader-0.4.tar.gz", hash = "sha256:47c75182589b91a4e1a85a136c074285a5ad4d9f39c63e0d7fb76391c4574cd1"},
+]
+
+[package.dependencies]
+packaging = "*"
+
+[package.extras]
+dev = ["changelist (==0.5)"]
+lint = ["pre-commit (==3.7.0)"]
+test = ["pytest (>=7.4)", "pytest-cov (>=4.1)"]
+
+[[package]]
+name = "matplotlib"
+version = "3.8.4"
+description = "Python plotting package"
+category = "main"
+optional = false
+python-versions = ">=3.9"
+files = [
+ {file = "matplotlib-3.8.4-cp310-cp310-macosx_10_12_x86_64.whl", hash = "sha256:abc9d838f93583650c35eca41cfcec65b2e7cb50fd486da6f0c49b5e1ed23014"},
+ {file = "matplotlib-3.8.4-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:8f65c9f002d281a6e904976007b2d46a1ee2bcea3a68a8c12dda24709ddc9106"},
+ {file = "matplotlib-3.8.4-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ce1edd9f5383b504dbc26eeea404ed0a00656c526638129028b758fd43fc5f10"},
+ {file = "matplotlib-3.8.4-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ecd79298550cba13a43c340581a3ec9c707bd895a6a061a78fa2524660482fc0"},
+ {file = "matplotlib-3.8.4-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:90df07db7b599fe7035d2f74ab7e438b656528c68ba6bb59b7dc46af39ee48ef"},
+ {file = "matplotlib-3.8.4-cp310-cp310-win_amd64.whl", hash = "sha256:ac24233e8f2939ac4fd2919eed1e9c0871eac8057666070e94cbf0b33dd9c338"},
+ {file = "matplotlib-3.8.4-cp311-cp311-macosx_10_12_x86_64.whl", hash = "sha256:72f9322712e4562e792b2961971891b9fbbb0e525011e09ea0d1f416c4645661"},
+ {file = "matplotlib-3.8.4-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:232ce322bfd020a434caaffbd9a95333f7c2491e59cfc014041d95e38ab90d1c"},
+ {file = "matplotlib-3.8.4-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6addbd5b488aedb7f9bc19f91cd87ea476206f45d7116fcfe3d31416702a82fa"},
+ {file = "matplotlib-3.8.4-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:cc4ccdc64e3039fc303defd119658148f2349239871db72cd74e2eeaa9b80b71"},
+ {file = "matplotlib-3.8.4-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:b7a2a253d3b36d90c8993b4620183b55665a429da8357a4f621e78cd48b2b30b"},
+ {file = "matplotlib-3.8.4-cp311-cp311-win_amd64.whl", hash = "sha256:8080d5081a86e690d7688ffa542532e87f224c38a6ed71f8fbed34dd1d9fedae"},
+ {file = "matplotlib-3.8.4-cp312-cp312-macosx_10_12_x86_64.whl", hash = "sha256:6485ac1f2e84676cff22e693eaa4fbed50ef5dc37173ce1f023daef4687df616"},
+ {file = "matplotlib-3.8.4-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:c89ee9314ef48c72fe92ce55c4e95f2f39d70208f9f1d9db4e64079420d8d732"},
+ {file = "matplotlib-3.8.4-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:50bac6e4d77e4262c4340d7a985c30912054745ec99756ce213bfbc3cb3808eb"},
+ {file = "matplotlib-3.8.4-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f51c4c869d4b60d769f7b4406eec39596648d9d70246428745a681c327a8ad30"},
+ {file = "matplotlib-3.8.4-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:b12ba985837e4899b762b81f5b2845bd1a28f4fdd1a126d9ace64e9c4eb2fb25"},
+ {file = "matplotlib-3.8.4-cp312-cp312-win_amd64.whl", hash = "sha256:7a6769f58ce51791b4cb8b4d7642489df347697cd3e23d88266aaaee93b41d9a"},
+ {file = "matplotlib-3.8.4-cp39-cp39-macosx_10_12_x86_64.whl", hash = "sha256:843cbde2f0946dadd8c5c11c6d91847abd18ec76859dc319362a0964493f0ba6"},
+ {file = "matplotlib-3.8.4-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:1c13f041a7178f9780fb61cc3a2b10423d5e125480e4be51beaf62b172413b67"},
+ {file = "matplotlib-3.8.4-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fb44f53af0a62dc80bba4443d9b27f2fde6acfdac281d95bc872dc148a6509cc"},
+ {file = "matplotlib-3.8.4-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:606e3b90897554c989b1e38a258c626d46c873523de432b1462f295db13de6f9"},
+ {file = "matplotlib-3.8.4-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:9bb0189011785ea794ee827b68777db3ca3f93f3e339ea4d920315a0e5a78d54"},
+ {file = "matplotlib-3.8.4-cp39-cp39-win_amd64.whl", hash = "sha256:6209e5c9aaccc056e63b547a8152661324404dd92340a6e479b3a7f24b42a5d0"},
+ {file = "matplotlib-3.8.4-pp39-pypy39_pp73-macosx_10_12_x86_64.whl", hash = "sha256:c7064120a59ce6f64103c9cefba8ffe6fba87f2c61d67c401186423c9a20fd35"},
+ {file = "matplotlib-3.8.4-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a0e47eda4eb2614300fc7bb4657fced3e83d6334d03da2173b09e447418d499f"},
+ {file = "matplotlib-3.8.4-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:493e9f6aa5819156b58fce42b296ea31969f2aab71c5b680b4ea7a3cb5c07d94"},
+ {file = "matplotlib-3.8.4.tar.gz", hash = "sha256:8aac397d5e9ec158960e31c381c5ffc52ddd52bd9a47717e2a694038167dffea"},
+]
+
+[package.dependencies]
+contourpy = ">=1.0.1"
+cycler = ">=0.10"
+fonttools = ">=4.22.0"
+importlib-resources = {version = ">=3.2.0", markers = "python_version < \"3.10\""}
+kiwisolver = ">=1.3.1"
+numpy = ">=1.21"
+packaging = ">=20.0"
+pillow = ">=8"
+pyparsing = ">=2.3.1"
+python-dateutil = ">=2.7"
+
+[[package]]
+name = "mpmath"
+version = "1.3.0"
+description = "Python library for arbitrary-precision floating-point arithmetic"
+category = "main"
+optional = false
+python-versions = "*"
+files = [
+ {file = "mpmath-1.3.0-py3-none-any.whl", hash = "sha256:a0b2b9fe80bbcd81a6647ff13108738cfb482d481d826cc0e02f5b35e5c88d2c"},
+ {file = "mpmath-1.3.0.tar.gz", hash = "sha256:7a28eb2a9774d00c7bc92411c19a89209d5da7c4c9a9e227be8330a23a25b91f"},
+]
+
+[package.extras]
+develop = ["codecov", "pycodestyle", "pytest (>=4.6)", "pytest-cov", "wheel"]
+docs = ["sphinx"]
+gmpy = ["gmpy2 (>=2.1.0a4)"]
+tests = ["pytest (>=4.6)"]
+
+[[package]]
+name = "networkx"
+version = "3.2.1"
+description = "Python package for creating and manipulating graphs and networks"
+category = "main"
+optional = false
+python-versions = ">=3.9"
+files = [
+ {file = "networkx-3.2.1-py3-none-any.whl", hash = "sha256:f18c69adc97877c42332c170849c96cefa91881c99a7cb3e95b7c659ebdc1ec2"},
+ {file = "networkx-3.2.1.tar.gz", hash = "sha256:9f1bb5cf3409bf324e0a722c20bdb4c20ee39bf1c30ce8ae499c8502b0b5e0c6"},
+]
+
+[package.extras]
+default = ["matplotlib (>=3.5)", "numpy (>=1.22)", "pandas (>=1.4)", "scipy (>=1.9,!=1.11.0,!=1.11.1)"]
+developer = ["changelist (==0.4)", "mypy (>=1.1)", "pre-commit (>=3.2)", "rtoml"]
+doc = ["nb2plots (>=0.7)", "nbconvert (<7.9)", "numpydoc (>=1.6)", "pillow (>=9.4)", "pydata-sphinx-theme (>=0.14)", "sphinx (>=7)", "sphinx-gallery (>=0.14)", "texext (>=0.6.7)"]
+extra = ["lxml (>=4.6)", "pydot (>=1.4.2)", "pygraphviz (>=1.11)", "sympy (>=1.10)"]
+test = ["pytest (>=7.2)", "pytest-cov (>=4.0)"]
+
+[[package]]
+name = "nibabel"
+version = "5.2.1"
+description = "Access a multitude of neuroimaging data formats"
+category = "main"
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "nibabel-5.2.1-py3-none-any.whl", hash = "sha256:2cbbc22985f7f9d39d050df47249771dfb8d48447f5e7a993177e4cabfe047f0"},
+ {file = "nibabel-5.2.1.tar.gz", hash = "sha256:b6c80b2e728e4bc2b65f1142d9b8d2287a9102a8bf8477e115ef0d8334559975"},
+]
+
+[package.dependencies]
+numpy = ">=1.20"
+packaging = ">=17"
+
+[package.extras]
+all = ["nibabel[dicomfs,minc2,spm,zstd]"]
+dev = ["tox"]
+dicom = ["pydicom (>=1.0.0)"]
+dicomfs = ["nibabel[dicom]", "pillow"]
+doc = ["matplotlib (>=1.5.3)", "numpydoc", "sphinx", "texext", "tomli"]
+doctest = ["tox"]
+minc2 = ["h5py"]
+spm = ["scipy"]
+style = ["tox"]
+test = ["pytest", "pytest-cov", "pytest-doctestplus", "pytest-httpserver", "pytest-xdist"]
+typing = ["tox"]
+zstd = ["pyzstd (>=0.14.3)"]
+
+[[package]]
+name = "numpy"
+version = "1.26.4"
+description = "Fundamental package for array computing in Python"
+category = "main"
+optional = false
+python-versions = ">=3.9"
+files = [
+ {file = "numpy-1.26.4-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:9ff0f4f29c51e2803569d7a51c2304de5554655a60c5d776e35b4a41413830d0"},
+ {file = "numpy-1.26.4-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:2e4ee3380d6de9c9ec04745830fd9e2eccb3e6cf790d39d7b98ffd19b0dd754a"},
+ {file = "numpy-1.26.4-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d209d8969599b27ad20994c8e41936ee0964e6da07478d6c35016bc386b66ad4"},
+ {file = "numpy-1.26.4-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ffa75af20b44f8dba823498024771d5ac50620e6915abac414251bd971b4529f"},
+ {file = "numpy-1.26.4-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:62b8e4b1e28009ef2846b4c7852046736bab361f7aeadeb6a5b89ebec3c7055a"},
+ {file = "numpy-1.26.4-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:a4abb4f9001ad2858e7ac189089c42178fcce737e4169dc61321660f1a96c7d2"},
+ {file = "numpy-1.26.4-cp310-cp310-win32.whl", hash = "sha256:bfe25acf8b437eb2a8b2d49d443800a5f18508cd811fea3181723922a8a82b07"},
+ {file = "numpy-1.26.4-cp310-cp310-win_amd64.whl", hash = "sha256:b97fe8060236edf3662adfc2c633f56a08ae30560c56310562cb4f95500022d5"},
+ {file = "numpy-1.26.4-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:4c66707fabe114439db9068ee468c26bbdf909cac0fb58686a42a24de1760c71"},
+ {file = "numpy-1.26.4-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:edd8b5fe47dab091176d21bb6de568acdd906d1887a4584a15a9a96a1dca06ef"},
+ {file = "numpy-1.26.4-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:7ab55401287bfec946ced39700c053796e7cc0e3acbef09993a9ad2adba6ca6e"},
+ {file = "numpy-1.26.4-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:666dbfb6ec68962c033a450943ded891bed2d54e6755e35e5835d63f4f6931d5"},
+ {file = "numpy-1.26.4-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:96ff0b2ad353d8f990b63294c8986f1ec3cb19d749234014f4e7eb0112ceba5a"},
+ {file = "numpy-1.26.4-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:60dedbb91afcbfdc9bc0b1f3f402804070deed7392c23eb7a7f07fa857868e8a"},
+ {file = "numpy-1.26.4-cp311-cp311-win32.whl", hash = "sha256:1af303d6b2210eb850fcf03064d364652b7120803a0b872f5211f5234b399f20"},
+ {file = "numpy-1.26.4-cp311-cp311-win_amd64.whl", hash = "sha256:cd25bcecc4974d09257ffcd1f098ee778f7834c3ad767fe5db785be9a4aa9cb2"},
+ {file = "numpy-1.26.4-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:b3ce300f3644fb06443ee2222c2201dd3a89ea6040541412b8fa189341847218"},
+ {file = "numpy-1.26.4-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:03a8c78d01d9781b28a6989f6fa1bb2c4f2d51201cf99d3dd875df6fbd96b23b"},
+ {file = "numpy-1.26.4-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9fad7dcb1aac3c7f0584a5a8133e3a43eeb2fe127f47e3632d43d677c66c102b"},
+ {file = "numpy-1.26.4-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:675d61ffbfa78604709862923189bad94014bef562cc35cf61d3a07bba02a7ed"},
+ {file = "numpy-1.26.4-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:ab47dbe5cc8210f55aa58e4805fe224dac469cde56b9f731a4c098b91917159a"},
+ {file = "numpy-1.26.4-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:1dda2e7b4ec9dd512f84935c5f126c8bd8b9f2fc001e9f54af255e8c5f16b0e0"},
+ {file = "numpy-1.26.4-cp312-cp312-win32.whl", hash = "sha256:50193e430acfc1346175fcbdaa28ffec49947a06918b7b92130744e81e640110"},
+ {file = "numpy-1.26.4-cp312-cp312-win_amd64.whl", hash = "sha256:08beddf13648eb95f8d867350f6a018a4be2e5ad54c8d8caed89ebca558b2818"},
+ {file = "numpy-1.26.4-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:7349ab0fa0c429c82442a27a9673fc802ffdb7c7775fad780226cb234965e53c"},
+ {file = "numpy-1.26.4-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:52b8b60467cd7dd1e9ed082188b4e6bb35aa5cdd01777621a1658910745b90be"},
+ {file = "numpy-1.26.4-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d5241e0a80d808d70546c697135da2c613f30e28251ff8307eb72ba696945764"},
+ {file = "numpy-1.26.4-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:f870204a840a60da0b12273ef34f7051e98c3b5961b61b0c2c1be6dfd64fbcd3"},
+ {file = "numpy-1.26.4-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:679b0076f67ecc0138fd2ede3a8fd196dddc2ad3254069bcb9faf9a79b1cebcd"},
+ {file = "numpy-1.26.4-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:47711010ad8555514b434df65f7d7b076bb8261df1ca9bb78f53d3b2db02e95c"},
+ {file = "numpy-1.26.4-cp39-cp39-win32.whl", hash = "sha256:a354325ee03388678242a4d7ebcd08b5c727033fcff3b2f536aea978e15ee9e6"},
+ {file = "numpy-1.26.4-cp39-cp39-win_amd64.whl", hash = "sha256:3373d5d70a5fe74a2c1bb6d2cfd9609ecf686d47a2d7b1d37a8f3b6bf6003aea"},
+ {file = "numpy-1.26.4-pp39-pypy39_pp73-macosx_10_9_x86_64.whl", hash = "sha256:afedb719a9dcfc7eaf2287b839d8198e06dcd4cb5d276a3df279231138e83d30"},
+ {file = "numpy-1.26.4-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:95a7476c59002f2f6c590b9b7b998306fba6a5aa646b1e22ddfeaf8f78c3a29c"},
+ {file = "numpy-1.26.4-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:7e50d0a0cc3189f9cb0aeb3a6a6af18c16f59f004b866cd2be1c14b36134a4a0"},
+ {file = "numpy-1.26.4.tar.gz", hash = "sha256:2a02aba9ed12e4ac4eb3ea9421c420301a0c6460d9830d74a9df87efa4912010"},
+]
+
+[[package]]
+name = "packaging"
+version = "24.0"
+description = "Core utilities for Python packages"
+category = "main"
+optional = false
+python-versions = ">=3.7"
+files = [
+ {file = "packaging-24.0-py3-none-any.whl", hash = "sha256:2ddfb553fdf02fb784c234c7ba6ccc288296ceabec964ad2eae3777778130bc5"},
+ {file = "packaging-24.0.tar.gz", hash = "sha256:eb82c5e3e56209074766e6885bb04b8c38a0c015d0a30036ebe7ece34c9989e9"},
+]
+
+[[package]]
+name = "pandas"
+version = "2.2.2"
+description = "Powerful data structures for data analysis, time series, and statistics"
+category = "main"
+optional = false
+python-versions = ">=3.9"
+files = [
+ {file = "pandas-2.2.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:90c6fca2acf139569e74e8781709dccb6fe25940488755716d1d354d6bc58bce"},
+ {file = "pandas-2.2.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:c7adfc142dac335d8c1e0dcbd37eb8617eac386596eb9e1a1b77791cf2498238"},
+ {file = "pandas-2.2.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4abfe0be0d7221be4f12552995e58723c7422c80a659da13ca382697de830c08"},
+ {file = "pandas-2.2.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8635c16bf3d99040fdf3ca3db669a7250ddf49c55dc4aa8fe0ae0fa8d6dcc1f0"},
+ {file = "pandas-2.2.2-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:40ae1dffb3967a52203105a077415a86044a2bea011b5f321c6aa64b379a3f51"},
+ {file = "pandas-2.2.2-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:8e5a0b00e1e56a842f922e7fae8ae4077aee4af0acb5ae3622bd4b4c30aedf99"},
+ {file = "pandas-2.2.2-cp310-cp310-win_amd64.whl", hash = "sha256:ddf818e4e6c7c6f4f7c8a12709696d193976b591cc7dc50588d3d1a6b5dc8772"},
+ {file = "pandas-2.2.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:696039430f7a562b74fa45f540aca068ea85fa34c244d0deee539cb6d70aa288"},
+ {file = "pandas-2.2.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:8e90497254aacacbc4ea6ae5e7a8cd75629d6ad2b30025a4a8b09aa4faf55151"},
+ {file = "pandas-2.2.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:58b84b91b0b9f4bafac2a0ac55002280c094dfc6402402332c0913a59654ab2b"},
+ {file = "pandas-2.2.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6d2123dc9ad6a814bcdea0f099885276b31b24f7edf40f6cdbc0912672e22eee"},
+ {file = "pandas-2.2.2-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:2925720037f06e89af896c70bca73459d7e6a4be96f9de79e2d440bd499fe0db"},
+ {file = "pandas-2.2.2-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:0cace394b6ea70c01ca1595f839cf193df35d1575986e484ad35c4aeae7266c1"},
+ {file = "pandas-2.2.2-cp311-cp311-win_amd64.whl", hash = "sha256:873d13d177501a28b2756375d59816c365e42ed8417b41665f346289adc68d24"},
+ {file = "pandas-2.2.2-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:9dfde2a0ddef507a631dc9dc4af6a9489d5e2e740e226ad426a05cabfbd7c8ef"},
+ {file = "pandas-2.2.2-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:e9b79011ff7a0f4b1d6da6a61aa1aa604fb312d6647de5bad20013682d1429ce"},
+ {file = "pandas-2.2.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1cb51fe389360f3b5a4d57dbd2848a5f033350336ca3b340d1c53a1fad33bcad"},
+ {file = "pandas-2.2.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:eee3a87076c0756de40b05c5e9a6069c035ba43e8dd71c379e68cab2c20f16ad"},
+ {file = "pandas-2.2.2-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:3e374f59e440d4ab45ca2fffde54b81ac3834cf5ae2cdfa69c90bc03bde04d76"},
+ {file = "pandas-2.2.2-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:43498c0bdb43d55cb162cdc8c06fac328ccb5d2eabe3cadeb3529ae6f0517c32"},
+ {file = "pandas-2.2.2-cp312-cp312-win_amd64.whl", hash = "sha256:d187d355ecec3629624fccb01d104da7d7f391db0311145817525281e2804d23"},
+ {file = "pandas-2.2.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:0ca6377b8fca51815f382bd0b697a0814c8bda55115678cbc94c30aacbb6eff2"},
+ {file = "pandas-2.2.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:9057e6aa78a584bc93a13f0a9bf7e753a5e9770a30b4d758b8d5f2a62a9433cd"},
+ {file = "pandas-2.2.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:001910ad31abc7bf06f49dcc903755d2f7f3a9186c0c040b827e522e9cef0863"},
+ {file = "pandas-2.2.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:66b479b0bd07204e37583c191535505410daa8df638fd8e75ae1b383851fe921"},
+ {file = "pandas-2.2.2-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:a77e9d1c386196879aa5eb712e77461aaee433e54c68cf253053a73b7e49c33a"},
+ {file = "pandas-2.2.2-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:92fd6b027924a7e178ac202cfbe25e53368db90d56872d20ffae94b96c7acc57"},
+ {file = "pandas-2.2.2-cp39-cp39-win_amd64.whl", hash = "sha256:640cef9aa381b60e296db324337a554aeeb883ead99dc8f6c18e81a93942f5f4"},
+ {file = "pandas-2.2.2.tar.gz", hash = "sha256:9e79019aba43cb4fda9e4d983f8e88ca0373adbb697ae9c6c43093218de28b54"},
+]
+
+[package.dependencies]
+numpy = [
+ {version = ">=1.22.4", markers = "python_version < \"3.11\""},
+ {version = ">=1.23.2", markers = "python_version == \"3.11\""},
+ {version = ">=1.26.0", markers = "python_version >= \"3.12\""},
+]
+python-dateutil = ">=2.8.2"
+pytz = ">=2020.1"
+tzdata = ">=2022.7"
+
+[package.extras]
+all = ["PyQt5 (>=5.15.9)", "SQLAlchemy (>=2.0.0)", "adbc-driver-postgresql (>=0.8.0)", "adbc-driver-sqlite (>=0.8.0)", "beautifulsoup4 (>=4.11.2)", "bottleneck (>=1.3.6)", "dataframe-api-compat (>=0.1.7)", "fastparquet (>=2022.12.0)", "fsspec (>=2022.11.0)", "gcsfs (>=2022.11.0)", "html5lib (>=1.1)", "hypothesis (>=6.46.1)", "jinja2 (>=3.1.2)", "lxml (>=4.9.2)", "matplotlib (>=3.6.3)", "numba (>=0.56.4)", "numexpr (>=2.8.4)", "odfpy (>=1.4.1)", "openpyxl (>=3.1.0)", "pandas-gbq (>=0.19.0)", "psycopg2 (>=2.9.6)", "pyarrow (>=10.0.1)", "pymysql (>=1.0.2)", "pyreadstat (>=1.2.0)", "pytest (>=7.3.2)", "pytest-xdist (>=2.2.0)", "python-calamine (>=0.1.7)", "pyxlsb (>=1.0.10)", "qtpy (>=2.3.0)", "s3fs (>=2022.11.0)", "scipy (>=1.10.0)", "tables (>=3.8.0)", "tabulate (>=0.9.0)", "xarray (>=2022.12.0)", "xlrd (>=2.0.1)", "xlsxwriter (>=3.0.5)", "zstandard (>=0.19.0)"]
+aws = ["s3fs (>=2022.11.0)"]
+clipboard = ["PyQt5 (>=5.15.9)", "qtpy (>=2.3.0)"]
+compression = ["zstandard (>=0.19.0)"]
+computation = ["scipy (>=1.10.0)", "xarray (>=2022.12.0)"]
+consortium-standard = ["dataframe-api-compat (>=0.1.7)"]
+excel = ["odfpy (>=1.4.1)", "openpyxl (>=3.1.0)", "python-calamine (>=0.1.7)", "pyxlsb (>=1.0.10)", "xlrd (>=2.0.1)", "xlsxwriter (>=3.0.5)"]
+feather = ["pyarrow (>=10.0.1)"]
+fss = ["fsspec (>=2022.11.0)"]
+gcp = ["gcsfs (>=2022.11.0)", "pandas-gbq (>=0.19.0)"]
+hdf5 = ["tables (>=3.8.0)"]
+html = ["beautifulsoup4 (>=4.11.2)", "html5lib (>=1.1)", "lxml (>=4.9.2)"]
+mysql = ["SQLAlchemy (>=2.0.0)", "pymysql (>=1.0.2)"]
+output-formatting = ["jinja2 (>=3.1.2)", "tabulate (>=0.9.0)"]
+parquet = ["pyarrow (>=10.0.1)"]
+performance = ["bottleneck (>=1.3.6)", "numba (>=0.56.4)", "numexpr (>=2.8.4)"]
+plot = ["matplotlib (>=3.6.3)"]
+postgresql = ["SQLAlchemy (>=2.0.0)", "adbc-driver-postgresql (>=0.8.0)", "psycopg2 (>=2.9.6)"]
+pyarrow = ["pyarrow (>=10.0.1)"]
+spss = ["pyreadstat (>=1.2.0)"]
+sql-other = ["SQLAlchemy (>=2.0.0)", "adbc-driver-postgresql (>=0.8.0)", "adbc-driver-sqlite (>=0.8.0)"]
+test = ["hypothesis (>=6.46.1)", "pytest (>=7.3.2)", "pytest-xdist (>=2.2.0)"]
+xml = ["lxml (>=4.9.2)"]
+
+[[package]]
+name = "pillow"
+version = "10.3.0"
+description = "Python Imaging Library (Fork)"
+category = "main"
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "pillow-10.3.0-cp310-cp310-macosx_10_10_x86_64.whl", hash = "sha256:90b9e29824800e90c84e4022dd5cc16eb2d9605ee13f05d47641eb183cd73d45"},
+ {file = "pillow-10.3.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:a2c405445c79c3f5a124573a051062300936b0281fee57637e706453e452746c"},
+ {file = "pillow-10.3.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:78618cdbccaa74d3f88d0ad6cb8ac3007f1a6fa5c6f19af64b55ca170bfa1edf"},
+ {file = "pillow-10.3.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:261ddb7ca91fcf71757979534fb4c128448b5b4c55cb6152d280312062f69599"},
+ {file = "pillow-10.3.0-cp310-cp310-manylinux_2_28_aarch64.whl", hash = "sha256:ce49c67f4ea0609933d01c0731b34b8695a7a748d6c8d186f95e7d085d2fe475"},
+ {file = "pillow-10.3.0-cp310-cp310-manylinux_2_28_x86_64.whl", hash = "sha256:b14f16f94cbc61215115b9b1236f9c18403c15dd3c52cf629072afa9d54c1cbf"},
+ {file = "pillow-10.3.0-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:d33891be6df59d93df4d846640f0e46f1a807339f09e79a8040bc887bdcd7ed3"},
+ {file = "pillow-10.3.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:b50811d664d392f02f7761621303eba9d1b056fb1868c8cdf4231279645c25f5"},
+ {file = "pillow-10.3.0-cp310-cp310-win32.whl", hash = "sha256:ca2870d5d10d8726a27396d3ca4cf7976cec0f3cb706debe88e3a5bd4610f7d2"},
+ {file = "pillow-10.3.0-cp310-cp310-win_amd64.whl", hash = "sha256:f0d0591a0aeaefdaf9a5e545e7485f89910c977087e7de2b6c388aec32011e9f"},
+ {file = "pillow-10.3.0-cp310-cp310-win_arm64.whl", hash = "sha256:ccce24b7ad89adb5a1e34a6ba96ac2530046763912806ad4c247356a8f33a67b"},
+ {file = "pillow-10.3.0-cp311-cp311-macosx_10_10_x86_64.whl", hash = "sha256:5f77cf66e96ae734717d341c145c5949c63180842a545c47a0ce7ae52ca83795"},
+ {file = "pillow-10.3.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:e4b878386c4bf293578b48fc570b84ecfe477d3b77ba39a6e87150af77f40c57"},
+ {file = "pillow-10.3.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:fdcbb4068117dfd9ce0138d068ac512843c52295ed996ae6dd1faf537b6dbc27"},
+ {file = "pillow-10.3.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9797a6c8fe16f25749b371c02e2ade0efb51155e767a971c61734b1bf6293994"},
+ {file = "pillow-10.3.0-cp311-cp311-manylinux_2_28_aarch64.whl", hash = "sha256:9e91179a242bbc99be65e139e30690e081fe6cb91a8e77faf4c409653de39451"},
+ {file = "pillow-10.3.0-cp311-cp311-manylinux_2_28_x86_64.whl", hash = "sha256:1b87bd9d81d179bd8ab871603bd80d8645729939f90b71e62914e816a76fc6bd"},
+ {file = "pillow-10.3.0-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:81d09caa7b27ef4e61cb7d8fbf1714f5aec1c6b6c5270ee53504981e6e9121ad"},
+ {file = "pillow-10.3.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:048ad577748b9fa4a99a0548c64f2cb8d672d5bf2e643a739ac8faff1164238c"},
+ {file = "pillow-10.3.0-cp311-cp311-win32.whl", hash = "sha256:7161ec49ef0800947dc5570f86568a7bb36fa97dd09e9827dc02b718c5643f09"},
+ {file = "pillow-10.3.0-cp311-cp311-win_amd64.whl", hash = "sha256:8eb0908e954d093b02a543dc963984d6e99ad2b5e36503d8a0aaf040505f747d"},
+ {file = "pillow-10.3.0-cp311-cp311-win_arm64.whl", hash = "sha256:4e6f7d1c414191c1199f8996d3f2282b9ebea0945693fb67392c75a3a320941f"},
+ {file = "pillow-10.3.0-cp312-cp312-macosx_10_10_x86_64.whl", hash = "sha256:e46f38133e5a060d46bd630faa4d9fa0202377495df1f068a8299fd78c84de84"},
+ {file = "pillow-10.3.0-cp312-cp312-macosx_11_0_arm64.whl", hash = "sha256:50b8eae8f7334ec826d6eeffaeeb00e36b5e24aa0b9df322c247539714c6df19"},
+ {file = "pillow-10.3.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:9d3bea1c75f8c53ee4d505c3e67d8c158ad4df0d83170605b50b64025917f338"},
+ {file = "pillow-10.3.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:19aeb96d43902f0a783946a0a87dbdad5c84c936025b8419da0a0cd7724356b1"},
+ {file = "pillow-10.3.0-cp312-cp312-manylinux_2_28_aarch64.whl", hash = "sha256:74d28c17412d9caa1066f7a31df8403ec23d5268ba46cd0ad2c50fb82ae40462"},
+ {file = "pillow-10.3.0-cp312-cp312-manylinux_2_28_x86_64.whl", hash = "sha256:ff61bfd9253c3915e6d41c651d5f962da23eda633cf02262990094a18a55371a"},
+ {file = "pillow-10.3.0-cp312-cp312-musllinux_1_1_aarch64.whl", hash = "sha256:d886f5d353333b4771d21267c7ecc75b710f1a73d72d03ca06df49b09015a9ef"},
+ {file = "pillow-10.3.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:4b5ec25d8b17217d635f8935dbc1b9aa5907962fae29dff220f2659487891cd3"},
+ {file = "pillow-10.3.0-cp312-cp312-win32.whl", hash = "sha256:51243f1ed5161b9945011a7360e997729776f6e5d7005ba0c6879267d4c5139d"},
+ {file = "pillow-10.3.0-cp312-cp312-win_amd64.whl", hash = "sha256:412444afb8c4c7a6cc11a47dade32982439925537e483be7c0ae0cf96c4f6a0b"},
+ {file = "pillow-10.3.0-cp312-cp312-win_arm64.whl", hash = "sha256:798232c92e7665fe82ac085f9d8e8ca98826f8e27859d9a96b41d519ecd2e49a"},
+ {file = "pillow-10.3.0-cp38-cp38-macosx_10_10_x86_64.whl", hash = "sha256:4eaa22f0d22b1a7e93ff0a596d57fdede2e550aecffb5a1ef1106aaece48e96b"},
+ {file = "pillow-10.3.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:cd5e14fbf22a87321b24c88669aad3a51ec052eb145315b3da3b7e3cc105b9a2"},
+ {file = "pillow-10.3.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:1530e8f3a4b965eb6a7785cf17a426c779333eb62c9a7d1bbcf3ffd5bf77a4aa"},
+ {file = "pillow-10.3.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:5d512aafa1d32efa014fa041d38868fda85028e3f930a96f85d49c7d8ddc0383"},
+ {file = "pillow-10.3.0-cp38-cp38-manylinux_2_28_aarch64.whl", hash = "sha256:339894035d0ede518b16073bdc2feef4c991ee991a29774b33e515f1d308e08d"},
+ {file = "pillow-10.3.0-cp38-cp38-manylinux_2_28_x86_64.whl", hash = "sha256:aa7e402ce11f0885305bfb6afb3434b3cd8f53b563ac065452d9d5654c7b86fd"},
+ {file = "pillow-10.3.0-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:0ea2a783a2bdf2a561808fe4a7a12e9aa3799b701ba305de596bc48b8bdfce9d"},
+ {file = "pillow-10.3.0-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:c78e1b00a87ce43bb37642c0812315b411e856a905d58d597750eb79802aaaa3"},
+ {file = "pillow-10.3.0-cp38-cp38-win32.whl", hash = "sha256:72d622d262e463dfb7595202d229f5f3ab4b852289a1cd09650362db23b9eb0b"},
+ {file = "pillow-10.3.0-cp38-cp38-win_amd64.whl", hash = "sha256:2034f6759a722da3a3dbd91a81148cf884e91d1b747992ca288ab88c1de15999"},
+ {file = "pillow-10.3.0-cp39-cp39-macosx_10_10_x86_64.whl", hash = "sha256:2ed854e716a89b1afcedea551cd85f2eb2a807613752ab997b9974aaa0d56936"},
+ {file = "pillow-10.3.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:dc1a390a82755a8c26c9964d457d4c9cbec5405896cba94cf51f36ea0d855002"},
+ {file = "pillow-10.3.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:4203efca580f0dd6f882ca211f923168548f7ba334c189e9eab1178ab840bf60"},
+ {file = "pillow-10.3.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3102045a10945173d38336f6e71a8dc71bcaeed55c3123ad4af82c52807b9375"},
+ {file = "pillow-10.3.0-cp39-cp39-manylinux_2_28_aarch64.whl", hash = "sha256:6fb1b30043271ec92dc65f6d9f0b7a830c210b8a96423074b15c7bc999975f57"},
+ {file = "pillow-10.3.0-cp39-cp39-manylinux_2_28_x86_64.whl", hash = "sha256:1dfc94946bc60ea375cc39cff0b8da6c7e5f8fcdc1d946beb8da5c216156ddd8"},
+ {file = "pillow-10.3.0-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:b09b86b27a064c9624d0a6c54da01c1beaf5b6cadfa609cf63789b1d08a797b9"},
+ {file = "pillow-10.3.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:d3b2348a78bc939b4fed6552abfd2e7988e0f81443ef3911a4b8498ca084f6eb"},
+ {file = "pillow-10.3.0-cp39-cp39-win32.whl", hash = "sha256:45ebc7b45406febf07fef35d856f0293a92e7417ae7933207e90bf9090b70572"},
+ {file = "pillow-10.3.0-cp39-cp39-win_amd64.whl", hash = "sha256:0ba26351b137ca4e0db0342d5d00d2e355eb29372c05afd544ebf47c0956ffeb"},
+ {file = "pillow-10.3.0-cp39-cp39-win_arm64.whl", hash = "sha256:50fd3f6b26e3441ae07b7c979309638b72abc1a25da31a81a7fbd9495713ef4f"},
+ {file = "pillow-10.3.0-pp310-pypy310_pp73-macosx_10_10_x86_64.whl", hash = "sha256:6b02471b72526ab8a18c39cb7967b72d194ec53c1fd0a70b050565a0f366d355"},
+ {file = "pillow-10.3.0-pp310-pypy310_pp73-macosx_11_0_arm64.whl", hash = "sha256:8ab74c06ffdab957d7670c2a5a6e1a70181cd10b727cd788c4dd9005b6a8acd9"},
+ {file = "pillow-10.3.0-pp310-pypy310_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:048eeade4c33fdf7e08da40ef402e748df113fd0b4584e32c4af74fe78baaeb2"},
+ {file = "pillow-10.3.0-pp310-pypy310_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9e2ec1e921fd07c7cda7962bad283acc2f2a9ccc1b971ee4b216b75fad6f0463"},
+ {file = "pillow-10.3.0-pp310-pypy310_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:4c8e73e99da7db1b4cad7f8d682cf6abad7844da39834c288fbfa394a47bbced"},
+ {file = "pillow-10.3.0-pp310-pypy310_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:16563993329b79513f59142a6b02055e10514c1a8e86dca8b48a893e33cf91e3"},
+ {file = "pillow-10.3.0-pp310-pypy310_pp73-win_amd64.whl", hash = "sha256:dd78700f5788ae180b5ee8902c6aea5a5726bac7c364b202b4b3e3ba2d293170"},
+ {file = "pillow-10.3.0-pp39-pypy39_pp73-macosx_10_10_x86_64.whl", hash = "sha256:aff76a55a8aa8364d25400a210a65ff59d0168e0b4285ba6bf2bd83cf675ba32"},
+ {file = "pillow-10.3.0-pp39-pypy39_pp73-macosx_11_0_arm64.whl", hash = "sha256:b7bc2176354defba3edc2b9a777744462da2f8e921fbaf61e52acb95bafa9828"},
+ {file = "pillow-10.3.0-pp39-pypy39_pp73-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:793b4e24db2e8742ca6423d3fde8396db336698c55cd34b660663ee9e45ed37f"},
+ {file = "pillow-10.3.0-pp39-pypy39_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d93480005693d247f8346bc8ee28c72a2191bdf1f6b5db469c096c0c867ac015"},
+ {file = "pillow-10.3.0-pp39-pypy39_pp73-manylinux_2_28_aarch64.whl", hash = "sha256:c83341b89884e2b2e55886e8fbbf37c3fa5efd6c8907124aeb72f285ae5696e5"},
+ {file = "pillow-10.3.0-pp39-pypy39_pp73-manylinux_2_28_x86_64.whl", hash = "sha256:1a1d1915db1a4fdb2754b9de292642a39a7fb28f1736699527bb649484fb966a"},
+ {file = "pillow-10.3.0-pp39-pypy39_pp73-win_amd64.whl", hash = "sha256:a0eaa93d054751ee9964afa21c06247779b90440ca41d184aeb5d410f20ff591"},
+ {file = "pillow-10.3.0.tar.gz", hash = "sha256:9d2455fbf44c914840c793e89aa82d0e1763a14253a000743719ae5946814b2d"},
+]
+
+[package.extras]
+docs = ["furo", "olefile", "sphinx (>=2.4)", "sphinx-copybutton", "sphinx-inline-tabs", "sphinx-removed-in", "sphinxext-opengraph"]
+fpx = ["olefile"]
+mic = ["olefile"]
+tests = ["check-manifest", "coverage", "defusedxml", "markdown2", "olefile", "packaging", "pyroma", "pytest", "pytest-cov", "pytest-timeout"]
+typing = ["typing-extensions"]
+xmp = ["defusedxml"]
+
+[[package]]
+name = "pluggy"
+version = "1.4.0"
+description = "plugin and hook calling mechanisms for python"
+category = "dev"
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "pluggy-1.4.0-py3-none-any.whl", hash = "sha256:7db9f7b503d67d1c5b95f59773ebb58a8c1c288129a88665838012cfb07b8981"},
+ {file = "pluggy-1.4.0.tar.gz", hash = "sha256:8c85c2876142a764e5b7548e7d9a0e0ddb46f5185161049a79b7e974454223be"},
+]
+
+[package.extras]
+dev = ["pre-commit", "tox"]
+testing = ["pytest", "pytest-benchmark"]
+
+[[package]]
+name = "pyparsing"
+version = "3.1.2"
+description = "pyparsing module - Classes and methods to define and execute parsing grammars"
+category = "main"
+optional = false
+python-versions = ">=3.6.8"
+files = [
+ {file = "pyparsing-3.1.2-py3-none-any.whl", hash = "sha256:f9db75911801ed778fe61bb643079ff86601aca99fcae6345aa67292038fb742"},
+ {file = "pyparsing-3.1.2.tar.gz", hash = "sha256:a1bac0ce561155ecc3ed78ca94d3c9378656ad4c94c1270de543f621420f94ad"},
+]
+
+[package.extras]
+diagrams = ["jinja2", "railroad-diagrams"]
+
+[[package]]
+name = "pytest"
+version = "8.1.1"
+description = "pytest: simple powerful testing with Python"
+category = "dev"
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "pytest-8.1.1-py3-none-any.whl", hash = "sha256:2a8386cfc11fa9d2c50ee7b2a57e7d898ef90470a7a34c4b949ff59662bb78b7"},
+ {file = "pytest-8.1.1.tar.gz", hash = "sha256:ac978141a75948948817d360297b7aae0fcb9d6ff6bc9ec6d514b85d5a65c044"},
+]
+
+[package.dependencies]
+colorama = {version = "*", markers = "sys_platform == \"win32\""}
+exceptiongroup = {version = ">=1.0.0rc8", markers = "python_version < \"3.11\""}
+iniconfig = "*"
+packaging = "*"
+pluggy = ">=1.4,<2.0"
+tomli = {version = ">=1", markers = "python_version < \"3.11\""}
+
+[package.extras]
+testing = ["argcomplete", "attrs (>=19.2)", "hypothesis (>=3.56)", "mock", "pygments (>=2.7.2)", "requests", "setuptools", "xmlschema"]
+
+[[package]]
+name = "python-dateutil"
+version = "2.9.0.post0"
+description = "Extensions to the standard Python datetime module"
+category = "main"
+optional = false
+python-versions = "!=3.0.*,!=3.1.*,!=3.2.*,>=2.7"
+files = [
+ {file = "python-dateutil-2.9.0.post0.tar.gz", hash = "sha256:37dd54208da7e1cd875388217d5e00ebd4179249f90fb72437e91a35459a0ad3"},
+ {file = "python_dateutil-2.9.0.post0-py2.py3-none-any.whl", hash = "sha256:a8b2bc7bffae282281c8140a97d3aa9c14da0b136dfe83f850eea9a5f7470427"},
+]
+
+[package.dependencies]
+six = ">=1.5"
+
+[[package]]
+name = "pytz"
+version = "2024.1"
+description = "World timezone definitions, modern and historical"
+category = "main"
+optional = false
+python-versions = "*"
+files = [
+ {file = "pytz-2024.1-py2.py3-none-any.whl", hash = "sha256:328171f4e3623139da4983451950b28e95ac706e13f3f2630a879749e7a8b319"},
+ {file = "pytz-2024.1.tar.gz", hash = "sha256:2a29735ea9c18baf14b448846bde5a48030ed267578472d8955cd0e7443a9812"},
+]
+
+[[package]]
+name = "ruff"
+version = "0.3.7"
+description = "An extremely fast Python linter and code formatter, written in Rust."
+category = "dev"
+optional = false
+python-versions = ">=3.7"
+files = [
+ {file = "ruff-0.3.7-py3-none-macosx_10_12_x86_64.macosx_11_0_arm64.macosx_10_12_universal2.whl", hash = "sha256:0e8377cccb2f07abd25e84fc5b2cbe48eeb0fea9f1719cad7caedb061d70e5ce"},
+ {file = "ruff-0.3.7-py3-none-macosx_10_12_x86_64.whl", hash = "sha256:15a4d1cc1e64e556fa0d67bfd388fed416b7f3b26d5d1c3e7d192c897e39ba4b"},
+ {file = "ruff-0.3.7-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:d28bdf3d7dc71dd46929fafeec98ba89b7c3550c3f0978e36389b5631b793663"},
+ {file = "ruff-0.3.7-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:379b67d4f49774ba679593b232dcd90d9e10f04d96e3c8ce4a28037ae473f7bb"},
+ {file = "ruff-0.3.7-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:c060aea8ad5ef21cdfbbe05475ab5104ce7827b639a78dd55383a6e9895b7c51"},
+ {file = "ruff-0.3.7-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:ebf8f615dde968272d70502c083ebf963b6781aacd3079081e03b32adfe4d58a"},
+ {file = "ruff-0.3.7-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:d48098bd8f5c38897b03604f5428901b65e3c97d40b3952e38637b5404b739a2"},
+ {file = "ruff-0.3.7-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:da8a4fda219bf9024692b1bc68c9cff4b80507879ada8769dc7e985755d662ea"},
+ {file = "ruff-0.3.7-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:6c44e0149f1d8b48c4d5c33d88c677a4aa22fd09b1683d6a7ff55b816b5d074f"},
+ {file = "ruff-0.3.7-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:3050ec0af72b709a62ecc2aca941b9cd479a7bf2b36cc4562f0033d688e44fa1"},
+ {file = "ruff-0.3.7-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:a29cc38e4c1ab00da18a3f6777f8b50099d73326981bb7d182e54a9a21bb4ff7"},
+ {file = "ruff-0.3.7-py3-none-musllinux_1_2_i686.whl", hash = "sha256:5b15cc59c19edca917f51b1956637db47e200b0fc5e6e1878233d3a938384b0b"},
+ {file = "ruff-0.3.7-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:e491045781b1e38b72c91247cf4634f040f8d0cb3e6d3d64d38dcf43616650b4"},
+ {file = "ruff-0.3.7-py3-none-win32.whl", hash = "sha256:bc931de87593d64fad3a22e201e55ad76271f1d5bfc44e1a1887edd0903c7d9f"},
+ {file = "ruff-0.3.7-py3-none-win_amd64.whl", hash = "sha256:5ef0e501e1e39f35e03c2acb1d1238c595b8bb36cf7a170e7c1df1b73da00e74"},
+ {file = "ruff-0.3.7-py3-none-win_arm64.whl", hash = "sha256:789e144f6dc7019d1f92a812891c645274ed08af6037d11fc65fcbc183b7d59f"},
+ {file = "ruff-0.3.7.tar.gz", hash = "sha256:d5c1aebee5162c2226784800ae031f660c350e7a3402c4d1f8ea4e97e232e3ba"},
+]
+
+[[package]]
+name = "scikit-image"
+version = "0.22.0"
+description = "Image processing in Python"
+category = "main"
+optional = false
+python-versions = ">=3.9"
+files = [
+ {file = "scikit_image-0.22.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:74ec5c1d4693506842cc7c9487c89d8fc32aed064e9363def7af08b8f8cbb31d"},
+ {file = "scikit_image-0.22.0-cp310-cp310-macosx_12_0_arm64.whl", hash = "sha256:a05ae4fe03d802587ed8974e900b943275548cde6a6807b785039d63e9a7a5ff"},
+ {file = "scikit_image-0.22.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6a92dca3d95b1301442af055e196a54b5a5128c6768b79fc0a4098f1d662dee6"},
+ {file = "scikit_image-0.22.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3663d063d8bf2fb9bdfb0ca967b9ee3b6593139c860c7abc2d2351a8a8863938"},
+ {file = "scikit_image-0.22.0-cp310-cp310-win_amd64.whl", hash = "sha256:ebdbdc901bae14dab637f8d5c99f6d5cc7aaf4a3b6f4003194e003e9f688a6fc"},
+ {file = "scikit_image-0.22.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:95d6da2d8a44a36ae04437c76d32deb4e3c993ffc846b394b9949fd8ded73cb2"},
+ {file = "scikit_image-0.22.0-cp311-cp311-macosx_12_0_arm64.whl", hash = "sha256:2c6ef454a85f569659b813ac2a93948022b0298516b757c9c6c904132be327e2"},
+ {file = "scikit_image-0.22.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e87872f067444ee90a00dd49ca897208308645382e8a24bd3e76f301af2352cd"},
+ {file = "scikit_image-0.22.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c5c378db54e61b491b9edeefff87e49fcf7fdf729bb93c777d7a5f15d36f743e"},
+ {file = "scikit_image-0.22.0-cp311-cp311-win_amd64.whl", hash = "sha256:2bcb74adb0634258a67f66c2bb29978c9a3e222463e003b67ba12056c003971b"},
+ {file = "scikit_image-0.22.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:003ca2274ac0fac252280e7179ff986ff783407001459ddea443fe7916e38cff"},
+ {file = "scikit_image-0.22.0-cp312-cp312-macosx_12_0_arm64.whl", hash = "sha256:cf3c0c15b60ae3e557a0c7575fbd352f0c3ce0afca562febfe3ab80efbeec0e9"},
+ {file = "scikit_image-0.22.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f5b23908dd4d120e6aecb1ed0277563e8cbc8d6c0565bdc4c4c6475d53608452"},
+ {file = "scikit_image-0.22.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:be79d7493f320a964f8fcf603121595ba82f84720de999db0fcca002266a549a"},
+ {file = "scikit_image-0.22.0-cp312-cp312-win_amd64.whl", hash = "sha256:722b970aa5da725dca55252c373b18bbea7858c1cdb406e19f9b01a4a73b30b2"},
+ {file = "scikit_image-0.22.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:22318b35044cfeeb63ee60c56fc62450e5fe516228138f1d06c7a26378248a86"},
+ {file = "scikit_image-0.22.0-cp39-cp39-macosx_12_0_arm64.whl", hash = "sha256:9e801c44a814afdadeabf4dffdffc23733e393767958b82319706f5fa3e1eaa9"},
+ {file = "scikit_image-0.22.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c472a1fb3665ec5c00423684590631d95f9afcbc97f01407d348b821880b2cb3"},
+ {file = "scikit_image-0.22.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:3b7a6c89e8d6252332121b58f50e1625c35f7d6a85489c0b6b7ee4f5155d547a"},
+ {file = "scikit_image-0.22.0-cp39-cp39-win_amd64.whl", hash = "sha256:5071b8f6341bfb0737ab05c8ab4ac0261f9e25dbcc7b5d31e5ed230fd24a7929"},
+ {file = "scikit_image-0.22.0.tar.gz", hash = "sha256:018d734df1d2da2719087d15f679d19285fce97cd37695103deadfaef2873236"},
+]
+
+[package.dependencies]
+imageio = ">=2.27"
+lazy_loader = ">=0.3"
+networkx = ">=2.8"
+numpy = ">=1.22"
+packaging = ">=21"
+pillow = ">=9.0.1"
+scipy = ">=1.8"
+tifffile = ">=2022.8.12"
+
+[package.extras]
+build = ["Cython (>=0.29.32)", "build", "meson-python (>=0.14)", "ninja", "numpy (>=1.22)", "packaging (>=21)", "pythran", "setuptools (>=67)", "spin (==0.6)", "wheel"]
+data = ["pooch (>=1.6.0)"]
+developer = ["pre-commit", "tomli"]
+docs = ["PyWavelets (>=1.1.1)", "dask[array] (>=2022.9.2)", "ipykernel", "ipywidgets", "kaleido", "matplotlib (>=3.5)", "myst-parser", "numpydoc (>=1.6)", "pandas (>=1.5)", "plotly (>=5.10)", "pooch (>=1.6)", "pydata-sphinx-theme (>=0.14.1)", "pytest-runner", "scikit-learn (>=1.1)", "seaborn (>=0.11)", "sphinx (>=7.2)", "sphinx-copybutton", "sphinx-gallery (>=0.14)", "sphinx_design (>=0.5)", "tifffile (>=2022.8.12)"]
+optional = ["PyWavelets (>=1.1.1)", "SimpleITK", "astropy (>=5.0)", "cloudpickle (>=0.2.1)", "dask[array] (>=2021.1.0)", "matplotlib (>=3.5)", "pooch (>=1.6.0)", "pyamg", "scikit-learn (>=1.1)"]
+test = ["asv", "matplotlib (>=3.5)", "numpydoc (>=1.5)", "pooch (>=1.6.0)", "pytest (>=7.0)", "pytest-cov (>=2.11.0)", "pytest-faulthandler", "pytest-localserver"]
+
+[[package]]
+name = "scikit-learn"
+version = "1.4.2"
+description = "A set of python modules for machine learning and data mining"
+category = "main"
+optional = false
+python-versions = ">=3.9"
+files = [
+ {file = "scikit-learn-1.4.2.tar.gz", hash = "sha256:daa1c471d95bad080c6e44b4946c9390a4842adc3082572c20e4f8884e39e959"},
+ {file = "scikit_learn-1.4.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:8539a41b3d6d1af82eb629f9c57f37428ff1481c1e34dddb3b9d7af8ede67ac5"},
+ {file = "scikit_learn-1.4.2-cp310-cp310-macosx_12_0_arm64.whl", hash = "sha256:68b8404841f944a4a1459b07198fa2edd41a82f189b44f3e1d55c104dbc2e40c"},
+ {file = "scikit_learn-1.4.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:81bf5d8bbe87643103334032dd82f7419bc8c8d02a763643a6b9a5c7288c5054"},
+ {file = "scikit_learn-1.4.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:36f0ea5d0f693cb247a073d21a4123bdf4172e470e6d163c12b74cbb1536cf38"},
+ {file = "scikit_learn-1.4.2-cp310-cp310-win_amd64.whl", hash = "sha256:87440e2e188c87db80ea4023440923dccbd56fbc2d557b18ced00fef79da0727"},
+ {file = "scikit_learn-1.4.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:45dee87ac5309bb82e3ea633955030df9bbcb8d2cdb30383c6cd483691c546cc"},
+ {file = "scikit_learn-1.4.2-cp311-cp311-macosx_12_0_arm64.whl", hash = "sha256:1d0b25d9c651fd050555aadd57431b53d4cf664e749069da77f3d52c5ad14b3b"},
+ {file = "scikit_learn-1.4.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b0203c368058ab92efc6168a1507d388d41469c873e96ec220ca8e74079bf62e"},
+ {file = "scikit_learn-1.4.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:44c62f2b124848a28fd695db5bc4da019287abf390bfce602ddc8aa1ec186aae"},
+ {file = "scikit_learn-1.4.2-cp311-cp311-win_amd64.whl", hash = "sha256:5cd7b524115499b18b63f0c96f4224eb885564937a0b3477531b2b63ce331904"},
+ {file = "scikit_learn-1.4.2-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:90378e1747949f90c8f385898fff35d73193dfcaec3dd75d6b542f90c4e89755"},
+ {file = "scikit_learn-1.4.2-cp312-cp312-macosx_12_0_arm64.whl", hash = "sha256:ff4effe5a1d4e8fed260a83a163f7dbf4f6087b54528d8880bab1d1377bd78be"},
+ {file = "scikit_learn-1.4.2-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:671e2f0c3f2c15409dae4f282a3a619601fa824d2c820e5b608d9d775f91780c"},
+ {file = "scikit_learn-1.4.2-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d36d0bc983336bbc1be22f9b686b50c964f593c8a9a913a792442af9bf4f5e68"},
+ {file = "scikit_learn-1.4.2-cp312-cp312-win_amd64.whl", hash = "sha256:d762070980c17ba3e9a4a1e043ba0518ce4c55152032f1af0ca6f39b376b5928"},
+ {file = "scikit_learn-1.4.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:d9993d5e78a8148b1d0fdf5b15ed92452af5581734129998c26f481c46586d68"},
+ {file = "scikit_learn-1.4.2-cp39-cp39-macosx_12_0_arm64.whl", hash = "sha256:426d258fddac674fdf33f3cb2d54d26f49406e2599dbf9a32b4d1696091d4256"},
+ {file = "scikit_learn-1.4.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5460a1a5b043ae5ae4596b3126a4ec33ccba1b51e7ca2c5d36dac2169f62ab1d"},
+ {file = "scikit_learn-1.4.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:49d64ef6cb8c093d883e5a36c4766548d974898d378e395ba41a806d0e824db8"},
+ {file = "scikit_learn-1.4.2-cp39-cp39-win_amd64.whl", hash = "sha256:c97a50b05c194be9146d61fe87dbf8eac62b203d9e87a3ccc6ae9aed2dfaf361"},
+]
+
+[package.dependencies]
+joblib = ">=1.2.0"
+numpy = ">=1.19.5"
+scipy = ">=1.6.0"
+threadpoolctl = ">=2.0.0"
+
+[package.extras]
+benchmark = ["matplotlib (>=3.3.4)", "memory-profiler (>=0.57.0)", "pandas (>=1.1.5)"]
+docs = ["Pillow (>=7.1.2)", "matplotlib (>=3.3.4)", "memory-profiler (>=0.57.0)", "numpydoc (>=1.2.0)", "pandas (>=1.1.5)", "plotly (>=5.14.0)", "pooch (>=1.6.0)", "scikit-image (>=0.17.2)", "seaborn (>=0.9.0)", "sphinx (>=6.0.0)", "sphinx-copybutton (>=0.5.2)", "sphinx-gallery (>=0.15.0)", "sphinx-prompt (>=1.3.0)", "sphinxext-opengraph (>=0.4.2)"]
+examples = ["matplotlib (>=3.3.4)", "pandas (>=1.1.5)", "plotly (>=5.14.0)", "pooch (>=1.6.0)", "scikit-image (>=0.17.2)", "seaborn (>=0.9.0)"]
+tests = ["black (>=23.3.0)", "matplotlib (>=3.3.4)", "mypy (>=1.3)", "numpydoc (>=1.2.0)", "pandas (>=1.1.5)", "polars (>=0.19.12)", "pooch (>=1.6.0)", "pyamg (>=4.0.0)", "pyarrow (>=12.0.0)", "pytest (>=7.1.2)", "pytest-cov (>=2.9.0)", "ruff (>=0.0.272)", "scikit-image (>=0.17.2)"]
+
+[[package]]
+name = "scipy"
+version = "1.13.0"
+description = "Fundamental algorithms for scientific computing in Python"
+category = "main"
+optional = false
+python-versions = ">=3.9"
+files = [
+ {file = "scipy-1.13.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:ba419578ab343a4e0a77c0ef82f088238a93eef141b2b8017e46149776dfad4d"},
+ {file = "scipy-1.13.0-cp310-cp310-macosx_12_0_arm64.whl", hash = "sha256:22789b56a999265431c417d462e5b7f2b487e831ca7bef5edeb56efe4c93f86e"},
+ {file = "scipy-1.13.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:05f1432ba070e90d42d7fd836462c50bf98bd08bed0aa616c359eed8a04e3922"},
+ {file = "scipy-1.13.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b8434f6f3fa49f631fae84afee424e2483289dfc30a47755b4b4e6b07b2633a4"},
+ {file = "scipy-1.13.0-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:dcbb9ea49b0167de4167c40eeee6e167caeef11effb0670b554d10b1e693a8b9"},
+ {file = "scipy-1.13.0-cp310-cp310-win_amd64.whl", hash = "sha256:1d2f7bb14c178f8b13ebae93f67e42b0a6b0fc50eba1cd8021c9b6e08e8fb1cd"},
+ {file = "scipy-1.13.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:0fbcf8abaf5aa2dc8d6400566c1a727aed338b5fe880cde64907596a89d576fa"},
+ {file = "scipy-1.13.0-cp311-cp311-macosx_12_0_arm64.whl", hash = "sha256:5e4a756355522eb60fcd61f8372ac2549073c8788f6114449b37e9e8104f15a5"},
+ {file = "scipy-1.13.0-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b5acd8e1dbd8dbe38d0004b1497019b2dbbc3d70691e65d69615f8a7292865d7"},
+ {file = "scipy-1.13.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:9ff7dad5d24a8045d836671e082a490848e8639cabb3dbdacb29f943a678683d"},
+ {file = "scipy-1.13.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:4dca18c3ffee287ddd3bc8f1dabaf45f5305c5afc9f8ab9cbfab855e70b2df5c"},
+ {file = "scipy-1.13.0-cp311-cp311-win_amd64.whl", hash = "sha256:a2f471de4d01200718b2b8927f7d76b5d9bde18047ea0fa8bd15c5ba3f26a1d6"},
+ {file = "scipy-1.13.0-cp312-cp312-macosx_10_9_x86_64.whl", hash = "sha256:d0de696f589681c2802f9090fff730c218f7c51ff49bf252b6a97ec4a5d19e8b"},
+ {file = "scipy-1.13.0-cp312-cp312-macosx_12_0_arm64.whl", hash = "sha256:b2a3ff461ec4756b7e8e42e1c681077349a038f0686132d623fa404c0bee2551"},
+ {file = "scipy-1.13.0-cp312-cp312-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6bf9fe63e7a4bf01d3645b13ff2aa6dea023d38993f42aaac81a18b1bda7a82a"},
+ {file = "scipy-1.13.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1e7626dfd91cdea5714f343ce1176b6c4745155d234f1033584154f60ef1ff42"},
+ {file = "scipy-1.13.0-cp312-cp312-musllinux_1_1_x86_64.whl", hash = "sha256:109d391d720fcebf2fbe008621952b08e52907cf4c8c7efc7376822151820820"},
+ {file = "scipy-1.13.0-cp312-cp312-win_amd64.whl", hash = "sha256:8930ae3ea371d6b91c203b1032b9600d69c568e537b7988a3073dfe4d4774f21"},
+ {file = "scipy-1.13.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:5407708195cb38d70fd2d6bb04b1b9dd5c92297d86e9f9daae1576bd9e06f602"},
+ {file = "scipy-1.13.0-cp39-cp39-macosx_12_0_arm64.whl", hash = "sha256:ac38c4c92951ac0f729c4c48c9e13eb3675d9986cc0c83943784d7390d540c78"},
+ {file = "scipy-1.13.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:09c74543c4fbeb67af6ce457f6a6a28e5d3739a87f62412e4a16e46f164f0ae5"},
+ {file = "scipy-1.13.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:28e286bf9ac422d6beb559bc61312c348ca9b0f0dae0d7c5afde7f722d6ea13d"},
+ {file = "scipy-1.13.0-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:33fde20efc380bd23a78a4d26d59fc8704e9b5fd9b08841693eb46716ba13d86"},
+ {file = "scipy-1.13.0-cp39-cp39-win_amd64.whl", hash = "sha256:45c08bec71d3546d606989ba6e7daa6f0992918171e2a6f7fbedfa7361c2de1e"},
+ {file = "scipy-1.13.0.tar.gz", hash = "sha256:58569af537ea29d3f78e5abd18398459f195546bb3be23d16677fb26616cc11e"},
+]
+
+[package.dependencies]
+numpy = ">=1.22.4,<2.3"
+
+[package.extras]
+dev = ["cython-lint (>=0.12.2)", "doit (>=0.36.0)", "mypy", "pycodestyle", "pydevtool", "rich-click", "ruff", "types-psutil", "typing_extensions"]
+doc = ["jupyterlite-pyodide-kernel", "jupyterlite-sphinx (>=0.12.0)", "jupytext", "matplotlib (>=3.5)", "myst-nb", "numpydoc", "pooch", "pydata-sphinx-theme (>=0.15.2)", "sphinx (>=5.0.0)", "sphinx-design (>=0.4.0)"]
+test = ["array-api-strict", "asv", "gmpy2", "hypothesis (>=6.30)", "mpmath", "pooch", "pytest", "pytest-cov", "pytest-timeout", "pytest-xdist", "scikit-umfpack", "threadpoolctl"]
+
+[[package]]
+name = "setuptools"
+version = "69.5.1"
+description = "Easily download, build, install, upgrade, and uninstall Python packages"
+category = "main"
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "setuptools-69.5.1-py3-none-any.whl", hash = "sha256:c636ac361bc47580504644275c9ad802c50415c7522212252c033bd15f301f32"},
+ {file = "setuptools-69.5.1.tar.gz", hash = "sha256:6c1fccdac05a97e598fb0ae3bbed5904ccb317337a51139dcd51453611bbb987"},
+]
+
+[package.extras]
+docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "pygments-github-lexers (==0.0.5)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-favicon", "sphinx-inline-tabs", "sphinx-lint", "sphinx-notfound-page (>=1,<2)", "sphinx-reredirects", "sphinxcontrib-towncrier"]
+testing = ["build[virtualenv]", "filelock (>=3.4.0)", "importlib-metadata", "ini2toml[lite] (>=0.9)", "jaraco.develop (>=7.21)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "mypy (==1.9)", "packaging (>=23.2)", "pip (>=19.1)", "pytest (>=6,!=8.1.1)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-home (>=0.5)", "pytest-mypy", "pytest-perf", "pytest-ruff (>=0.2.1)", "pytest-timeout", "pytest-xdist (>=3)", "tomli", "tomli-w (>=1.0.0)", "virtualenv (>=13.0.0)", "wheel"]
+testing-integration = ["build[virtualenv] (>=1.0.3)", "filelock (>=3.4.0)", "jaraco.envs (>=2.2)", "jaraco.path (>=3.2.0)", "packaging (>=23.2)", "pytest", "pytest-enabler", "pytest-xdist", "tomli", "virtualenv (>=13.0.0)", "wheel"]
+
+[[package]]
+name = "six"
+version = "1.16.0"
+description = "Python 2 and 3 compatibility utilities"
+category = "main"
+optional = false
+python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*"
+files = [
+ {file = "six-1.16.0-py2.py3-none-any.whl", hash = "sha256:8abb2f1d86890a2dfb989f9a77cfcfd3e47c2a354b01111771326f8aa26e0254"},
+ {file = "six-1.16.0.tar.gz", hash = "sha256:1e61c37477a1626458e36f7b1d82aa5c9b094fa4802892072e49de9c60c4c926"},
+]
+
+[[package]]
+name = "sympy"
+version = "1.12"
+description = "Computer algebra system (CAS) in Python"
+category = "main"
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "sympy-1.12-py3-none-any.whl", hash = "sha256:c3588cd4295d0c0f603d0f2ae780587e64e2efeedb3521e46b9bb1d08d184fa5"},
+ {file = "sympy-1.12.tar.gz", hash = "sha256:ebf595c8dac3e0fdc4152c51878b498396ec7f30e7a914d6071e674d49420fb8"},
+]
+
+[package.dependencies]
+mpmath = ">=0.19"
+
+[[package]]
+name = "tabulate"
+version = "0.9.0"
+description = "Pretty-print tabular data"
+category = "main"
+optional = false
+python-versions = ">=3.7"
+files = [
+ {file = "tabulate-0.9.0-py3-none-any.whl", hash = "sha256:024ca478df22e9340661486f85298cff5f6dcdba14f3813e8830015b9ed1948f"},
+ {file = "tabulate-0.9.0.tar.gz", hash = "sha256:0095b12bf5966de529c0feb1fa08671671b3368eec77d7ef7ab114be2c068b3c"},
+]
+
+[package.extras]
+widechars = ["wcwidth"]
+
+[[package]]
+name = "threadpoolctl"
+version = "3.4.0"
+description = "threadpoolctl"
+category = "main"
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "threadpoolctl-3.4.0-py3-none-any.whl", hash = "sha256:8f4c689a65b23e5ed825c8436a92b818aac005e0f3715f6a1664d7c7ee29d262"},
+ {file = "threadpoolctl-3.4.0.tar.gz", hash = "sha256:f11b491a03661d6dd7ef692dd422ab34185d982466c49c8f98c8f716b5c93196"},
+]
+
+[[package]]
+name = "tifffile"
+version = "2024.2.12"
+description = "Read and write TIFF files"
+category = "main"
+optional = false
+python-versions = ">=3.9"
+files = [
+ {file = "tifffile-2024.2.12-py3-none-any.whl", hash = "sha256:870998f82fbc94ff7c3528884c1b0ae54863504ff51dbebea431ac3fa8fb7c21"},
+ {file = "tifffile-2024.2.12.tar.gz", hash = "sha256:4920a3ec8e8e003e673d3c6531863c99eedd570d1b8b7e141c072ed78ff8030d"},
+]
+
+[package.dependencies]
+numpy = "*"
+
+[package.extras]
+all = ["defusedxml", "fsspec", "imagecodecs (>=2023.8.12)", "lxml", "matplotlib", "zarr"]
+
+[[package]]
+name = "tomli"
+version = "2.0.1"
+description = "A lil' TOML parser"
+category = "dev"
+optional = false
+python-versions = ">=3.7"
+files = [
+ {file = "tomli-2.0.1-py3-none-any.whl", hash = "sha256:939de3e7a6161af0c887ef91b7d41a53e7c5a1ca976325f429cb46ea9bc30ecc"},
+ {file = "tomli-2.0.1.tar.gz", hash = "sha256:de526c12914f0c550d15924c62d72abc48d6fe7364aa87328337a31007fe8a4f"},
+]
+
+[[package]]
+name = "tzdata"
+version = "2024.1"
+description = "Provider of IANA time zone data"
+category = "main"
+optional = false
+python-versions = ">=2"
+files = [
+ {file = "tzdata-2024.1-py2.py3-none-any.whl", hash = "sha256:9068bc196136463f5245e51efda838afa15aaeca9903f49050dfa2679db4d252"},
+ {file = "tzdata-2024.1.tar.gz", hash = "sha256:2674120f8d891909751c38abcdfd386ac0a5a1127954fbc332af6b5ceae07efd"},
+]
+
+[[package]]
+name = "zipp"
+version = "3.18.1"
+description = "Backport of pathlib-compatible object wrapper for zip files"
+category = "main"
+optional = false
+python-versions = ">=3.8"
+files = [
+ {file = "zipp-3.18.1-py3-none-any.whl", hash = "sha256:206f5a15f2af3dbaee80769fb7dc6f249695e940acca08dfb2a4769fe61e538b"},
+ {file = "zipp-3.18.1.tar.gz", hash = "sha256:2884ed22e7d8961de1c9a05142eb69a247f120291bc0206a00a7642f09b5b715"},
+]
+
+[package.extras]
+docs = ["furo", "jaraco.packaging (>=9.3)", "jaraco.tidelift (>=1.4)", "rst.linker (>=1.9)", "sphinx (>=3.5)", "sphinx-lint"]
+testing = ["big-O", "jaraco.functools", "jaraco.itertools", "more-itertools", "pytest (>=6)", "pytest-checkdocs (>=2.4)", "pytest-cov", "pytest-enabler (>=2.2)", "pytest-ignore-flaky", "pytest-mypy", "pytest-ruff (>=0.2.1)"]
+
+[metadata]
+lock-version = "2.0"
+python-versions = "^3.9"
+content-hash = "075c39f62fe44c86f33067cdf1517b96f702bbc2a8a6384fd4953db81cb96000"
diff --git a/pyproject.toml b/pyproject.toml
new file mode 100644
index 0000000..dd8ad6a
--- /dev/null
+++ b/pyproject.toml
@@ -0,0 +1,81 @@
+[tool.poetry]
+name = "nilabels"
+version = "0.1.0"
+description = ""
+authors = ["SebastianoF