Update ci.yml #2
23 errors
Black found 23 errors
Annotations
Check failure on line 48 in /home/runner/work/linc-convert/linc-convert/scripts/utils.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/utils.py#L14-L48
Returns
-------
orientation : str
A three-letter permutation of {('R', 'L'), ('A', 'P'), ('S', 'I')}
"""
- orientation = {
- 'coronal': 'LI',
- 'axial': 'LP',
- 'sagittal': 'PI',
- }.get(orientation.lower(), orientation).upper()
+ orientation = (
+ {
+ "coronal": "LI",
+ "axial": "LP",
+ "sagittal": "PI",
+ }
+ .get(orientation.lower(), orientation)
+ .upper()
+ )
if len(orientation) == 2:
- if 'L' not in orientation and 'R' not in orientation:
- orientation += 'R'
- if 'P' not in orientation and 'A' not in orientation:
- orientation += 'A'
- if 'I' not in orientation and 'S' not in orientation:
- orientation += 'S'
+ if "L" not in orientation and "R" not in orientation:
+ orientation += "R"
+ if "P" not in orientation and "A" not in orientation:
+ orientation += "A"
+ if "I" not in orientation and "S" not in orientation:
+ orientation += "S"
return orientation
def orientation_to_affine(orientation, vxw=1, vxh=1, vxd=1):
orientation = orientation_ensure_3d(orientation)
affine = np.zeros([4, 4])
vx = np.asarray([vxw, vxh, vxd])
for i in range(3):
letter = orientation[i]
- sign = -1 if letter in 'LPI' else 1
- letter = {'L': 'R', 'P': 'A', 'I': 'S'}.get(letter, letter)
- index = list('RAS').index(letter)
+ sign = -1 if letter in "LPI" else 1
+ letter = {"L": "R", "P": "A", "I": "S"}.get(letter, letter)
+ index = list("RAS").index(letter)
affine[index, i] = sign * vx[i]
return affine
def center_affine(affine, shape):
Check failure on line 282 in /home/runner/work/linc-convert/linc-convert/scripts/utils.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/utils.py#L62-L282
def make_compressor(name, **prm):
if not isinstance(name, str):
return name
name = name.lower()
- if name == 'blosc':
+ if name == "blosc":
Compressor = numcodecs.Blosc
- elif name == 'zlib':
+ elif name == "zlib":
Compressor = numcodecs.Zlib
else:
- raise ValueError('Unknown compressor', name)
+ raise ValueError("Unknown compressor", name)
return Compressor(**prm)
ome_valid_units = {
- 'space': [
- 'angstrom',
- 'attometer',
- 'centimeter',
- 'decimeter',
- 'exameter',
- 'femtometer',
- 'foot',
- 'gigameter',
- 'hectometer',
- 'inch',
- 'kilometer',
- 'megameter',
- 'meter',
- 'micrometer',
- 'mile',
- 'millimeter',
- 'nanometer',
- 'parsec',
- 'petameter',
- 'picometer',
- 'terameter',
- 'yard',
- 'yoctometer',
- 'yottameter',
- 'zeptometer',
- 'zettameter',
+ "space": [
+ "angstrom",
+ "attometer",
+ "centimeter",
+ "decimeter",
+ "exameter",
+ "femtometer",
+ "foot",
+ "gigameter",
+ "hectometer",
+ "inch",
+ "kilometer",
+ "megameter",
+ "meter",
+ "micrometer",
+ "mile",
+ "millimeter",
+ "nanometer",
+ "parsec",
+ "petameter",
+ "picometer",
+ "terameter",
+ "yard",
+ "yoctometer",
+ "yottameter",
+ "zeptometer",
+ "zettameter",
],
- 'time': [
- 'attosecond',
- 'centisecond',
- 'day',
- 'decisecond',
- 'exasecond',
- 'femtosecond',
- 'gigasecond',
- 'hectosecond',
- 'hour',
- 'kilosecond',
- 'megasecond',
- 'microsecond',
- 'millisecond',
- 'minute',
- 'nanosecond',
- 'petasecond',
- 'picosecond',
- 'second',
- 'terasecond',
- 'yoctosecond',
- 'yottasecond',
- 'zeptosecond',
- 'zettasecond',
- ]
+ "time": [
+ "attosecond",
+ "centisecond",
+ "day",
+ "decisecond",
+ "exasecond",
+ "femtosecond",
+ "gigasecond",
+ "hectosecond",
+ "hour",
+ "kilosecond",
+ "megasecond",
+ "microsecond",
+ "millisecond",
+ "minute",
+ "nanosecond",
+ "petasecond",
+ "picosecond",
+ "second",
+ "terasecond",
+ "yoctosecond",
+ "yottasecond",
+ "zeptosecond",
+ "zettasecond",
+ ],
}
nifti_valid_units = [
- 'unknown',
- 'meter',
- 'mm',
- 'micron',
- 'sec',
- 'msec',
- 'usec',
- 'hz',
- 'ppm',
- 'rads',
+ "unknown",
+ "meter",
+ "mm",
+ "micron",
+ "sec",
+ "msec",
+ "usec",
+ "hz",
+ "ppm",
+ "rads",
]
si_prefix_short2long = {
- 'Q': 'quetta',
- 'R': 'ronna',
- 'Y': 'yotta',
- 'Z': 'zetta',
- 'E': 'exa',
- 'P': 'peta',
- 'T': 'tera',
- 'G': 'giga',
- 'M': 'mega',
- 'K': 'kilo',
- 'k': 'kilo',
- 'H': 'hecto',
- 'h': 'hecto',
- 'D': 'deca',
- 'da': 'deca',
- 'd': 'deci',
- 'c': 'centi',
- 'm': 'milli',
- 'u': 'micro',
- 'μ': 'micro',
- 'n': 'nano',
- 'p': 'pico',
- 'f': 'femto',
- 'a': 'atto',
- 'z': 'zepto',
- 'y': 'yocto',
- 'r': 'ronto',
- 'q': 'quecto',
-}
-
-si_prefix_long2short = {
- long: short
- for short, long in si_prefix_short2long.items()
-}
+ "Q": "quetta",
+ "R": "ronna",
+ "Y": "yotta",
+ "Z": "zetta",
+ "E": "exa",
+ "P": "peta",
+ "T": "tera",
+ "G": "giga",
+ "M": "mega",
+ "K": "kilo",
+ "k": "kilo",
+ "H": "hecto",
+ "h": "hecto",
+ "D": "deca",
+ "da": "deca",
+ "d": "deci",
+ "c": "centi",
+ "m": "milli",
+ "u": "micro",
+ "μ": "micro",
+ "n": "nano",
+ "p": "pico",
+ "f": "femto",
+ "a": "atto",
+ "z": "zepto",
+ "y": "yocto",
+ "r": "ronto",
+ "q": "quecto",
+}
+
+si_prefix_long2short = {long: short for short, long in si_prefix_short2long.items()}
si_prefix_exponent = {
- 'Q': 30,
- 'R': 27,
- 'Y': 24,
- 'Z': 21,
- 'E': 18,
- 'P': 15,
- 'T': 12,
- 'G': 9,
- 'M': 6,
- 'K': 3,
- 'k': 3,
- 'H': 2,
- 'h': 2,
- 'D': 1,
- 'da': 1,
- '': 0,
- 'd': -1,
- 'c': -2,
- 'm': -3,
- 'u': -6,
- 'μ': -6,
- 'n': -9,
- 'p': -12,
- 'f': -15,
- 'a': -18,
- 'z': -21,
- 'y': -24,
- 'r': -27,
- 'q': -30,
+ "Q": 30,
+ "R": 27,
+ "Y": 24,
+ "Z": 21,
+ "E": 18,
+ "P": 15,
+ "T": 12,
+ "G": 9,
+ "M": 6,
+ "K": 3,
+ "k": 3,
+ "H": 2,
+ "h": 2,
+ "D": 1,
+ "da": 1,
+ "": 0,
+ "d": -1,
+ "c": -2,
+ "m": -3,
+ "u": -6,
+ "μ": -6,
+ "n": -9,
+ "p": -12,
+ "f": -15,
+ "a": -18,
+ "z": -21,
+ "y": -24,
+ "r": -27,
+ "q": -30,
}
unit_space_short2long = {
- short + 'm': long + 'meter'
- for short, long in si_prefix_short2long.items()
-}
-unit_space_short2long.update({
- 'm': 'meter',
- 'mi': 'mile',
- 'yd': 'yard',
- 'ft': 'foot',
- 'in': 'inch',
- "'": 'foot',
- '"': 'inch',
- 'Å': 'angstrom',
- 'pc': 'parsec',
-})
-unit_space_long2short = {
- long: short
- for short, long in unit_space_short2long.items()
-}
-unit_space_long2short['micron'] = 'u'
+ short + "m": long + "meter" for short, long in si_prefix_short2long.items()
+}
+unit_space_short2long.update(
+ {
+ "m": "meter",
+ "mi": "mile",
+ "yd": "yard",
+ "ft": "foot",
+ "in": "inch",
+ "'": "foot",
+ '"': "inch",
+ "Å": "angstrom",
+ "pc": "parsec",
+ }
+)
+unit_space_long2short = {long: short for short, long in unit_space_short2long.items()}
+unit_space_long2short["micron"] = "u"
unit_time_short2long = {
- short + 's': long + 'second'
- for short, long in si_prefix_short2long.items()
-}
-unit_time_short2long.update({
- 'y': 'year',
- 'd': 'day',
- 'h': 'hour',
- 'm': 'minute',
- 's': 'second',
-})
-unit_time_long2short = {
- long: short
- for short, long in unit_time_short2long.items()
-}
+ short + "s": long + "second" for short, long in si_prefix_short2long.items()
+}
+unit_time_short2long.update(
+ {
+ "y": "year",
+ "d": "day",
+ "h": "hour",
+ "m": "minute",
+ "s": "second",
+ }
+)
+unit_time_long2short = {long: short for short, long in unit_time_short2long.items()}
unit_space_scale = {
- prefix + 'm': 10**exponent
- for prefix, exponent in si_prefix_exponent.items()
-}
-unit_space_scale.update({
- 'mi': 1609.344,
- 'yd': 0.9144,
- 'ft': 0.3048,
- "'": 0.3048,
- 'in': 25.4E-3,
- '"': 25.4E-3,
- 'Å': 1E-10,
- 'pc': 3.0857E16,
-})
+ prefix + "m": 10**exponent for prefix, exponent in si_prefix_exponent.items()
+}
+unit_space_scale.update(
+ {
+ "mi": 1609.344,
+ "yd": 0.9144,
+ "ft": 0.3048,
+ "'": 0.3048,
+ "in": 25.4e-3,
+ '"': 25.4e-3,
+ "Å": 1e-10,
+ "pc": 3.0857e16,
+ }
+)
unit_time_scale = {
- prefix + 's': 10**exponent
- for prefix, exponent in si_prefix_exponent.items()
-}
-unit_time_scale.update({
- 'y': 365.25*24*60*60,
- 'd': 24*60*60,
- 'h': 60*60,
- 'm': 60,
-})
+ prefix + "s": 10**exponent for prefix, exponent in si_prefix_exponent.items()
+}
+unit_time_scale.update(
+ {
+ "y": 365.25 * 24 * 60 * 60,
+ "d": 24 * 60 * 60,
+ "h": 60 * 60,
+ "m": 60,
+ }
+)
def convert_unit(value, src, dst):
src = unit_to_scale(src)
dst = unit_to_scale(dst)
Check failure on line 312 in /home/runner/work/linc-convert/linc-convert/scripts/utils.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/utils.py#L287-L312
unit = unit_space_short2long[unit]
elif unit in unit_time_short2long:
unit = unit_time_short2long[unit]
elif unit in si_prefix_short2long:
unit = si_prefix_short2long[unit]
- if unit not in (*ome_valid_units['space'], *ome_valid_units['time']):
- raise ValueError('Unknow unit')
+ if unit not in (*ome_valid_units["space"], *ome_valid_units["time"]):
+ raise ValueError("Unknow unit")
return unit
def to_nifti_unit(unit):
unit = to_ome_unit(unit)
return {
- 'meter': 'meter',
- 'millimeter': 'mm',
- 'micrometer': 'micron',
- 'second': 'sec',
- 'millisecond': 'msec',
- 'microsecond': 'usec',
- }.get(unit, 'unknown')
+ "meter": "meter",
+ "millimeter": "mm",
+ "micrometer": "micron",
+ "second": "sec",
+ "millisecond": "msec",
+ "microsecond": "usec",
+ }.get(unit, "unknown")
def unit_to_scale(unit):
if unit in unit_space_long2short:
unit = unit_space_long2short[unit]
Check failure on line 325 in /home/runner/work/linc-convert/linc-convert/scripts/utils.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/utils.py#L318-L325
elif unit in unit_time_scale:
unit = unit_time_scale[unit]
elif unit in si_prefix_exponent:
unit = 10 ** si_prefix_exponent[unit]
if isinstance(unit, str):
- raise ValueError('Unknown unit', unit)
+ raise ValueError("Unknown unit", unit)
return unit
Check failure on line 21 in /home/runner/work/linc-convert/linc-convert/scripts/jp2_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/jp2_to_zarr.py#L11-L21
glymur
zarr
nibabel
cyclopts
"""
+
import cyclopts
import glymur
import zarr
import ast
import numcodecs
Check failure on line 55 in /home/runner/work/linc-convert/linc-convert/scripts/jp2_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/jp2_to_zarr.py#L25-L55
import numpy as np
from glob import glob
import nibabel as nib
from typing import Optional
-HOME = '/space/aspasia/2/users/linc/000003'
+HOME = "/space/aspasia/2/users/linc/000003"
# Path to LincBrain dataset
-LINCSET = os.path.join(HOME, 'sourcedata')
-LINCOUT = os.path.join(HOME, 'rawdata')
+LINCSET = os.path.join(HOME, "sourcedata")
+LINCOUT = os.path.join(HOME, "rawdata")
app = cyclopts.App(help_format="markdown")
@app.default
def convert(
inp: str = None,
out: str = None,
subjects: list = [],
*,
chunk: int = 4096,
- compressor: str = 'blosc',
+ compressor: str = "blosc",
compressor_opt: str = "{}",
max_load: int = 16384,
nii: bool = False,
- orientation: str = 'coronal',
+ orientation: str = "coronal",
center: bool = True,
thickness: Optional[float] = None,
):
"""
This command converts JPEG2000 files generated by MBF-Neurolucida
Check failure on line 298 in /home/runner/work/linc-convert/linc-convert/scripts/jp2_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/jp2_to_zarr.py#L96-L298
Set RAS[0, 0, 0] at FOV center
thickness
Slice thickness
"""
for LINCSUB in subjects:
- print('working on subject', LINCSUB)
- HISTO_FOLDER = os.path.join(LINCSET, f'sub-{LINCSUB}/micr')
- OUT_FOLDER = os.path.join(LINCOUT, f'sub-{LINCSUB}/micr')
+ print("working on subject", LINCSUB)
+ HISTO_FOLDER = os.path.join(LINCSET, f"sub-{LINCSUB}/micr")
+ OUT_FOLDER = os.path.join(LINCOUT, f"sub-{LINCSUB}/micr")
os.makedirs(OUT_FOLDER, exist_ok=True)
- inp_dir = list(sorted(glob(os.path.join(HISTO_FOLDER, f'*nDF.jp2'))))
-
- start_num, end_num = 0, len(inp_dir)-1
- out = os.path.join(OUT_FOLDER, f'sub-{LINCSUB}_sample-slice{start_num:04d}slice{end_num:04d}_stain-LY_DF')
- out += '.nii.zarr' if nii else '.ome.zarr'
- nii = nii or out.endswith('.nii.zarr')
+ inp_dir = list(sorted(glob(os.path.join(HISTO_FOLDER, f"*nDF.jp2"))))
+
+ start_num, end_num = 0, len(inp_dir) - 1
+ out = os.path.join(
+ OUT_FOLDER,
+ f"sub-{LINCSUB}_sample-slice{start_num:04d}slice{end_num:04d}_stain-LY_DF",
+ )
+ out += ".nii.zarr" if nii else ".ome.zarr"
+ nii = nii or out.endswith(".nii.zarr")
print(out)
if isinstance(compressor_opt, str):
compressor_opt = ast.literal_eval(compressor_opt)
# Prepare Zarr group
omz = zarr.storage.DirectoryStore(out)
omz = zarr.group(store=omz, overwrite=True)
- nblevel, has_channel, dtype_jp2 = float('inf'), float('inf'), ''
+ nblevel, has_channel, dtype_jp2 = float("inf"), float("inf"), ""
# get new_size
new_height, new_width = 0, 0
for inp in inp_dir:
jp2 = glymur.Jp2k(inp)
nblevel = min(nblevel, jp2.codestream.segment[2].num_res)
has_channel = min(has_channel, jp2.ndim - 2)
dtype_jp2 = np.dtype(jp2.dtype).str
if jp2.shape[0] > new_height:
new_height = jp2.shape[0]
- if jp2.shape[1] > new_width:
+ if jp2.shape[1] > new_width:
new_width = jp2.shape[1]
- new_size = (new_height, new_width, 3) if has_channel else (new_height, new_width)
+ new_size = (
+ (new_height, new_width, 3) if has_channel else (new_height, new_width)
+ )
print(len(inp_dir), new_size, nblevel, has_channel)
-
# Prepare chunking options
opt = {
- 'chunks': list(new_size[2:]) + [1] + [chunk, chunk],
- 'dimension_separator': r'/',
- 'order': 'F',
- 'dtype': dtype_jp2,
- 'fill_value': None,
- 'compressor': make_compressor(compressor, **compressor_opt),
+ "chunks": list(new_size[2:]) + [1] + [chunk, chunk],
+ "dimension_separator": r"/",
+ "order": "F",
+ "dtype": dtype_jp2,
+ "fill_value": None,
+ "compressor": make_compressor(compressor, **compressor_opt),
}
print(opt)
-
# Write each level
for level in range(nblevel):
shape = [ceildiv(s, 2**level) for s in new_size[:2]]
- shape = [new_size[2]] + [len(inp_dir)] + [s for s in shape]
-
- omz.create_dataset(f'{level}', shape=shape, **opt)
- array = omz[f'{level}']
+ shape = [new_size[2]] + [len(inp_dir)] + [s for s in shape]
+
+ omz.create_dataset(f"{level}", shape=shape, **opt)
+ array = omz[f"{level}"]
# Write each slice
- for idx, inp in enumerate(inp_dir):
+ for idx, inp in enumerate(inp_dir):
j2k = glymur.Jp2k(inp)
vxw, vxh = get_pixelsize(j2k)
subdat = WrappedJ2K(j2k, level=level)
subdat_size = subdat.shape
- print('Convert level', level, 'with shape', shape, 'for slice', idx, 'with size', subdat_size)
+ print(
+ "Convert level",
+ level,
+ "with shape",
+ shape,
+ "for slice",
+ idx,
+ "with size",
+ subdat_size,
+ )
# offset while attaching
- x, y = (int((shape[-2] - subdat_size[-2])/2), int((shape[-1] - subdat_size[-1])/2))
-
+ x, y = (
+ int((shape[-2] - subdat_size[-2]) / 2),
+ int((shape[-1] - subdat_size[-1]) / 2),
+ )
+
if max_load is None or (shape[-2] < max_load and shape[-1] < max_load):
- array[..., idx, :, :] = np.zeros((3, shape[-2], shape[-1]), dtype = np.uint8)
- array[..., idx, x : x + subdat_size[1], y : y + subdat_size[2]] = subdat[...]
+ array[..., idx, :, :] = np.zeros(
+ (3, shape[-2], shape[-1]), dtype=np.uint8
+ )
+ array[..., idx, x : x + subdat_size[1], y : y + subdat_size[2]] = (
+ subdat[...]
+ )
else:
ni = ceildiv(shape[-2], max_load)
nj = ceildiv(shape[-1], max_load)
-
+
for i in range(ni):
for j in range(nj):
- print(f'\r{i+1}/{ni}, {j+1}/{nj}', end=' ')
- start_x, end_x = i*max_load, min((i+1)*max_load, shape[-2])
- start_y, end_y = j*max_load, min((j+1)*max_load, shape[-1])
- array[..., idx, start_x:end_x, start_y:end_y] = np.zeros((3, end_x-start_x, end_y-start_y), dtype = np.uint8)
+ print(f"\r{i+1}/{ni}, {j+1}/{nj}", end=" ")
+ start_x, end_x = i * max_load, min(
+ (i + 1) * max_load, shape[-2]
+ )
+ start_y, end_y = j * max_load, min(
+ (j + 1) * max_load, shape[-1]
+ )
+ array[..., idx, start_x:end_x, start_y:end_y] = np.zeros(
+ (3, end_x - start_x, end_y - start_y), dtype=np.uint8
+ )
if end_x <= x or end_y <= y:
- continue
+ continue
if start_x >= subdat_size[-2] or start_y >= subdat_size[-1]:
- continue
+ continue
array[
...,
- idx,
- x + start_x: x + min(end_x, subdat_size[-2]),
- y + start_y: y + min(end_y, subdat_size[-1]),
- ] = subdat[
+ idx,
+ x + start_x : x + min(end_x, subdat_size[-2]),
+ y + start_y : y + min(end_y, subdat_size[-1]),
+ ] = subdat[
...,
- start_x: min((i+1)*max_load, subdat_size[-2]),
- start_y: min((j+1)*max_load, subdat_size[-1]),
+ start_x : min((i + 1) * max_load, subdat_size[-2]),
+ start_y : min((j + 1) * max_load, subdat_size[-1]),
]
- print('')
-
+ print("")
+
# Write OME-Zarr multiscale metadata
- print('Write metadata')
- multiscales = [{
- 'version': '0.4',
- 'axes': [
- {"name": "z", "type": "space", "unit": "micrometer"},
- {"name": "y", "type": "distance", "unit": "micrometer"},
- {"name": "x", "type": "space", "unit": "micrometer"}
- ],
- 'datasets': [],
- 'type': 'jpeg2000',
- 'name': '',
- }]
+ print("Write metadata")
+ multiscales = [
+ {
+ "version": "0.4",
+ "axes": [
+ {"name": "z", "type": "space", "unit": "micrometer"},
+ {"name": "y", "type": "distance", "unit": "micrometer"},
+ {"name": "x", "type": "space", "unit": "micrometer"},
+ ],
+ "datasets": [],
+ "type": "jpeg2000",
+ "name": "",
+ }
+ ]
if has_channel:
- multiscales[0]['axes'].insert(0, {"name": "c", "type": "channel"})
+ multiscales[0]["axes"].insert(0, {"name": "c", "type": "channel"})
for n in range(nblevel):
- shape0 = omz['0'].shape[-2:]
+ shape0 = omz["0"].shape[-2:]
shape = omz[str(n)].shape[-2:]
- multiscales[0]['datasets'].append({})
- level = multiscales[0]['datasets'][-1]
+ multiscales[0]["datasets"].append({})
+ level = multiscales[0]["datasets"][-1]
level["path"] = str(n)
# I assume that wavelet transforms end up aligning voxel edges
# across levels, so the effective scaling is the shape ratio,
# and there is a half voxel shift wrt to the "center of first voxel"
# frame
level["coordinateTransformations"] = [
{
"type": "scale",
- "scale": [1.0] * has_channel + [
- 1.0,
- (shape0[0]/shape[0])*vxh,
- (shape0[1]/shape[1])*vxw,
- ]
+ "scale": [1.0] * has_channel
+ + [
+ 1.0,
+ (shape0[0] / shape[0]) * vxh,
+ (shape0[1] / shape[1]) * vxw,
+ ],
},
{
"type": "translation",
- "translation": [0.0] * has_channel + [
- 0.0,
- (shape0[0]/shape[0] - 1)*vxh*0.5,
- (shape0[1]/shape[1] - 1)*vxw*0.5,
- ]
- }
+ "translation": [0.0] * has_channel
+ + [
+ 0.0,
+ (shape0[0] / shape[0] - 1) * vxh * 0.5,
+ (shape0[1] / shape[1] - 1) * vxw * 0.5,
+ ],
+ },
]
multiscales[0]["coordinateTransformations"] = [
- {
- "scale": [1.0] * (3 + has_channel),
- "type": "scale"
- }
+ {"scale": [1.0] * (3 + has_channel), "type": "scale"}
]
omz.attrs["multiscales"] = multiscales
-
- # Write sidecar .json file
- json_name = os.path.join(OUT_FOLDER, f'sub-{LINCSUB}_sample-slice{start_num:04d}slice{end_num:04d}_stain-LY_DF.json')
+ # Write sidecar .json file
+ json_name = os.path.join(
+ OUT_FOLDER,
+ f"sub-{LINCSUB}_sample-slice{start_num:04d}slice{end_num:04d}_stain-LY_DF.json",
+ )
dic = {}
- dic['PixelSize'] = json.dumps([vxw, vxh])
- dic['PixelSizeUnits'] = 'um'
- dic['SliceThickness'] = 1.2
- dic['SliceThicknessUnits'] = 'mm'
- dic['SampleStaining'] = 'LY'
+ dic["PixelSize"] = json.dumps([vxw, vxh])
+ dic["PixelSizeUnits"] = "um"
+ dic["SliceThickness"] = 1.2
+ dic["SliceThicknessUnits"] = "mm"
+ dic["SampleStaining"] = "LY"
with open(json_name, "w") as outfile:
- json.dump(dic, outfile)
- outfile.write('\n')
-
+ json.dump(dic, outfile)
+ outfile.write("\n")
def orientation_ensure_3d(orientation):
- orientation = {
- 'coronal': 'LI',
- 'axial': 'LP',
- 'sagittal': 'PI',
- }.get(orientation.lower(), orientation).upper()
+ orientation = (
+ {
+ "coronal": "LI",
+ "axial": "LP",
+ "sagittal": "PI",
+ }
+ .get(orientation.lower(), orientation)
+ .upper()
+ )
if len(orientation) == 2:
- if 'L' not in orientation and 'R' not in orientation:
- orientation += 'R'
- if 'P' not in orientation and 'A' not in orientation:
- orientation += 'A'
- if 'I' not in orientation and 'S' not in orientation:
- orientation += 'S'
+ if "L" not in orientation and "R" not in orientation:
+ orientation += "R"
+ if "P" not in orientation and "A" not in orientation:
+ orientation += "A"
+ if "I" not in orientation and "S" not in orientation:
+ orientation += "S"
return orientation
def orientation_to_affine(orientation, vxw=1, vxh=1, vxd=1):
orientation = orientation_ensure_3d(orientation)
affine = np.zeros([4, 4])
vx = np.asarray([vxw, vxh, vxd])
for i in range(3):
letter = orientation[i]
- sign = -1 if letter in 'LPI' else 1
- letter = {'L': 'R', 'P': 'A', 'I': 'S'}.get(letter, letter)
- index = list('RAS').index(letter)
+ sign = -1 if letter in "LPI" else 1
+ letter = {"L": "R", "P": "A", "I": "S"}.get(letter, letter)
+ index = list("RAS").index(letter)
affine[index, i] = sign * vx[i]
return affine
def center_affine(affine, shape):
Check failure on line 363 in /home/runner/work/linc-convert/linc-convert/scripts/jp2_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/jp2_to_zarr.py#L352-L363
if not isinstance(index, tuple):
index = (index,)
if Ellipsis not in index:
index += (Ellipsis,)
if any(idx is None for idx in index):
- raise TypeError('newaxis not supported')
+ raise TypeError("newaxis not supported")
# substitute ellipses
new_index = []
has_seen_ellipsis = False
last_was_ellipsis = False
Check failure on line 381 in /home/runner/work/linc-convert/linc-convert/scripts/jp2_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/jp2_to_zarr.py#L364-L381
for idx in index:
if idx is Ellipsis:
if not has_seen_ellipsis:
new_index += [slice(None)] * nb_ellipsis
elif not last_was_ellipsis:
- raise ValueError('Multiple ellipses should be contiguous')
+ raise ValueError("Multiple ellipses should be contiguous")
has_seen_ellipsis = True
last_was_ellipsis = True
elif not isinstance(idx, slice):
- raise TypeError('Only slices are supported')
+ raise TypeError("Only slices are supported")
elif idx.step not in (None, 1):
- raise ValueError('Striding not supported')
+ raise ValueError("Striding not supported")
else:
last_was_ellipsis = False
new_index += [idx]
index = new_index
Check failure on line 435 in /home/runner/work/linc-convert/linc-convert/scripts/jp2_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/jp2_to_zarr.py#L404-L435
def make_compressor(name, **prm):
if not isinstance(name, str):
return name
name = name.lower()
- if name == 'blosc':
+ if name == "blosc":
Compressor = numcodecs.Blosc
- elif name == 'zlib':
+ elif name == "zlib":
Compressor = numcodecs.Zlib
else:
- raise ValueError('Unknown compressor', name)
+ raise ValueError("Unknown compressor", name)
return Compressor(**prm)
def get_pixelsize(j2k):
# Adobe XMP metadata
# https://en.wikipedia.org/wiki/Extensible_Metadata_Platform
- XMP_UUID = 'BE7ACFCB97A942E89C71999491E3AFAC'
- TAG_Images = '{http://ns.adobe.com/xap/1.0/}Images'
- Tag_Desc = '{http://www.w3.org/1999/02/22-rdf-syntax-ns#}Description'
- Tag_PixelWidth = '{http://ns.adobe.com/xap/1.0/}PixelWidth'
- Tag_PixelHeight = '{http://ns.adobe.com/xap/1.0/}PixelHeight'
+ XMP_UUID = "BE7ACFCB97A942E89C71999491E3AFAC"
+ TAG_Images = "{http://ns.adobe.com/xap/1.0/}Images"
+ Tag_Desc = "{http://www.w3.org/1999/02/22-rdf-syntax-ns#}Description"
+ Tag_PixelWidth = "{http://ns.adobe.com/xap/1.0/}PixelWidth"
+ Tag_PixelHeight = "{http://ns.adobe.com/xap/1.0/}PixelHeight"
vxw = vxh = 1.0
for box in j2k.box:
- if getattr(box, 'uuid', None) == uuid.UUID(XMP_UUID):
+ if getattr(box, "uuid", None) == uuid.UUID(XMP_UUID):
try:
images = list(box.data.iter(TAG_Images))[0]
desc = list(images.iter(Tag_Desc))[0]
vxw = float(desc.attrib[Tag_PixelWidth])
vxh = float(desc.attrib[Tag_PixelHeight])
Check failure on line 443 in /home/runner/work/linc-convert/linc-convert/scripts/jp2_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/jp2_to_zarr.py#L437-L443
return vxw, vxh
if __name__ == "__main__":
app()
-
Check failure on line 36 in /home/runner/work/linc-convert/linc-convert/scripts/lsm_tiff_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/lsm_tiff_to_zarr.py#L13-L36
tifffile
zarr
nibabel
cyclopts
"""
+
import cyclopts
import zarr
import os
import re
import ast
import numpy as np
import nibabel as nib
from tifffile import TiffFile
from glob import glob
from typing import Optional, List
+
# local modules
-from utils import \
- make_compressor, ceildiv, orientation_to_affine, center_affine
+from utils import make_compressor, ceildiv, orientation_to_affine, center_affine
app = cyclopts.App(help_format="markdown")
Check failure on line 52 in /home/runner/work/linc-convert/linc-convert/scripts/lsm_tiff_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/lsm_tiff_to_zarr.py#L37-L52
def convert(
inp: str,
out: str = None,
*,
chunk: int = 128,
- compressor: str = 'blosc',
+ compressor: str = "blosc",
compressor_opt: str = "{}",
max_load: int = 512,
nii: bool = False,
- orientation: str = 'coronal',
+ orientation: str = "coronal",
center: bool = True,
thickness: Optional[float] = None,
voxel_size: List[float] = (1, 1, 1),
):
"""
Check failure on line 116 in /home/runner/work/linc-convert/linc-convert/scripts/lsm_tiff_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/lsm_tiff_to_zarr.py#L99-L116
if max_load % 2:
max_load += 1
CHUNK_PATTERN = re.compile(
- r'^(?P<prefix>\w*)'
- r'_z(?P<z>[0-9]+)'
- r'_y(?P<y>[0-9]+)'
- r'(?P<suffix>\w*)$'
+ r"^(?P<prefix>\w*)" r"_z(?P<z>[0-9]+)" r"_y(?P<y>[0-9]+)" r"(?P<suffix>\w*)$"
)
- all_chunks_dirnames = list(sorted(glob(os.path.join(inp, '*_z*_y*'))))
+ all_chunks_dirnames = list(sorted(glob(os.path.join(inp, "*_z*_y*"))))
all_chunks_info = dict(
dirname=[],
prefix=[],
suffix=[],
z=[],
Check failure on line 200 in /home/runner/work/linc-convert/linc-convert/scripts/lsm_tiff_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/lsm_tiff_to_zarr.py#L126-L200
)
# parse all directory names
for dirname in all_chunks_dirnames:
parsed = CHUNK_PATTERN.fullmatch(os.path.basename(dirname))
- all_chunks_info['dirname'].append(dirname)
- all_chunks_info['prefix'].append(parsed.group('prefix'))
- all_chunks_info['suffix'].append(parsed.group('suffix'))
- all_chunks_info['z'].append(int(parsed.group('z')))
- all_chunks_info['y'].append(int(parsed.group('y')))
+ all_chunks_info["dirname"].append(dirname)
+ all_chunks_info["prefix"].append(parsed.group("prefix"))
+ all_chunks_info["suffix"].append(parsed.group("suffix"))
+ all_chunks_info["z"].append(int(parsed.group("z")))
+ all_chunks_info["y"].append(int(parsed.group("y")))
# default output name
if not out:
- out = all_chunks_info['prefix'][0] + all_chunks_info['suffix'][0]
- out += '.nii.zarr' if nii else '.ome.zarr'
- nii = nii or out.endswith('.nii.zarr')
+ out = all_chunks_info["prefix"][0] + all_chunks_info["suffix"][0]
+ out += ".nii.zarr" if nii else ".ome.zarr"
+ nii = nii or out.endswith(".nii.zarr")
# parse all individual file names
- nchunkz = max(all_chunks_info['z'])
- nchunky = max(all_chunks_info['y'])
+ nchunkz = max(all_chunks_info["z"])
+ nchunky = max(all_chunks_info["y"])
allshapes = [[(0, 0, 0) for _ in range(nchunky)] for _ in range(nchunkz)]
nchannels = 0
dtype = None
for zchunk in range(nchunkz):
for ychunk in range(nchunky):
- for i in range(len(all_chunks_info['dirname'])):
- if all_chunks_info['z'][i] == zchunk + 1 \
- and all_chunks_info['y'][i] == ychunk + 1:
+ for i in range(len(all_chunks_info["dirname"])):
+ if (
+ all_chunks_info["z"][i] == zchunk + 1
+ and all_chunks_info["y"][i] == ychunk + 1
+ ):
break
- dirname = all_chunks_info['dirname'][i]
- planes_filenames \
- = list(sorted(glob(os.path.join(dirname, '*.tiff'))))
+ dirname = all_chunks_info["dirname"][i]
+ planes_filenames = list(sorted(glob(os.path.join(dirname, "*.tiff"))))
PLANE_PATTERN = re.compile(
- os.path.basename(dirname) +
- r'_plane(?P<z>[0-9]+)'
- r'_c(?P<c>[0-9]+)'
- r'.tiff$'
+ os.path.basename(dirname) + r"_plane(?P<z>[0-9]+)"
+ r"_c(?P<c>[0-9]+)"
+ r".tiff$"
)
for fname in planes_filenames:
parsed = PLANE_PATTERN.fullmatch(os.path.basename(fname))
- all_chunks_info['planes'][i]['fname'].append(fname)
- all_chunks_info['planes'][i]['z'].append(int(parsed.group('z')))
- all_chunks_info['planes'][i]['c'].append(int(parsed.group('c')))
+ all_chunks_info["planes"][i]["fname"].append(fname)
+ all_chunks_info["planes"][i]["z"].append(int(parsed.group("z")))
+ all_chunks_info["planes"][i]["c"].append(int(parsed.group("c")))
f = TiffFile(fname)
dtype = f.pages[0].dtype
yx_shape = f.pages[0].shape
- all_chunks_info['planes'][i]['yx_shape'].append(yx_shape)
-
- nplanes = max(all_chunks_info['planes'][i]['z'])
- nchannels = max(nchannels, max(all_chunks_info['planes'][i]['c']))
-
- yx_shape = set(all_chunks_info['planes'][i]['yx_shape'])
+ all_chunks_info["planes"][i]["yx_shape"].append(yx_shape)
+
+ nplanes = max(all_chunks_info["planes"][i]["z"])
+ nchannels = max(nchannels, max(all_chunks_info["planes"][i]["c"]))
+
+ yx_shape = set(all_chunks_info["planes"][i]["yx_shape"])
if not len(yx_shape) == 1:
- raise ValueError('Incompatible chunk shapes')
+ raise ValueError("Incompatible chunk shapes")
yx_shape = list(yx_shape)[0]
allshapes[zchunk][ychunk] = (nplanes, *yx_shape)
# check that all chink shapes are compatible
for zchunk in range(nchunkz):
if len(set(shape[1] for shape in allshapes[zchunk])) != 1:
- raise ValueError('Incompatible Y shapes')
+ raise ValueError("Incompatible Y shapes")
for ychunk in range(nchunky):
if len(set(shape[ychunk][0] for shape in allshapes)) != 1:
- raise ValueError('Incompatible Z shapes')
+ raise ValueError("Incompatible Z shapes")
if len(set(shape[2] for subshapes in allshapes for shape in subshapes)) != 1:
- raise ValueError('Incompatible X shapes')
+ raise ValueError("Incompatible X shapes")
# compute full shape
fullshape = [0, 0, 0]
fullshape[0] = sum(shape[0][0] for shape in allshapes)
fullshape[1] = sum(shape[1] for shape in allshapes[0])
Check failure on line 259 in /home/runner/work/linc-convert/linc-convert/scripts/lsm_tiff_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/lsm_tiff_to_zarr.py#L203-L259
omz = zarr.storage.DirectoryStore(out)
omz = zarr.group(store=omz, overwrite=True)
# Prepare chunking options
opt = {
- 'chunks': [nchannels] + [chunk] * 3,
- 'dimension_separator': r'/',
- 'order': 'F',
- 'dtype': np.dtype(dtype).str,
- 'fill_value': None,
- 'compressor': make_compressor(compressor, **compressor_opt),
+ "chunks": [nchannels] + [chunk] * 3,
+ "dimension_separator": r"/",
+ "order": "F",
+ "dtype": np.dtype(dtype).str,
+ "fill_value": None,
+ "compressor": make_compressor(compressor, **compressor_opt),
}
# write first level
- omz.create_dataset('0', shape=[nchannels, *fullshape], **opt)
- array = omz['0']
- print('Write level 0 with shape', [nchannels, *fullshape])
- for i, dirname in enumerate(all_chunks_info['dirname']):
- chunkz = all_chunks_info['z'][i] - 1
- chunky = all_chunks_info['y'][i] - 1
- planes = all_chunks_info['planes'][i]
- for j, fname in enumerate(planes['fname']):
- subz = planes['z'][j] - 1
- subc = planes['c'][j] - 1
- yx_shape = planes['yx_shape'][j]
+ omz.create_dataset("0", shape=[nchannels, *fullshape], **opt)
+ array = omz["0"]
+ print("Write level 0 with shape", [nchannels, *fullshape])
+ for i, dirname in enumerate(all_chunks_info["dirname"]):
+ chunkz = all_chunks_info["z"][i] - 1
+ chunky = all_chunks_info["y"][i] - 1
+ planes = all_chunks_info["planes"][i]
+ for j, fname in enumerate(planes["fname"]):
+ subz = planes["z"][j] - 1
+ subc = planes["c"][j] - 1
+ yx_shape = planes["yx_shape"][j]
zstart = sum(shape[0][0] for shape in allshapes[:chunkz])
- ystart = sum(shape[1] for subshapes in allshapes for shape in subshapes[:chunky])
- print(f'Write plane ({subc}, {zstart + subz}, {ystart}:{ystart + yx_shape[0]})', end='\r')
+ ystart = sum(
+ shape[1] for subshapes in allshapes for shape in subshapes[:chunky]
+ )
+ print(
+ f"Write plane ({subc}, {zstart + subz}, {ystart}:{ystart + yx_shape[0]})",
+ end="\r",
+ )
slicer = (
subc,
zstart + subz,
slice(ystart, ystart + yx_shape[0]),
slice(None),
)
f = TiffFile(fname)
array[slicer] = f.asarray()
- print('')
+ print("")
# build pyramid using median windows
level = 0
while any(x > 1 for x in omz[str(level)].shape[-3:]):
prev_array = omz[str(level)]
prev_shape = prev_array.shape[-3:]
level += 1
- new_shape = list(map(lambda x: max(1, x//2), prev_shape))
+ new_shape = list(map(lambda x: max(1, x // 2), prev_shape))
if all(x < chunk for x in new_shape):
break
- print('Compute level', level, 'with shape', new_shape)
+ print("Compute level", level, "with shape", new_shape)
omz.create_dataset(str(level), shape=[nchannels, *new_shape], **opt)
new_array = omz[str(level)]
nz, ny, nx = prev_array.shape[-3:]
ncz = ceildiv(nz, max_load)
Check failure on line 401 in /home/runner/work/linc-convert/linc-convert/scripts/lsm_tiff_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/lsm_tiff_to_zarr.py#L261-L401
for cz in range(ncz):
for cy in range(ncy):
for cx in range(ncx):
- print(f'chunk ({cz}, {cy}, {cx}) / ({ncz}, {ncy}, {ncx})',
- end='\r')
+ print(f"chunk ({cz}, {cy}, {cx}) / ({ncz}, {ncy}, {ncx})", end="\r")
dat = prev_array[
...,
- cz*max_load:(cz+1)*max_load,
- cy*max_load:(cy+1)*max_load,
- cx*max_load:(cx+1)*max_load,
+ cz * max_load : (cz + 1) * max_load,
+ cy * max_load : (cy + 1) * max_load,
+ cx * max_load : (cx + 1) * max_load,
]
crop = [0 if x == 1 else x % 2 for x in dat.shape[-3:]]
slicer = [slice(-1) if x else slice(None) for x in crop]
dat = dat[(Ellipsis, *slicer)]
pz, py, px = dat.shape[-3:]
- dat = dat.reshape([
- nchannels,
- max(pz//2, 1), min(pz, 2),
- max(py//2, 1), min(py, 2),
- max(px//2, 1), min(px, 2),
- ])
+ dat = dat.reshape(
+ [
+ nchannels,
+ max(pz // 2, 1),
+ min(pz, 2),
+ max(py // 2, 1),
+ min(py, 2),
+ max(px // 2, 1),
+ min(px, 2),
+ ]
+ )
dat = dat.transpose([0, 1, 3, 5, 2, 4, 6])
- dat = dat.reshape([
- nchannels,
- max(pz//2, 1),
- max(py//2, 1),
- max(px//2, 1),
- -1,
- ])
+ dat = dat.reshape(
+ [
+ nchannels,
+ max(pz // 2, 1),
+ max(py // 2, 1),
+ max(px // 2, 1),
+ -1,
+ ]
+ )
dat = np.median(dat, -1)
new_array[
...,
- cz*max_load//2:(cz+1)*max_load//2,
- cy*max_load//2:(cy+1)*max_load//2,
- cx*max_load//2:(cx+1)*max_load//2,
+ cz * max_load // 2 : (cz + 1) * max_load // 2,
+ cy * max_load // 2 : (cy + 1) * max_load // 2,
+ cx * max_load // 2 : (cx + 1) * max_load // 2,
] = dat
- print('')
+ print("")
nblevel = level
# Write OME-Zarr multiscale metadata
- print('Write metadata')
- multiscales = [{
- 'version': '0.4',
- 'axes': [
- {"name": "z", "type": "space", "unit": "micrometer"},
- {"name": "y", "type": "space", "unit": "micrometer"},
- {"name": "x", "type": "space", "unit": "micrometer"}
- ],
- 'datasets': [],
- 'type': 'median window 2x2x2',
- 'name': '',
- }]
- multiscales[0]['axes'].insert(0, {"name": "c", "type": "channel"})
+ print("Write metadata")
+ multiscales = [
+ {
+ "version": "0.4",
+ "axes": [
+ {"name": "z", "type": "space", "unit": "micrometer"},
+ {"name": "y", "type": "space", "unit": "micrometer"},
+ {"name": "x", "type": "space", "unit": "micrometer"},
+ ],
+ "datasets": [],
+ "type": "median window 2x2x2",
+ "name": "",
+ }
+ ]
+ multiscales[0]["axes"].insert(0, {"name": "c", "type": "channel"})
voxel_size = list(map(float, reversed(voxel_size)))
factor = [1] * 3
for n in range(nblevel):
shape = omz[str(n)].shape[-3:]
- multiscales[0]['datasets'].append({})
- level = multiscales[0]['datasets'][-1]
+ multiscales[0]["datasets"].append({})
+ level = multiscales[0]["datasets"][-1]
level["path"] = str(n)
# We made sure that the downsampling level is exactly 2
# However, once a dimension has size 1, we stop downsampling.
if n > 0:
- shape_prev = omz[str(n-1)].shape[-3:]
+ shape_prev = omz[str(n - 1)].shape[-3:]
if shape_prev[0] != shape[0]:
factor[0] *= 2
if shape_prev[1] != shape[1]:
factor[1] *= 2
if shape_prev[2] != shape[2]:
factor[2] *= 2
level["coordinateTransformations"] = [
{
"type": "scale",
- "scale": [1.0] + [
+ "scale": [1.0]
+ + [
factor[0] * voxel_size[0],
factor[1] * voxel_size[1],
factor[2] * voxel_size[2],
- ]
+ ],
},
{
"type": "translation",
- "translation": [0.0] + [
+ "translation": [0.0]
+ + [
(factor[0] - 1) * voxel_size[0] * 0.5,
(factor[1] - 1) * voxel_size[1] * 0.5,
(factor[2] - 1) * voxel_size[2] * 0.5,
- ]
- }
+ ],
+ },
]
multiscales[0]["coordinateTransformations"] = [
- {
- "scale": [1.0] * 4,
- "type": "scale"
- }
+ {"scale": [1.0] * 4, "type": "scale"}
]
omz.attrs["multiscales"] = multiscales
if not nii:
- print('done.')
+ print("done.")
return
# Write NIfTI-Zarr header
# NOTE: we use nifti2 because dimensions typically do not fit in a short
# TODO: we do not write the json zattrs, but it should be added in
# once the nifti-zarr package is released
- shape = list(reversed(omz['0'].shape))
- shape = shape[:3] + [1] + shape[3:] # insert time dimension
+ shape = list(reversed(omz["0"].shape))
+ shape = shape[:3] + [1] + shape[3:] # insert time dimension
affine = orientation_to_affine(orientation, *voxel_size)
if center:
affine = center_affine(affine, shape[:3])
header = nib.Nifti2Header()
header.set_data_shape(shape)
- header.set_data_dtype(omz['0'].dtype)
+ header.set_data_dtype(omz["0"].dtype)
header.set_qform(affine)
header.set_sform(affine)
- header.set_xyzt_units(nib.nifti1.unit_codes.code['micron'])
- header.structarr['magic'] = b'nz2\0'
- header = np.frombuffer(header.structarr.tobytes(), dtype='u1')
+ header.set_xyzt_units(nib.nifti1.unit_codes.code["micron"])
+ header.structarr["magic"] = b"nz2\0"
+ header = np.frombuffer(header.structarr.tobytes(), dtype="u1")
opt = {
- 'chunks': [len(header)],
- 'dimension_separator': r'/',
- 'order': 'F',
- 'dtype': '|u1',
- 'fill_value': None,
- 'compressor': None,
+ "chunks": [len(header)],
+ "dimension_separator": r"/",
+ "order": "F",
+ "dtype": "|u1",
+ "fill_value": None,
+ "compressor": None,
}
- omz.create_dataset('nifti', data=header, shape=shape, **opt)
- print('done.')
+ omz.create_dataset("nifti", data=header, shape=shape, **opt)
+ print("done.")
if __name__ == "__main__":
app()
Check failure on line 20 in /home/runner/work/linc-convert/linc-convert/scripts/oct_mat_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/oct_mat_to_zarr.py#L10-L20
scipy
zarr
nibabel
cyclopts
"""
+
import ast
import json
import math
import os
import re
Check failure on line 41 in /home/runner/work/linc-convert/linc-convert/scripts/oct_mat_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/oct_mat_to_zarr.py#L29-L41
import numpy as np
import zarr
from scipy.io import loadmat
from utils import (
- ceildiv, make_compressor, convert_unit, to_ome_unit, to_nifti_unit,
- orientation_to_affine, center_affine
+ ceildiv,
+ make_compressor,
+ convert_unit,
+ to_ome_unit,
+ to_nifti_unit,
+ orientation_to_affine,
+ center_affine,
)
app = cyclopts.App(help_format="markdown")
Check failure on line 78 in /home/runner/work/linc-convert/linc-convert/scripts/oct_mat_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/oct_mat_to_zarr.py#L43-L78
@wraps(func)
def wrapper(inp, out=None, **kwargs):
if out is None:
out = os.path.splitext(inp[0])[0]
- out += '.nii.zarr' if kwargs.get('nii', False) else '.ome.zarr'
- kwargs['nii'] = kwargs.get('nii', False) or out.endswith('.nii.zarr')
- with mapmat(inp, kwargs.get('key', None)) as dat:
+ out += ".nii.zarr" if kwargs.get("nii", False) else ".ome.zarr"
+ kwargs["nii"] = kwargs.get("nii", False) or out.endswith(".nii.zarr")
+ with mapmat(inp, kwargs.get("key", None)) as dat:
return func(dat, out, **kwargs)
return wrapper
@app.default
@automap
def convert(
- inp: List[str],
- out: Optional[str] = None,
- *,
- key: Optional[str] = None,
- meta: str = None,
- chunk: int = 128,
- compressor: str = 'blosc',
- compressor_opt: str = "{}",
- max_load: int = 128,
- max_levels: int = 5,
- no_pool: Optional[int] = None,
- nii: bool = False,
- orientation: str = 'RAS',
- center: bool = True,
+ inp: List[str],
+ out: Optional[str] = None,
+ *,
+ key: Optional[str] = None,
+ meta: str = None,
+ chunk: int = 128,
+ compressor: str = "blosc",
+ compressor_opt: str = "{}",
+ max_load: int = 128,
+ max_levels: int = 5,
+ no_pool: Optional[int] = None,
+ nii: bool = False,
+ orientation: str = "RAS",
+ center: bool = True,
):
"""
This command converts OCT volumes stored in raw matlab files
into a pyramidal OME-ZARR (or NIfTI-Zarr) hierarchy.
Check failure on line 130 in /home/runner/work/linc-convert/linc-convert/scripts/oct_mat_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/oct_mat_to_zarr.py#L108-L130
if isinstance(compressor_opt, str):
compressor_opt = ast.literal_eval(compressor_opt)
# Write OME-Zarr multiscale metadata
if meta:
- print('Write JSON')
- with open(meta, 'r') as f:
+ print("Write JSON")
+ with open(meta, "r") as f:
meta_txt = f.read()
meta_json = make_json(meta_txt)
- path_json = '.'.join(out.split('.')[:-2]) + '.json'
- with open(path_json, 'w') as f:
+ path_json = ".".join(out.split(".")[:-2]) + ".json"
+ with open(path_json, "w") as f:
json.dump(meta_json, f, indent=4)
- vx = meta_json['PixelSize']
- unit = meta_json['PixelSizeUnits']
+ vx = meta_json["PixelSize"]
+ unit = meta_json["PixelSizeUnits"]
else:
vx = [1] * 3
- unit = 'um'
+ unit = "um"
# Prepare Zarr group
omz = zarr.storage.DirectoryStore(out)
omz = zarr.group(store=omz, overwrite=True)
Check failure on line 342 in /home/runner/work/linc-convert/linc-convert/scripts/oct_mat_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/oct_mat_to_zarr.py#L131-L342
raise Exception("Input is not a numpy array. This is likely unexpected")
if len(inp.shape) < 3:
raise Exception("Input array is not 3d")
# Prepare chunking options
opt = {
- 'dimension_separator': r'/',
- 'order': 'F',
- 'dtype': np.dtype(inp.dtype).str,
- 'fill_value': None,
- 'compressor': make_compressor(compressor, **compressor_opt),
+ "dimension_separator": r"/",
+ "order": "F",
+ "dtype": np.dtype(inp.dtype).str,
+ "fill_value": None,
+ "compressor": make_compressor(compressor, **compressor_opt),
}
inp_chunk = [min(x, max_load) for x in inp.shape]
nk = ceildiv(inp.shape[0], inp_chunk[0])
nj = ceildiv(inp.shape[1], inp_chunk[1])
ni = ceildiv(inp.shape[2], inp_chunk[2])
- nblevels = min([
- int(math.ceil(math.log2(x)))
- for i, x in enumerate(inp.shape)
- if i != no_pool
- ])
+ nblevels = min(
+ [int(math.ceil(math.log2(x))) for i, x in enumerate(inp.shape) if i != no_pool]
+ )
nblevels = min(nblevels, int(math.ceil(math.log2(max_load))))
nblevels = min(nblevels, max_levels)
# create all arrays in the group
shape_level = inp.shape
for level in range(nblevels):
- opt['chunks'] = [min(x, chunk) for x in shape_level]
+ opt["chunks"] = [min(x, chunk) for x in shape_level]
omz.create_dataset(str(level), shape=shape_level, **opt)
- shape_level = [
- x if i == no_pool else x // 2
- for i, x in enumerate(shape_level)
- ]
+ shape_level = [x if i == no_pool else x // 2 for i, x in enumerate(shape_level)]
# iterate across input chunks
for i, j, k in product(range(ni), range(nj), range(nk)):
level_chunk = inp_chunk
loaded_chunk = inp[
- k * level_chunk[0]:(k + 1) * level_chunk[0],
- j * level_chunk[1]:(j + 1) * level_chunk[1],
- i * level_chunk[2]:(i + 1) * level_chunk[2],
- ]
+ k * level_chunk[0] : (k + 1) * level_chunk[0],
+ j * level_chunk[1] : (j + 1) * level_chunk[1],
+ i * level_chunk[2] : (i + 1) * level_chunk[2],
+ ]
for level in range(nblevels):
out_level = omz[str(level)]
- print(f'[{i + 1:03d}, {j + 1:03d}, {k + 1:03d}]', '/',
- f'[{ni:03d}, {nj:03d}, {nk:03d}]',
- f'({1 + level}/{nblevels})', end='\r')
+ print(
+ f"[{i + 1:03d}, {j + 1:03d}, {k + 1:03d}]",
+ "/",
+ f"[{ni:03d}, {nj:03d}, {nk:03d}]",
+ f"({1 + level}/{nblevels})",
+ end="\r",
+ )
# save current chunk
out_level[
- k * level_chunk[0]:k * level_chunk[0] + loaded_chunk.shape[0],
- j * level_chunk[1]:j * level_chunk[1] + loaded_chunk.shape[1],
- i * level_chunk[2]:i * level_chunk[2] + loaded_chunk.shape[2],
+ k * level_chunk[0] : k * level_chunk[0] + loaded_chunk.shape[0],
+ j * level_chunk[1] : j * level_chunk[1] + loaded_chunk.shape[1],
+ i * level_chunk[2] : i * level_chunk[2] + loaded_chunk.shape[2],
] = loaded_chunk
# ensure divisible by 2
loaded_chunk = loaded_chunk[
slice(2 * (loaded_chunk.shape[0] // 2) if 0 != no_pool else None),
slice(2 * (loaded_chunk.shape[1] // 2) if 1 != no_pool else None),
slice(2 * (loaded_chunk.shape[2] // 2) if 2 != no_pool else None),
]
# mean pyramid (average each 2x2x2 patch)
if no_pool == 0:
loaded_chunk = (
- loaded_chunk[:, 0::2, 0::2] +
- loaded_chunk[:, 0::2, 1::2] +
- loaded_chunk[:, 1::2, 0::2] +
- loaded_chunk[:, 1::2, 1::2]
+ loaded_chunk[:, 0::2, 0::2]
+ + loaded_chunk[:, 0::2, 1::2]
+ + loaded_chunk[:, 1::2, 0::2]
+ + loaded_chunk[:, 1::2, 1::2]
) / 4
elif no_pool == 1:
loaded_chunk = (
- loaded_chunk[0::2, :, 0::2] +
- loaded_chunk[0::2, :, 1::2] +
- loaded_chunk[1::2, :, 0::2] +
- loaded_chunk[1::2, :, 1::2]
+ loaded_chunk[0::2, :, 0::2]
+ + loaded_chunk[0::2, :, 1::2]
+ + loaded_chunk[1::2, :, 0::2]
+ + loaded_chunk[1::2, :, 1::2]
) / 4
elif no_pool == 2:
loaded_chunk = (
- loaded_chunk[0::2, 0::2, :] +
- loaded_chunk[0::2, 1::2, :] +
- loaded_chunk[1::2, 0::2, :] +
- loaded_chunk[1::2, 1::2, :]
+ loaded_chunk[0::2, 0::2, :]
+ + loaded_chunk[0::2, 1::2, :]
+ + loaded_chunk[1::2, 0::2, :]
+ + loaded_chunk[1::2, 1::2, :]
) / 4
else:
loaded_chunk = (
- loaded_chunk[0::2, 0::2, 0::2] +
- loaded_chunk[0::2, 0::2, 1::2] +
- loaded_chunk[0::2, 1::2, 0::2] +
- loaded_chunk[0::2, 1::2, 1::2] +
- loaded_chunk[1::2, 0::2, 0::2] +
- loaded_chunk[1::2, 0::2, 1::2] +
- loaded_chunk[1::2, 1::2, 0::2] +
- loaded_chunk[1::2, 1::2, 1::2]
+ loaded_chunk[0::2, 0::2, 0::2]
+ + loaded_chunk[0::2, 0::2, 1::2]
+ + loaded_chunk[0::2, 1::2, 0::2]
+ + loaded_chunk[0::2, 1::2, 1::2]
+ + loaded_chunk[1::2, 0::2, 0::2]
+ + loaded_chunk[1::2, 0::2, 1::2]
+ + loaded_chunk[1::2, 1::2, 0::2]
+ + loaded_chunk[1::2, 1::2, 1::2]
) / 8
level_chunk = [
- x if i == no_pool else x // 2
- for i, x in enumerate(level_chunk)
+ x if i == no_pool else x // 2 for i, x in enumerate(level_chunk)
]
- print('')
+ print("")
# Write OME-Zarr multiscale metadata
- print('Write metadata')
+ print("Write metadata")
print(unit)
ome_unit = to_ome_unit(unit)
- multiscales = [{
- 'version': '0.4',
- 'axes': [
- {"name": "z", "type": "space", "unit": ome_unit},
- {"name": "y", "type": "space", "unit": ome_unit},
- {"name": "x", "type": "space", "unit": ome_unit}
- ],
- 'datasets': [],
- 'type': ('2x2x2' if no_pool is None else '2x2') + 'mean window',
- 'name': '',
- }]
+ multiscales = [
+ {
+ "version": "0.4",
+ "axes": [
+ {"name": "z", "type": "space", "unit": ome_unit},
+ {"name": "y", "type": "space", "unit": ome_unit},
+ {"name": "x", "type": "space", "unit": ome_unit},
+ ],
+ "datasets": [],
+ "type": ("2x2x2" if no_pool is None else "2x2") + "mean window",
+ "name": "",
+ }
+ ]
for n in range(nblevels):
- multiscales[0]['datasets'].append({})
- level = multiscales[0]['datasets'][-1]
+ multiscales[0]["datasets"].append({})
+ level = multiscales[0]["datasets"][-1]
level["path"] = str(n)
# With a moving window, the scaling factor is exactly 2, and
# the edges of the top-left voxel are aligned
level["coordinateTransformations"] = [
{
"type": "scale",
"scale": [
- (1 if no_pool == 0 else 2 ** n) * vx[0],
- (1 if no_pool == 1 else 2 ** n) * vx[1],
- (1 if no_pool == 2 else 2 ** n) * vx[2],
- ]
+ (1 if no_pool == 0 else 2**n) * vx[0],
+ (1 if no_pool == 1 else 2**n) * vx[1],
+ (1 if no_pool == 2 else 2**n) * vx[2],
+ ],
},
{
"type": "translation",
"translation": [
- (0 if no_pool == 0 else (2 ** n - 1)) * vx[0] * 0.5,
- (0 if no_pool == 1 else (2 ** n - 1)) * vx[1] * 0.5,
- (0 if no_pool == 2 else (2 ** n - 1)) * vx[2] * 0.5,
- ]
- }
+ (0 if no_pool == 0 else (2**n - 1)) * vx[0] * 0.5,
+ (0 if no_pool == 1 else (2**n - 1)) * vx[1] * 0.5,
+ (0 if no_pool == 2 else (2**n - 1)) * vx[2] * 0.5,
+ ],
+ },
]
multiscales[0]["coordinateTransformations"] = [
- {
- "scale": [1.0] * 3,
- "type": "scale"
- }
+ {"scale": [1.0] * 3, "type": "scale"}
]
omz.attrs["multiscales"] = multiscales
if not nii:
- print('done.')
+ print("done.")
return
# Write NIfTI-Zarr header
# NOTE: we use nifti2 because dimensions typically do not fit in a short
# TODO: we do not write the json zattrs, but it should be added in
# once the nifti-zarr package is released
- shape = list(reversed(omz['0'].shape))
+ shape = list(reversed(omz["0"].shape))
affine = orientation_to_affine(orientation, *vx[::-1])
if center:
affine = center_affine(affine, shape[:3])
header = nib.Nifti2Header()
header.set_data_shape(shape)
- header.set_data_dtype(omz['0'].dtype)
+ header.set_data_dtype(omz["0"].dtype)
header.set_qform(affine)
header.set_sform(affine)
header.set_xyzt_units(nib.nifti1.unit_codes.code[to_nifti_unit(unit)])
- header.structarr['magic'] = b'nz2\0'
- header = np.frombuffer(header.structarr.tobytes(), dtype='u1')
+ header.structarr["magic"] = b"nz2\0"
+ header = np.frombuffer(header.structarr.tobytes(), dtype="u1")
opt = {
- 'chunks': [len(header)],
- 'dimension_separator': r'/',
- 'order': 'F',
- 'dtype': '|u1',
- 'fill_value': None,
- 'compressor': None,
+ "chunks": [len(header)],
+ "dimension_separator": r"/",
+ "order": "F",
+ "dtype": "|u1",
+ "fill_value": None,
+ "compressor": None,
}
- omz.create_dataset('nifti', data=header, shape=shape, **opt)
- print('done.')
+ omz.create_dataset("nifti", data=header, shape=shape, **opt)
+ print("done.")
@contextmanager
def mapmat(fnames, key=None):
"""Load or memory-map an array stored in a .mat file"""
loaded_data = []
for fname in fnames:
try:
# "New" .mat file
- f = h5py.File(fname, 'r')
+ f = h5py.File(fname, "r")
except Exception:
# "Old" .mat file
f = loadmat(fname)
if key is None:
if len(f.keys()) > 1:
- warn(f'More than one key in .mat file {fname}, arbitrarily loading "{f.keys[0]}"')
+ warn(
+ f'More than one key in .mat file {fname}, arbitrarily loading "{f.keys[0]}"'
+ )
key = f.keys()[0]
if key not in f.keys():
raise Exception(f"Key {key} not found in file {fname}")
if len(fnames) == 1:
yield f.get(key)
- if hasattr(f, 'close'):
+ if hasattr(f, "close"):
f.close()
break
loaded_data.append(f.get(key))
yield np.stack(loaded_data, axis=-1)
Check failure on line 437 in /home/runner/work/linc-convert/linc-convert/scripts/oct_mat_to_zarr.py
github-actions / Black
/home/runner/work/linc-convert/linc-convert/scripts/oct_mat_to_zarr.py#L359-L437
Slice #:23
Modality: dBI
"""
def parse_value_unit(string, n=None):
- number = r'-?(\d+\.?\d*|\d*\.?\d+)(E-?\d+)?'
- value = 'x'.join([number] * (n or 1))
- match = re.fullmatch(r'(?P<value>' + value + r')(?P<unit>\w*)', string)
- value, unit = match.group('value'), match.group('unit')
- value = list(map(float, value.split('x')))
+ number = r"-?(\d+\.?\d*|\d*\.?\d+)(E-?\d+)?"
+ value = "x".join([number] * (n or 1))
+ match = re.fullmatch(r"(?P<value>" + value + r")(?P<unit>\w*)", string)
+ value, unit = match.group("value"), match.group("unit")
+ value = list(map(float, value.split("x")))
if n is None:
value = value[0]
return value, unit
meta = {
- 'BodyPart': 'BRAIN',
- 'Environment': 'exvivo',
- 'SampleStaining': 'none',
+ "BodyPart": "BRAIN",
+ "Environment": "exvivo",
+ "SampleStaining": "none",
}
- for line in oct_meta.split('\n'):
- if ':' not in line:
+ for line in oct_meta.split("\n"):
+ if ":" not in line:
continue
- key, value = line.split(':')
+ key, value = line.split(":")
key, value = key.strip(), value.strip()
- if key == 'Image medium':
+ if key == "Image medium":
parts = value.split()
- if 'TDE' in parts:
- parts[parts.index('TDE')] = "2,2' Thiodiethanol (TDE)"
- meta['SampleMedium'] = ' '.join(parts)
-
- elif key == 'Center Wavelength':
- value, unit = parse_value_unit(value)
- meta['Wavelength'] = value
- meta['WavelengthUnit'] = unit
-
- elif key == 'Axial resolution':
- value, unit = parse_value_unit(value)
- meta['ResolutionAxial'] = value
- meta['ResolutionAxialUnit'] = unit
-
- elif key == 'Lateral resolution':
- value, unit = parse_value_unit(value)
- meta['ResolutionLateral'] = value
- meta['ResolutionLateralUnit'] = unit
-
- elif key == 'Voxel size':
+ if "TDE" in parts:
+ parts[parts.index("TDE")] = "2,2' Thiodiethanol (TDE)"
+ meta["SampleMedium"] = " ".join(parts)
+
+ elif key == "Center Wavelength":
+ value, unit = parse_value_unit(value)
+ meta["Wavelength"] = value
+ meta["WavelengthUnit"] = unit
+
+ elif key == "Axial resolution":
+ value, unit = parse_value_unit(value)
+ meta["ResolutionAxial"] = value
+ meta["ResolutionAxialUnit"] = unit
+
+ elif key == "Lateral resolution":
+ value, unit = parse_value_unit(value)
+ meta["ResolutionLateral"] = value
+ meta["ResolutionLateralUnit"] = unit
+
+ elif key == "Voxel size":
value, unit = parse_value_unit(value, n=3)
- meta['PixelSize'] = value
- meta['PixelSizeUnits'] = unit
-
- elif key == 'Depth focus range':
- value, unit = parse_value_unit(value)
- meta['DepthFocusRange'] = value
- meta['DepthFocusRangeUnit'] = unit
-
- elif key == 'Number of focuses':
- value, unit = parse_value_unit(value)
- meta['FocusCount'] = int(value)
-
- elif key == 'Slice thickness':
- value, unit = parse_value_unit(value)
- unit = convert_unit(value, unit[:-1], 'u')
- meta['SliceThickness'] = value
-
- elif key == 'Number of slices':
- value, unit = parse_value_unit(value)
- meta['SliceCount'] = int(value)
-
- elif key == 'Modality':
- meta['OCTModality'] = value
+ meta["PixelSize"] = value
+ meta["PixelSizeUnits"] = unit
+
+ elif key == "Depth focus range":
+ value, unit = parse_value_unit(value)
+ meta["DepthFocusRange"] = value
+ meta["DepthFocusRangeUnit"] = unit
+
+ elif key == "Number of focuses":
+ value, unit = parse_value_unit(value)
+ meta["FocusCount"] = int(value)
+
+ elif key == "Slice thickness":
+ value, unit = parse_value_unit(value)
+ unit = convert_unit(value, unit[:-1], "u")
+ meta["SliceThickness"] = value
+
+ elif key == "Number of slices":
+ value, unit = parse_value_unit(value)
+ meta["SliceCount"] = int(value)
+
+ elif key == "Modality":
+ meta["OCTModality"] = value
else:
continue
return meta