From 21f9823bff67fe978a73e40f60f7091c6cbc237f Mon Sep 17 00:00:00 2001 From: FRIGO Matteo Date: Mon, 26 Jul 2021 15:04:18 +0200 Subject: [PATCH] fix empty line in subject list and management of 3d pet atlases This commit fixes two problems: - Subject lists passed to `react_masks` having an empty line raised an error. This commit makes the script skip those lines instead of raising the error. - 3-dimensional PET atlases (i.e., with only one PET image) raised errors when passed to the `react` and `react_masks` scripts. This commit fixes this problem and the programs now correctly process 3-dimensional PET atlases. Minor style issues were fixed. --- react/__init__.py | 2 +- react/utils.py | 4 ++-- script/react | 25 ++++++++++++++----------- script/react_masks | 19 ++++++++++++------- setup.py | 2 ++ 5 files changed, 31 insertions(+), 21 deletions(-) diff --git a/react/__init__.py b/react/__init__.py index e3e3b8d..0ffd241 100644 --- a/react/__init__.py +++ b/react/__init__.py @@ -1,2 +1,2 @@ -__version__ = '0.1.4' +__version__ = '0.1.5' from . import utils diff --git a/react/utils.py b/react/utils.py index b6942e7..3006f33 100644 --- a/react/utils.py +++ b/react/utils.py @@ -58,7 +58,7 @@ def check_can_write_file(fpath: str, force: bool = False, raise FileNotFoundError(f'Directory does not exist: {d}') -def normalize_3d_volume(v: np.ndarray): +def normalize_3d_volume(v: np.ndarray) -> np.ndarray: """ Normalize the positive values of a 3-dimensional numpy array. @@ -94,7 +94,7 @@ def normalize_3d_volume(v: np.ndarray): return data -def volume4d_to_matrix(v: np.ndarray): +def volume4d_to_matrix(v: np.ndarray) -> np.ndarray: """ This function transforms a 4-dimensional volume in a matrix whose columns are the vectorization of the 4th dimension of the input array. diff --git a/script/react b/script/react index 42dff95..97e02b4 100755 --- a/script/react +++ b/script/react @@ -16,8 +16,8 @@ OUT_REACT_STAGE2 = '_react_stage2.nii.gz' OUT_REACT_STAGE2_IC = '_react_stage2_IC' DESCRIPTION = f""" -v{react.__version__} Receptor-Enriched Analysis of Functional Connectivity by Targets. -All files must be in the same space. +v{react.__version__} Receptor-Enriched Analysis of Functional Connectivity by +Targets. All files must be in the same space. """ PROG = 'react' EPILOG = 'REFERENCE: https://doi.org/10.1016/j.neuroimage.2019.04.007 - ' \ @@ -30,7 +30,7 @@ EPILOG = 'REFERENCE: https://doi.org/10.1016/j.neuroimage.2019.04.007 - ' \ def get_parsed_args(): parser = argparse.ArgumentParser(prog=PROG, epilog=EPILOG, - description=DESCRIPTION) + description=DESCRIPTION) parser.add_argument( 'in_fmri', @@ -58,7 +58,7 @@ def get_parsed_args(): parser.add_argument( 'pet_atlas', type=str, - help='4D file containing the PET atlases ' + help='3D or 4D file containing the PET atlases ' 'to be used in the REACT analysis. ' 'It is recommended to rescale each PET atlas ' 'between 0 and 1 before running REACT. ' @@ -128,11 +128,11 @@ def main(): n_times = volume_fmri.shape[-1] volume_pet = nib.load(fpath_pet) - if volume_pet.shape[:-1] != volume_fmri.shape[:-1]: - raise ValueError(f'PET volume has shape {volume_pet.shape[:-1]} while ' - f'rs-fMRI volume has shape {volume_fmri.shape[:-1]}. ' + if volume_pet.shape[:3] != volume_fmri.shape[:3]: + raise ValueError(f'PET volume has shape {volume_pet.shape[:3]} while ' + f'rs-fMRI volume has shape {volume_fmri.shape[:3]}. ' f'They must be equal.') - n_pet_maps = volume_pet.shape[-1] + n_pet_maps = 1 if volume_pet.ndim == 3 else volume_pet.shape[3] fpath_ic = [] for i in range(n_pet_maps): @@ -164,7 +164,10 @@ def main(): logging.info('Initiating stage 1') rsfmri = volume4d_to_matrix(volume_fmri.get_fdata()) - pet = volume4d_to_matrix(volume_pet.get_fdata()) + pet = volume_pet.get_fdata() + if pet.ndim == 3: + pet = pet[..., np.newaxis] + pet = volume4d_to_matrix(pet) scaler_y = StandardScaler(with_mean=True, with_std=False) y = scaler_y.fit_transform(rsfmri)[mask_stage1, :] @@ -200,12 +203,12 @@ def main(): beta2 = np.reshape(beta2, volume_pet.shape) nib.save(nib.Nifti1Image(beta2, affine=volume_pet.affine), - fpath_result_stage2) + fpath_result_stage2) logging.info(f'Saved result stage 2 in {fpath_result_stage1}') for i, f in enumerate(fpath_ic): nib.save(nib.Nifti1Image(np.squeeze(beta2[..., i]), - affine=volume_pet.affine), f) + affine=volume_pet.affine), f) logging.info(f'Saved result IC{i} of stage 2 in {f}') diff --git a/script/react_masks b/script/react_masks index 147d89d..df5c399 100644 --- a/script/react_masks +++ b/script/react_masks @@ -28,7 +28,7 @@ EPILOG = 'REFERENCE: https://doi.org/10.1016/j.neuroimage.2019.04.007 - ' \ def get_parsed_args(): parser = argparse.ArgumentParser(prog=PROG, epilog=EPILOG, - description=DESCRIPTION) + description=DESCRIPTION) parser.add_argument( 'subject_list', @@ -41,7 +41,7 @@ def get_parsed_args(): parser.add_argument( 'pet_atlas', type=str, - help='4D file containing the PET atlases ' + help='3D or 4D file containing the PET atlases ' 'to be used in the REACT analysis. ' 'E.g., `/home/study/data/PETatlas.nii.gz` .' @@ -80,7 +80,8 @@ def main(): check_can_write_file(path.join(args.out_masks, f), args.force) with open(args.subject_list, 'rt') as f: - sublist = [fp.rstrip('\n') for fp in f.readlines()] + # it skips empty lines + sublist = [l.rstrip('\n') for l in list(filter(str.rstrip, f))] volume = nib.load(sublist[0]) data = np.zeros(volume.shape[:3]) @@ -98,17 +99,21 @@ def main(): fout = path.join(args.out_masks, OUT_MASK_STAGE_2) nib.save(nib.Nifti1Image(mask_st2.astype(np.int8), affine=volume.affine), - fout) + fout) volume = nib.load(args.pet_atlas) - data = np.sum(volume.get_fdata() > 0, axis=-1) == volume.shape[-1] + affine = volume.affine.copy() + volume = volume.get_fdata() + if volume.ndim == 3: + volume = volume[..., np.newaxis] + data = np.sum(volume > 0, axis=3) == volume.shape[3] if not np.all(mask_st2.shape == data.shape): raise ValueError('Shape of input volume is incompatible.') mask_st1 = np.logical_and(mask_st2, data > 0) fout = path.join(args.out_masks, OUT_MASK_STAGE_1) - nib.save(nib.Nifti1Image(mask_st1.astype(np.int8), affine=volume.affine), - fout) + nib.save(nib.Nifti1Image(mask_st1.astype(np.int8), affine=affine), + fout) if __name__ == '__main__': diff --git a/setup.py b/setup.py index c522e43..7afdc4a 100644 --- a/setup.py +++ b/setup.py @@ -22,6 +22,8 @@ setup( author='Ottavia Dipasquale, Matteo Frigo', + author_email='ottavia.dipasquale@kcl.ac.uk', + license='MIT', classifiers=[ 'Intended Audience :: Healthcare Industry', 'Intended Audience :: Science/Research',