diff --git a/sdcflows/workflows/base.py b/sdcflows/workflows/base.py index cc533c6cb1..1ec4edb153 100644 --- a/sdcflows/workflows/base.py +++ b/sdcflows/workflows/base.py @@ -1,16 +1,12 @@ # emacs: -*- mode: python; py-indent-offset: 4; indent-tabs-mode: nil -*- # vi: set ft=python sts=4 ts=4 sw=4 et: """SDC workflows coordination.""" -from collections import defaultdict - from nipype.pipeline import engine as pe from nipype.interfaces import utility as niu from nipype import logging from niworkflows.engine.workflows import LiterateWorkflow as Workflow -# Fieldmap workflows -from .pepolar import init_pepolar_unwarp_wf LOGGER = logging.getLogger('nipype.workflow') FMAP_PRIORITY = { @@ -23,12 +19,12 @@ DEFAULT_MEMORY_MIN_GB = 0.01 -def init_sdc_wf(distorted_ref, omp_nthreads=1, debug=False, ignore=None): +def init_sdc_estimate_wf(fmaps, epi_meta, omp_nthreads=1, debug=False, ignore=None): """ Build a :abbr:`SDC (susceptibility distortion correction)` workflow. - This workflow implements the heuristics to choose a - :abbr:`SDC (susceptibility distortion correction)` strategy. + This workflow implements the heuristics to choose an estimation + methodology for :abbr:`SDC (susceptibility distortion correction)`. When no field map information is present within the BIDS inputs, the EXPERIMENTAL "fieldmap-less SyN" can be performed, using the ``--use-syn`` argument. When ``--force-syn`` is specified, @@ -41,8 +37,14 @@ def init_sdc_wf(distorted_ref, omp_nthreads=1, debug=False, ignore=None): Parameters ---------- - distorted_ref : pybids.BIDSFile - A BIDSFile object with suffix ``bold``, ``sbref`` or ``dwi``. + fmaps : list of pybids dicts + A list of dictionaries with the available fieldmaps + (and their metadata using the key ``'metadata'`` for the + case of :abbr:`PEPOLAR (Phase-Encoding POLARity)` fieldmaps). + epi_meta : dict + BIDS metadata dictionary corresponding to the + :abbr:`EPI (echo-planar imaging)` run (i.e., suffix ``bold``, + ``sbref``, or ``dwi``) for which the fieldmap is being estimated. omp_nthreads : int Maximum number of threads an individual process may use debug : bool @@ -50,37 +52,30 @@ def init_sdc_wf(distorted_ref, omp_nthreads=1, debug=False, ignore=None): Inputs ------ - distorted_ref + epi_file A reference image calculated at a previous stage - ref_brain + epi_brain Same as above, but brain-masked - ref_mask + epi_mask Brain mask for the run - t1_brain + t1w_brain T1w image, brain-masked, for the fieldmap-less SyN method std2anat_xfm - List of standard-to-T1w transforms generated during spatial + Standard-to-T1w transform generated during spatial normalization (only for the fieldmap-less SyN method). - template : str - Name of template from which prior knowledge will be mapped - into the subject's T1w reference - (only for the fieldmap-less SyN method) - templates : str - Name of templates that index the ``std2anat_xfm`` input list - (only for the fieldmap-less SyN method). Outputs ------- - distorted_ref - An unwarped BOLD reference - bold_mask + epi_file + An unwarped EPI scan reference + epi_mask The corresponding new mask after unwarping - bold_ref_brain - Brain-extracted, unwarped BOLD reference + epi_brain + Brain-extracted, unwarped EPI scan reference out_warp The deformation field to unwarp the susceptibility distortions - syn_bold_ref - If ``--force-syn``, an unwarped BOLD reference with this + syn_ref + If ``--force-syn``, an unwarped EPI scan reference with this method (for reporting purposes) """ @@ -90,19 +85,16 @@ def init_sdc_wf(distorted_ref, omp_nthreads=1, debug=False, ignore=None): if not isinstance(ignore, (list, tuple)): ignore = tuple(ignore) - fmaps = defaultdict(list, []) - for associated in distorted_ref.get_associations(kind='InformedBy'): - if associated.suffix in list(FMAP_PRIORITY.keys()): - fmaps[associated.suffix].append(associated) + # TODO: To be removed (filter out unsupported fieldmaps): + fmaps = [fmap for fmap in fmaps if fmap['suffix'] in FMAP_PRIORITY] - workflow = Workflow(name='sdc_wf' if distorted_ref else 'sdc_bypass_wf') + workflow = Workflow(name='sdc_estimate_wf' if fmaps else 'sdc_bypass_wf') inputnode = pe.Node(niu.IdentityInterface( - fields=['distorted_ref', 'ref_brain', 'ref_mask', - 't1_brain', 'std2anat_xfm', 'template', 'templates']), + fields=['epi_file', 'epi_brain', 'epi_mask', 't1w_brain', 'std2anat_xfm']), name='inputnode') outputnode = pe.Node(niu.IdentityInterface( - fields=['output_ref', 'ref_mask', 'ref_brain', + fields=['output_ref', 'epi_mask', 'epi_brain', 'out_warp', 'syn_ref', 'method']), name='outputnode') @@ -115,121 +107,125 @@ def init_sdc_wf(distorted_ref, omp_nthreads=1, debug=False, ignore=None): """ outputnode.inputs.method = 'None' workflow.connect([ - (inputnode, outputnode, [('distorted_ref', 'output_ref'), - ('ref_mask', 'ref_mask'), - ('ref_brain', 'ref_brain')]), + (inputnode, outputnode, [('epi_file', 'output_ref'), + ('epi_mask', 'epi_mask'), + ('epi_brain', 'epi_brain')]), ]) return workflow workflow.__postdesc__ = """\ -Based on the estimated susceptibility distortion, an -unwarped BOLD reference was calculated for a more accurate -co-registration with the anatomical reference. +Based on the estimated susceptibility distortion, an unwarped +EPI (echo-planar imaging) reference was calculated for a more +accurate co-registration with the anatomical reference. """ + # In case there are multiple fieldmaps prefer EPI + fmaps.sort(key=lambda fmap: FMAP_PRIORITY[fmap['suffix']]) + fmap = fmaps[0] + # PEPOLAR path if 'epi' in fmaps: + from .pepolar import init_pepolar_unwarp_wf, check_pes outputnode.inputs.method = 'PEB/PEPOLAR (phase-encoding based / PE-POLARity)' + + # Filter out EPI fieldmaps to be used + fmaps_epi = [(epi.path, epi.get_metadata()['PhaseEncodingDirection']) + for epi in fmaps['epi']] + + # Find matched PE directions + matched_pe = check_pes(fmaps_epi, epi_meta['PhaseEncodingDirection']) + # Get EPI polarities and their metadata sdc_unwarp_wf = init_pepolar_unwarp_wf( - bold_meta=distorted_ref.get_metadata(), - epi_fmaps=[(fmap, fmap.get_metadata()["PhaseEncodingDirection"]) - for fmap in fmaps['epi']], - omp_nthreads=omp_nthreads, - name='pepolar_unwarp_wf') + matched_pe=matched_pe, + omp_nthreads=omp_nthreads) + sdc_unwarp_wf.inputs.inputnode.epi_pe_dir = epi_meta['PhaseEncodingDirection'] + sdc_unwarp_wf.inputs.inputnode.fmaps_epi = fmaps_epi workflow.connect([ (inputnode, sdc_unwarp_wf, [ - ('distorted_ref', 'inputnode.in_reference'), - ('bold_mask', 'inputnode.in_mask'), - ('bold_ref_brain', 'inputnode.in_reference_brain')]), + ('epi_file', 'inputnode.in_reference'), + ('epi_brain', 'inputnode.in_reference_brain'), + ('epi_mask', 'inputnode.in_mask')]), ]) # FIELDMAP path - # elif 'fieldmap' in fmaps: - # # Import specific workflows here, so we don't break everything with one - # # unused workflow. - # suffices = {f.suffix for f in fmaps['fieldmap']} - # if 'fieldmap' in suffices: - # from .fmap import init_fmap_wf - # outputnode.inputs.method = 'FMB (fieldmap-based)' - # fmap_estimator_wf = init_fmap_wf( - # omp_nthreads=omp_nthreads, - # fmap_bspline=False) - # # set inputs - # fmap_estimator_wf.inputs.inputnode.fieldmap = fmap['fieldmap'] - # fmap_estimator_wf.inputs.inputnode.magnitude = fmap['magnitude'] - - # if fmap['suffix'] == 'phasediff': - # from .phdiff import init_phdiff_wf - # fmap_estimator_wf = init_phdiff_wf(omp_nthreads=omp_nthreads) - # # set inputs - # fmap_estimator_wf.inputs.inputnode.phasediff = fmap['phasediff'] - # fmap_estimator_wf.inputs.inputnode.magnitude = [ - # fmap_ for key, fmap_ in sorted(fmap.items()) - # if key.startswith("magnitude") - # ] - - # sdc_unwarp_wf = init_sdc_unwarp_wf( - # omp_nthreads=omp_nthreads, - # fmap_demean=fmap_demean, - # debug=debug, - # name='sdc_unwarp_wf') - # sdc_unwarp_wf.inputs.inputnode.metadata = bold_meta - - # workflow.connect([ - # (inputnode, sdc_unwarp_wf, [ - # ('distorted_ref', 'inputnode.in_reference'), - # ('bold_ref_brain', 'inputnode.in_reference_brain'), - # ('bold_mask', 'inputnode.in_mask')]), - # (fmap_estimator_wf, sdc_unwarp_wf, [ - # ('outputnode.fmap', 'inputnode.fmap'), - # ('outputnode.fmap_ref', 'inputnode.fmap_ref'), - # ('outputnode.fmap_mask', 'inputnode.fmap_mask')]), - # ]) - - # # FIELDMAP-less path - # if any(fm['suffix'] == 'syn' for fm in fmaps): - # # Select template - # sdc_select_std = pe.Node(KeySelect( - # fields=['std2anat_xfm']), - # name='sdc_select_std', run_without_submitting=True) - - # syn_sdc_wf = init_syn_sdc_wf( - # bold_pe=bold_meta.get('PhaseEncodingDirection', None), - # omp_nthreads=omp_nthreads) - - # workflow.connect([ - # (inputnode, sdc_select_std, [ - # ('template', 'key'), - # ('templates', 'keys'), - # ('std2anat_xfm', 'std2anat_xfm')]), - # (sdc_select_std, syn_sdc_wf, [ - # ('std2anat_xfm', 'inputnode.std2anat_xfm')]), - # (inputnode, syn_sdc_wf, [ - # ('t1_brain', 'inputnode.t1_brain'), - # ('distorted_ref', 'inputnode.distorted_ref'), - # ('bold_ref_brain', 'inputnode.bold_ref_brain'), - # ('template', 'inputnode.template')]), - # ]) - - # # XXX Eliminate branch when forcing isn't an option - # if fmap['suffix'] == 'syn': # No fieldmaps, but --use-syn - # outputnode.inputs.method = 'FLB ("fieldmap-less", SyN-based)' - # sdc_unwarp_wf = syn_sdc_wf - # else: # --force-syn was called when other fieldmap was present - # sdc_unwarp_wf.__desc__ = None - # workflow.connect([ - # (syn_sdc_wf, outputnode, [ - # ('outputnode.out_reference', 'syn_bold_ref')]), - # ]) + elif 'fieldmap' in fmaps: + from .unwarp import init_sdc_unwarp_wf + # Import specific workflows here, so we don't break everything with one + # unused workflow. + suffices = {f.suffix for f in fmaps['fieldmap']} + if 'fieldmap' in suffices: + from .fmap import init_fmap_wf + outputnode.inputs.method = 'FMB (fieldmap-based)' + fmap_wf = init_fmap_wf( + omp_nthreads=omp_nthreads, + fmap_bspline=False) + # set inputs + fmap_wf.inputs.inputnode.magnitude = fmap['magnitude'] + fmap_wf.inputs.inputnode.fieldmap = fmap['fieldmap'] + elif 'phasediff' in suffices: + from .phdiff import init_phdiff_wf + fmap_wf = init_phdiff_wf(omp_nthreads=omp_nthreads) + # set inputs + fmap_wf.inputs.inputnode.phasediff = fmap['phasediff'] + fmap_wf.inputs.inputnode.magnitude = [ + fmap_ for key, fmap_ in sorted(fmap.items()) + if key.startswith("magnitude") + ] + else: + raise ValueError('Fieldmaps of types %s are not supported' % + ', '.join(['"%s"' % f for f in suffices])) + + sdc_unwarp_wf = init_sdc_unwarp_wf( + omp_nthreads=omp_nthreads, + debug=debug, + name='sdc_unwarp_wf') + sdc_unwarp_wf.inputs.inputnode.metadata = epi_meta + + workflow.connect([ + (inputnode, sdc_unwarp_wf, [ + ('epi_file', 'inputnode.in_reference'), + ('epi_brain', 'inputnode.in_reference_brain'), + ('epi_mask', 'inputnode.in_mask')]), + (fmap_wf, sdc_unwarp_wf, [ + ('outputnode.fmap', 'inputnode.fmap'), + ('outputnode.fmap_ref', 'inputnode.fmap_ref'), + ('outputnode.fmap_mask', 'inputnode.fmap_mask')]), + ]) + + # FIELDMAP-less path + if any(fm['suffix'] == 'syn' for fm in fmaps): + from .syn import init_syn_sdc_wf + syn_sdc_wf = init_syn_sdc_wf( + epi_pe=epi_meta.get('PhaseEncodingDirection', None), + omp_nthreads=omp_nthreads) + + workflow.connect([ + (inputnode, syn_sdc_wf, [ + ('t1w_brain', 'inputnode.t1w_brain'), + ('epi_file', 'inputnode.epi_file'), + ('epi_brain', 'inputnode.epi_brain'), + ('std2anat_xfm', 'inputnode.std2anat_xfm')]), + ]) + + # XXX Eliminate branch when forcing isn't an option + if fmap['suffix'] == 'syn': # No fieldmaps, but --use-syn + outputnode.inputs.method = 'FLB ("fieldmap-less", SyN-based)' + sdc_unwarp_wf = syn_sdc_wf + else: # --force-syn was called when other fieldmap was present + sdc_unwarp_wf.__desc__ = None + workflow.connect([ + (syn_sdc_wf, outputnode, [ + ('outputnode.out_reference', 'syn_ref')]), + ]) workflow.connect([ (sdc_unwarp_wf, outputnode, [ ('outputnode.out_warp', 'out_warp'), - ('outputnode.out_reference', 'distorted_ref'), - ('outputnode.out_reference_brain', 'bold_ref_brain'), - ('outputnode.out_mask', 'bold_mask')]), + ('outputnode.out_reference', 'epi_file'), + ('outputnode.out_reference_brain', 'epi_brain'), + ('outputnode.out_mask', 'epi_mask')]), ]) return workflow diff --git a/sdcflows/workflows/pepolar.py b/sdcflows/workflows/pepolar.py index 409024f1d7..c29a1cfebf 100644 --- a/sdcflows/workflows/pepolar.py +++ b/sdcflows/workflows/pepolar.py @@ -73,15 +73,15 @@ def init_pepolar_unwarp_wf(omp_nthreads=1, matched_pe=False, fmaps_epi : list of tuple(pathlike, str) The list of EPI images that will be used in PE-Polar correction, and their corresponding ``PhaseEncodingDirection`` metadata. - The workflow will use the ``bold_pe_dir`` input to separate out those + The workflow will use the ``epi_pe_dir`` input to separate out those EPI acquisitions with opposed PE blips and those with matched PE blips (the latter could be none, and ``in_reference_brain`` would then be used). The workflow raises a ``ValueError`` when no images with opposed PE blips are found. - bold_pe_dir : str + epi_pe_dir : str The baseline PE direction. in_reference : pathlike - The baseline reference image (must correspond to ``bold_pe_dir``). + The baseline reference image (must correspond to ``epi_pe_dir``). in_reference_brain : pathlike The reference image above, but skullstripped. in_mask : pathlike @@ -110,7 +110,7 @@ def init_pepolar_unwarp_wf(omp_nthreads=1, matched_pe=False, inputnode = pe.Node(niu.IdentityInterface( fields=['fmaps_epi', 'in_reference', 'in_reference_brain', - 'in_mask', 'bold_pe_dir']), name='inputnode') + 'in_mask', 'epi_pe_dir']), name='inputnode') outputnode = pe.Node(niu.IdentityInterface( fields=['out_reference', 'out_reference_brain', 'out_warp', 'out_mask']), @@ -140,11 +140,11 @@ def init_pepolar_unwarp_wf(omp_nthreads=1, matched_pe=False, omp_nthreads=omp_nthreads) workflow.connect([ - (inputnode, qwarp, [(('bold_pe_dir', _qwarp_args), 'args')]), + (inputnode, qwarp, [(('epi_pe_dir', _qwarp_args), 'args')]), (inputnode, cphdr_warp, [('in_reference', 'hdr_file')]), (inputnode, prepare_epi_wf, [ ('fmaps_epi', 'inputnode.maps_pe'), - ('bold_pe_dir', 'inputnode.epi_pe'), + ('epi_pe_dir', 'inputnode.epi_pe'), ('in_reference_brain', 'inputnode.ref_brain')]), (prepare_epi_wf, qwarp, [('outputnode.opposed_pe', 'base_file'), ('outputnode.matched_pe', 'in_file')]), diff --git a/sdcflows/workflows/phdiff.py b/sdcflows/workflows/phdiff.py index fbd909e726..239850637b 100644 --- a/sdcflows/workflows/phdiff.py +++ b/sdcflows/workflows/phdiff.py @@ -84,8 +84,8 @@ def init_phdiff_wf(omp_nthreads, name='phdiff_wf'): workflow = Workflow(name=name) workflow.__desc__ = """\ A deformation field to correct for susceptibility distortions was estimated -based on a field map that was co-registered to the BOLD reference, -using a custom workflow of *fMRIPrep* derived from D. Greve's `epidewarp.fsl` +based on a field map that was co-registered to the EPI (echo-planar imaging) reference +run, using a custom workflow of *SDCFlows* derived from D. Greve's `epidewarp.fsl` [script](http://www.nmr.mgh.harvard.edu/~greve/fbirn/b0/epidewarp.fsl) and further improvements of HCP Pipelines [@hcppipelines]. """ diff --git a/sdcflows/workflows/syn.py b/sdcflows/workflows/syn.py index e0442e4ab0..73eaadda3b 100644 --- a/sdcflows/workflows/syn.py +++ b/sdcflows/workflows/syn.py @@ -39,7 +39,7 @@ LOGGER = logging.getLogger('nipype.workflow') -def init_syn_sdc_wf(omp_nthreads, bold_pe=None, +def init_syn_sdc_wf(omp_nthreads, epi_pe=None, atlas_threshold=3, name='syn_sdc_wf'): """ Build the *fieldmap-less* susceptibility-distortion estimation workflow. @@ -65,18 +65,18 @@ def init_syn_sdc_wf(omp_nthreads, bold_pe=None, from sdcflows.workflows.syn import init_syn_sdc_wf wf = init_syn_sdc_wf( - bold_pe='j', + epi_pe='j', omp_nthreads=8) Inputs ------ - bold_ref + in_reference reference image - bold_ref_brain + in_reference_brain skull-stripped reference image template : str Name of template targeted by ``template`` output space - t1_brain + t1w_brain skull-stripped, bias-corrected structural image std2anat_xfm inverse registration transform of T1w image to MNI template @@ -84,9 +84,9 @@ def init_syn_sdc_wf(omp_nthreads, bold_pe=None, Outputs ------- out_reference - the ``bold_ref`` image after unwarping + the ``in_reference`` image after unwarping out_reference_brain - the ``bold_ref_brain`` image after unwarping + the ``in_reference_brain`` image after unwarping out_warp the corresponding :abbr:`DFM (displacements field map)` compatible with ANTs @@ -110,9 +110,9 @@ def init_syn_sdc_wf(omp_nthreads, bold_pe=None, `_. """ - if bold_pe is None or bold_pe[0] not in ['i', 'j']: + if epi_pe is None or epi_pe[0] not in ['i', 'j']: LOGGER.warning('Incorrect phase-encoding direction, assuming PA (posterior-to-anterior).') - bold_pe = 'j' + epi_pe = 'j' workflow = Workflow(name=name) workflow.__desc__ = """\ @@ -127,8 +127,8 @@ def init_syn_sdc_wf(omp_nthreads, bold_pe=None, template [@fieldmapless3]. """.format(ants_ver=Registration().version or '') inputnode = pe.Node( - niu.IdentityInterface(['bold_ref', 'bold_ref_brain', 'template', - 't1_brain', 'std2anat_xfm']), + niu.IdentityInterface(['in_reference', 'in_reference_brain', 'template', + 't1w_brain', 'std2anat_xfm']), name='inputnode') outputnode = pe.Node( niu.IdentityInterface(['out_reference', 'out_reference_brain', @@ -172,7 +172,7 @@ def init_syn_sdc_wf(omp_nthreads, bold_pe=None, mem_gb=DEFAULT_MEMORY_MIN_GB) fixed_image_masks.inputs.in1 = 'NULL' - restrict = [[int(bold_pe[0] == 'i'), int(bold_pe[0] == 'j'), 0]] * 2 + restrict = [[int(epi_pe[0] == 'i'), int(epi_pe[0] == 'j'), 0]] * 2 syn = pe.Node( Registration(from_file=syn_transform, restrict_deformation=restrict), name='syn', n_procs=omp_nthreads) @@ -184,28 +184,28 @@ def init_syn_sdc_wf(omp_nthreads, bold_pe=None, skullstrip_bold_wf = init_skullstrip_bold_wf() workflow.connect([ - (inputnode, invert_t1w, [('t1_brain', 'in_file'), - ('bold_ref', 'ref_file')]), - (inputnode, ref_2_t1, [('bold_ref_brain', 'moving_image')]), + (inputnode, invert_t1w, [('t1w_brain', 'in_file'), + ('in_reference', 'ref_file')]), + (inputnode, ref_2_t1, [('in_reference_brain', 'moving_image')]), (invert_t1w, ref_2_t1, [('out_file', 'fixed_image')]), - (inputnode, t1_2_ref, [('bold_ref', 'reference_image')]), + (inputnode, t1_2_ref, [('in_reference', 'reference_image')]), (invert_t1w, t1_2_ref, [('out_file', 'input_image')]), (ref_2_t1, t1_2_ref, [('forward_transforms', 'transforms')]), (ref_2_t1, transform_list, [('forward_transforms', 'in1')]), (inputnode, transform_list, [ ('std2anat_xfm', 'in2'), (('template', _prior_path), 'in3')]), - (inputnode, atlas_2_ref, [('bold_ref', 'reference_image')]), + (inputnode, atlas_2_ref, [('in_reference', 'reference_image')]), (transform_list, atlas_2_ref, [('out', 'transforms')]), (atlas_2_ref, threshold_atlas, [('output_image', 'in_file')]), (threshold_atlas, fixed_image_masks, [('out_file', 'in2')]), - (inputnode, syn, [('bold_ref_brain', 'moving_image')]), + (inputnode, syn, [('in_reference_brain', 'moving_image')]), (t1_2_ref, syn, [('output_image', 'fixed_image')]), (fixed_image_masks, syn, [('out', 'fixed_image_masks')]), (syn, outputnode, [('forward_transforms', 'out_warp')]), (syn, unwarp_ref, [('forward_transforms', 'transforms')]), - (inputnode, unwarp_ref, [('bold_ref', 'reference_image'), - ('bold_ref', 'input_image')]), + (inputnode, unwarp_ref, [('in_reference', 'reference_image'), + ('in_reference', 'input_image')]), (unwarp_ref, skullstrip_bold_wf, [ ('output_image', 'inputnode.in_file')]), (unwarp_ref, outputnode, [('output_image', 'out_reference')]), diff --git a/sdcflows/workflows/tests/test_pepolar.py b/sdcflows/workflows/tests/test_pepolar.py index c35021785a..6a5a169e1f 100644 --- a/sdcflows/workflows/tests/test_pepolar.py +++ b/sdcflows/workflows/tests/test_pepolar.py @@ -132,7 +132,7 @@ def test_pepolar_wf1(bids_layouts, output_path, dataset, workdir): wf = init_pepolar_unwarp_wf(omp_nthreads=cpu_count(), matched_pe=matched_pe) wf.inputs.inputnode.fmaps_epi = [(im.path, im.get_metadata()['PhaseEncodingDirection']) for im in epidata] - wf.inputs.inputnode.bold_pe_dir = bold.get_metadata()['PhaseEncodingDirection'] + wf.inputs.inputnode.epi_pe_dir = bold.get_metadata()['PhaseEncodingDirection'] if output_path: from nipype.interfaces import utility as niu @@ -166,7 +166,7 @@ def test_pepolar_wf1(bids_layouts, output_path, dataset, workdir): boiler.connect([ (wf, split_field, [ - ('inputnode.bold_pe_dir', 'pe_dir'), + ('inputnode.epi_pe_dir', 'pe_dir'), ('outputnode.out_warp', 'in_field')]), (split_field, rep, [ ('out', 'fieldmap')]), diff --git a/sdcflows/workflows/unwarp.py b/sdcflows/workflows/unwarp.py index c12ca9ad51..ae74732b77 100644 --- a/sdcflows/workflows/unwarp.py +++ b/sdcflows/workflows/unwarp.py @@ -23,7 +23,7 @@ from nipype.interfaces import ants, fsl, utility as niu from niworkflows.engine.workflows import LiterateWorkflow as Workflow from niworkflows.interfaces import itk -from niworkflows.interfaces.images import DemeanImage, FilledImageLike +from niworkflows.interfaces.images import FilledImageLike from niworkflows.interfaces.registration import ANTSApplyTransformsRPT, ANTSRegistrationRPT from niworkflows.interfaces.bids import DerivativesDataSink from niworkflows.func.util import init_enhance_and_skullstrip_bold_wf @@ -31,7 +31,7 @@ from ..interfaces.fmap import get_ees as _get_ees, FieldToRadS -def init_sdc_unwarp_wf(omp_nthreads, fmap_demean, debug, name='sdc_unwarp_wf'): +def init_sdc_unwarp_wf(omp_nthreads, debug, name='sdc_unwarp_wf'): """ Apply the warping given by a displacements fieldmap. @@ -48,41 +48,48 @@ def init_sdc_unwarp_wf(omp_nthreads, fmap_demean, debug, name='sdc_unwarp_wf'): from sdcflows.workflows.unwarp import init_sdc_unwarp_wf wf = init_sdc_unwarp_wf(omp_nthreads=8, - fmap_demean=True, debug=False) + Parameters + ---------- + omp_nthreads : int + Maximum number of threads an individual process may use. + debug : bool + Run fast configurations of registrations. + name : str + Unique name of this workflow. Inputs - - in_reference - the reference image - in_reference_brain - the reference image (skull-stripped) - in_mask - a brain mask corresponding to ``in_reference`` - metadata - metadata associated to the ``in_reference`` EPI input - fmap - the fieldmap in Hz - fmap_ref - the reference (anatomical) image corresponding to ``fmap`` - fmap_mask - a brain mask corresponding to ``fmap`` + ------ + in_reference + the reference image + in_reference_brain + the reference image (skull-stripped) + in_mask + a brain mask corresponding to ``in_reference`` + metadata + metadata associated to the ``in_reference`` EPI input + fmap + the fieldmap in Hz + fmap_ref + the reference (anatomical) image corresponding to ``fmap`` + fmap_mask + a brain mask corresponding to ``fmap`` Outputs - - out_reference - the ``in_reference`` after unwarping - out_reference_brain - the ``in_reference`` after unwarping and skullstripping - out_warp - the corresponding :abbr:`DFM (displacements field map)` compatible with - ANTs - out_jacobian - the jacobian of the field (for drop-out alleviation) - out_mask - mask of the unwarped input file + ------- + out_reference + the ``in_reference`` after unwarping + out_reference_brain + the ``in_reference`` after unwarping and skullstripping + out_warp + the corresponding :abbr:`DFM (displacements field map)` compatible with + ANTs + out_jacobian + the jacobian of the field (for drop-out alleviation) + out_mask + mask of the unwarped input file """ workflow = Workflow(name=name) @@ -192,23 +199,8 @@ def init_sdc_unwarp_wf(omp_nthreads, fmap_demean, debug, name='sdc_unwarp_wf'): ('outputnode.mask_file', 'out_mask'), ('outputnode.skull_stripped_file', 'out_reference_brain')]), (jac_dfm, outputnode, [('jacobian_image', 'out_jacobian')]), + (gen_vsm, vsm2dfm, [('shift_out_file', 'in_file')]), ]) - - if fmap_demean: - # Demean within mask - demean = pe.Node(DemeanImage(), name='demean') - - workflow.connect([ - (gen_vsm, demean, [('shift_out_file', 'in_file')]), - (fmap_mask2ref_apply, demean, [('output_image', 'in_mask')]), - (demean, vsm2dfm, [('out_file', 'in_file')]), - ]) - - else: - workflow.connect([ - (gen_vsm, vsm2dfm, [('shift_out_file', 'in_file')]), - ]) - return workflow