Canlab Automated Preprocessing with Nipype

Written by Luke Chang 9/24/2014 for Columbia PTSD Data


We create a preprocessing pipeline using Nipype for every subject and iterate over runs. The pipeline is using spm for now to be consistent with Tor's original pipeline, but it is really easy to switch out any part with other software packages (e.g., fsl, nipy, afni, freesurfer, etc.). The pipeline can be automatically distributed to multiple processors using "multiprocessing" plugin or to Blanca using the "sge" plugin. This iPython notebook can be run on your local computer via an SSH tunnel to Blanca. Here is a helpful link with shortcuts for using iPython notebooks. There is also a separate python script that can be run on an interactive node on blanca that works better right now until I solve the queue issue (currently waiting to submit jobs until previous one are out of completed state ~ 20 min)

You will probably need to set up paths to my python build if you run this on Blanca (/projects/luch0518/software/). The RC "modules" are outdated an cannot be updated without root access.

Preprocessing Steps

  1. Slice timing
  2. Realignment to mean image
  3. Coregister anatomical to functional
  4. Segment anatomical with bias correction
  5. Normalize anatomical to MNI space
  6. Apply normalization to functionals
  7. Smooth - 8mm
  8. Artifact Detection - 3 sd of global mean (3.5 sd of root mean squared sucessive differences, mahalonobis distance across slices? ask tor for more details, but ignore fo now)
  9. Create nuisance covariate matrix for glm (note the file has headers and NaNs for missing data in the derivative columns)

Figures Generated

  • Realignment Parameters (plot_realign/Realignment_parameters.pdf)
  • MNI space single subject overlayed on mean warped functional (plot_normalization_check/Normalized_Functional_Check.pdf)
  • Spikes identified using global signal (art/plot.rafunc.png)

To Do

  • Push general code to github and import functions
  • Finish making all of Tor's Plots
  • Figure out how Tor identifies spikes using mahalanobis distance
  • Figure out how to fix delay introduced by completed states in Blanca Queue
  • Add Sagittal plane to coregistration check
  • Make function to look at overlap of all normalized binary masks. Will need to add new node for this.

How to Set up iPython Notebook on Blanca

  1. 1) log on to Blanca and make sure you are using Luke's Python Anaconda distribution


  2. Start ipython notebook server on blanca. Disable browser and specify port

    ``` ipython notebook --no-browser --port=8889```

  3. On your local computer, login to blanca using portforwarding

    ``` ssh -N -f -L localhost:8889:localhost:8889 [email protected] ```

  4. Open your browser to localhost:8889/tree
  5. The iPython notebook server should shutdown when you close the SSH connection, but just in case you can also kill the process. Here is the command to find the process id:

    ```ps aux | grep localhost:8889```

  6. Nipype can submit jobs automatically to Blanca. You can check your queue or delete any process

    Check queue

    ```qstat -u $USER```

    Delete Queue

    ```qselect -u $USER | xargs qdel```

Function definitions

The following code creates a preprocessing workflow. We define several helper functions and also create a few new interfaces for new functionality to nipype such as plotting a montage of the normalization, creating a covariate matrix, and plotting realignment parameters. It is quite easy to code wrappers for other functions including matlab. In the preprocessing workflow, we define nodes and then how the output of each node is routed or connected to the input of the next node. This creates a graphical model of the pipeline.

In [1]:
from nipype.interfaces import spm
import as nio           # Data i/o
import nipype.interfaces.utility as util     # utility
from nipype.pipeline.engine import Node, Workflow
from nipype.interfaces.base import BaseInterface, TraitedSpec, File, traits
import nipype.algorithms.rapidart as ra      # artifact detection
from nipype.interfaces.nipy.preprocess import ComputeMask
import nipype.interfaces.matlab as mlab
import os
import nibabel as nib
from IPython.display import Image
import glob

# Specify various inputs files for pipeline
# spm_path = '/projects/ics/software/spm/spm8_r5236/'
spm_path = '/Users/lukechang/Documents/Matlab/spm8/'
canonical_file = spm_path + 'canonical/single_subj_T1.nii'
template_file = spm_path + 'templates/T1.nii'

# Set the way matlab should be called
# mlab.MatlabCommand.set_default_matlab_cmd("matlab -nodesktop -nosplash -nojvm -noFigureWindows")
mlab.MatlabCommand.set_default_matlab_cmd("matlab -nodesktop -nosplash")

## Functions

def get_n_slices(volume):
    import nibabel as nib
    nii = nib.load(volume)
    return nii.get_shape()[2]

def get_ta(tr, n_slices):
    return tr - tr/float(n_slices)

def get_slice_order(volume):
    import nibabel as nib
    nii = nib.load(volume)
    n_slices = nii.get_shape()[2]
    return range(1,n_slices+1)

def get_vox_dims(volume):
    import nibabel as nib
    if isinstance(volume, list):
        volume = volume[0]
    nii = nib.load(volume)
    hdr = nii.get_header()
    voxdims = hdr.get_zooms()
    return [float(voxdims[0]), float(voxdims[1]), float(voxdims[2])]

class Plot_Coregistration_Montage_InputSpec(TraitedSpec):
    wra_img = File(exists=True, mandatory=True) 
    canonical_img = File(exists=True, mandatory=True)
    title = traits.Str("Normalized Functional Check", usedefault=True)
class Plot_Coregistration_Montage_OutputSpec(TraitedSpec):
    plot = File(exists=True)

class Plot_Coregistration_Montage(BaseInterface):
    # This function creates a plot of an axial montage of the average normalized functional data 
    # with the spm MNI space single subject T1 overlaid. Useful for checking normalization
    input_spec = Plot_Coregistration_Montage_InputSpec
    output_spec = Plot_Coregistration_Montage_OutputSpec

    def _run_interface(self, runtime):
        import nibabel as nib
        from nilearn import plotting, datasets, image
        from nipype.interfaces.base import isdefined
        import numpy as np
        import pylab as plt
        import os
        wra_img = nib.load(self.inputs.wra_img)
        canonical_img = nib.load(self.inputs.canonical_img)
        title = self.inputs.title
        mean_wraimg = image.mean_img(wra_img)
        if title != "":
            filename = title.replace(" ", "_")+".pdf"
            filename = "plot.pdf"
        fig = plotting.plot_anat(mean_wraimg, title="wrafunc & canonical single subject", cut_coords=range(-40, 40, 10), display_mode='z')
        self._plot = filename
        return runtime
    def _list_outputs(self):
        outputs = self._outputs().get()
        outputs["plot"] = os.path.abspath(self._plot)
        return outputs
class PlotRealignmentParametersInputSpec(TraitedSpec):
    realignment_parameters = File(exists=True, mandatory=True)
    outlier_files = File(exists=True)
    title = traits.Str("Realignment parameters", usedefault=True)
    dpi = traits.Int(300, usedefault = True)
class PlotRealignmentParametersOutputSpec(TraitedSpec):
    plot = File(exists=True)

class PlotRealignmentParameters(BaseInterface):
    #This function is adapted from Chris Gorgolewski's code and generates a plot of the realignment parameters

    input_spec = PlotRealignmentParametersInputSpec
    output_spec = PlotRealignmentParametersOutputSpec
    def _run_interface(self, runtime):
        from nipype.interfaces.base import isdefined
        import numpy as np
        import pylab as plt
        import os
        realignment_parameters = np.loadtxt(self.inputs.realignment_parameters)
        title = self.inputs.title
        F = plt.figure(figsize=(8.3,11.7))
        F.text(0.5, 0.96, self.inputs.title, horizontalalignment='center')
        ax1 = plt.subplot2grid((2,2),(0,0), colspan=2)
        handles =ax1.plot(realignment_parameters[:,0:3])
        ax1.legend(handles, ["x translation", "y translation", "z translation"], loc=0)
        ax1.set_xlabel("image #")
        ax1.set_ylim(bottom = realignment_parameters[:,0:3].min(), top = realignment_parameters[:,0:3].max())
        ax2 = plt.subplot2grid((2,2),(1,0), colspan=2)
        handles= ax2.plot(realignment_parameters[:,3:6]*180.0/np.pi)
        ax2.legend(handles, ["pitch", "roll", "yaw"], loc=0)
        ax2.set_xlabel("image #")
        ax2.set_ylim(bottom=(realignment_parameters[:,3:6]*180.0/np.pi).min(), top= (realignment_parameters[:,3:6]*180.0/np.pi).max())
        if isdefined(self.inputs.outlier_files):
                outliers = np.loadtxt(self.inputs.outlier_files)
            except IOError as e:
                if e.args[0] == "End-of-file reached before encountering data.":
                if outliers.size > 0:
                    ax1.vlines(outliers, ax1.get_ylim()[0], ax1.get_ylim()[1])
                    ax2.vlines(outliers, ax2.get_ylim()[0], ax2.get_ylim()[1])
        if title != "":
            filename = title.replace(" ", "_")+".pdf"
            filename = "plot.pdf"
        del F
        self._plot = filename
        return runtime
    def _list_outputs(self):
        outputs = self._outputs().get()
        outputs["plot"] = os.path.abspath(self._plot)
        return outputs

class Create_Covariates_InputSpec(TraitedSpec):	
    realignment_parameters = File(exists=True, mandatory=True) 
    spike_id = File(exists=True, mandatory=True)

class Create_Covariates_OutputSpec(TraitedSpec):
    covariates = File(exists=True)

class Create_Covariates(BaseInterface):
    # This function creates a matrix of the nuisance covariates used for the GLM.
    # Note that the output file has headers and also has NaNs that may need to be accounted for.
    input_spec = Create_Covariates_InputSpec
    output_spec = Create_Covariates_OutputSpec

    def _run_interface(self, runtime):
        import pandas as pd
        import numpy as np

        ra = pd.read_table(self.inputs.realignment_parameters, header=None, sep=r"\s*", names=['ra' + str(x) for x in range(1,7)])
        spike = pd.read_table(self.inputs.spike_id, header=None,names=['Spikes'])

        ra = ra-ra.mean() #mean center
        ra[['rasq' + str(x) for x in range(1,7)]] = ra**2 #add squared
        ra[['radiff' + str(x) for x in range(1,7)]] = pd.DataFrame(ra[ra.columns[0:6]].diff()) #derivative
        ra[['radiffsq' + str(x) for x in range(1,7)]] = pd.DataFrame(ra[ra.columns[0:6]].diff())**2 #derivatives squared

        #build spike regressors
        for i,loc in enumerate(spike['Spikes']):
            ra['spike' + str(i+1)] = 0
            ra['spike' + str(i+1)].iloc[loc] = 1

        filename = 'covariates.csv'
        ra.to_csv(filename, index=False) #write out to file
        self._covariates = filename

        return runtime

    def _list_outputs(self):
        outputs = self._outputs().get()
        outputs["covariates"] = os.path.abspath(self._covariates)
        return outputs

def create_preproc_func_pipeline(data_dir=None, subject_id=None, task_list=None):

    ## Set up Nodes

    #Setup Data Source for Input Data
    #     data_dir = '/Users/lukechang/Dropbox/PTSD/Data/Imaging'
    #     task_list = ['s1_r1Cond','s1_r1Ext','s1_r2Rec','s1_r2Ren']
    ds = Node(nio.DataGrabber(infields=['subject_id', 'task_id'], outfields=['func', 'struc']),name='datasource')
    ds.inputs.base_directory = os.path.abspath(data_dir + '/' + subject_id)
    ds.inputs.template = '*'
    ds.inputs.sort_filelist = True
    ds.inputs.template_args = {'func': [['task_id']], 'struc':[]}
    ds.inputs.field_template = {'func': 'Functional/Raw/%s/func.nii','struc': 'Structural/SPGR/spgr.nii'}
    ds.inputs.subject_id = subject_id
    ds.inputs.task_id = task_list
    ds.iterables = ('task_id',task_list)
    # #show datafiles

    # #Setup Data Sinker for writing output files
    # datasink = Node(nio.DataSink(), name='sinker')
    # datasink.inputs.base_directory = '/path/to/output'
    # workflow.connect(realigner, 'realignment_parameters', datasink, '[email protected]')
    # datasink.inputs.substitutions = [('_variable', 'variable'),('file_subject_', '')]

    #Get Timing Acquisition for slice timing
    tr = 2
    ta = Node(interface=util.Function(input_names=['tr', 'n_slices'], output_names=['ta'],  function = get_ta), name="ta")

    #Slice Timing: sequential ascending 
    slice_timing = Node(interface=spm.SliceTiming(), name="slice_timing")
    slice_timing.inputs.time_repetition = tr
    slice_timing.inputs.ref_slice = 1

    #Realignment - 6 parameters - realign to first image of very first series.
    realign = Node(interface=spm.Realign(), name="realign")
    realign.inputs.register_to_mean = True

    #Plot Realignment
    plot_realign = Node(interface=PlotRealignmentParameters(), name="plot_realign")

    #Artifact Detection
    art = Node(interface=ra.ArtifactDetect(), name="art")
    art.inputs.use_differences      = [True,False]
    art.inputs.use_norm             = True
    art.inputs.norm_threshold       = 1
    art.inputs.zintensity_threshold = 3
    art.inputs.mask_type            = 'file'
    art.inputs.parameter_source     = 'SPM'
    #Coregister - 12 parameters, cost function = 'nmi', fwhm 7, interpolate, don't mask
    #anatomical to functional mean across all available data.
    coregister = Node(interface=spm.Coregister(), name="coregister")
    coregister.inputs.jobtype = 'estimate'

    # Segment structural, gray/white/csf,mni, 
    segment = Node(interface=spm.Segment(), name="segment")
    segment.inputs.save_bias_corrected = True
    #Normalize - structural to MNI - then apply this to the coregistered functionals
    normalize = Node(interface=spm.Normalize(), name = "normalize")
    normalize.inputs.template = os.path.abspath(template_file)

    #Plot normalization Check
    plot_normalization_check = Node(interface=Plot_Coregistration_Montage(), name="plot_normalization_check")
    plot_normalization_check.inputs.canonical_img = canonical_file
    #Create Mask
    compute_mask = Node(interface=ComputeMask(), name="compute_mask")
    #remove lower 5% of histogram of mean image
    compute_mask.inputs.m = .05
    #implicit masking (.im) = 0, dtype = 0
    smooth = Node(interface=spm.Smooth(), name = "smooth")
    fwhmlist = [8]
    smooth.iterables = ('fwhm',fwhmlist)

    #Create Covariate matrix
    make_covariates = Node(interface=Create_Covariates(), name="make_covariates")
    ## Create Pipeline

    Preprocessed = Workflow(name="Preprocessed")
    Preprocessed.base_dir = os.path.abspath(data_dir + '/' + subject_id + '/Functional')

                        (ds, ta, [(('func', get_n_slices), "n_slices")]),
                        (ta, slice_timing, [("ta", "time_acquisition")]),
                        (ds, slice_timing, [('func', 'in_files'),
                                            (('func', get_n_slices), "num_slices"),
                                            (('func', get_slice_order), "slice_order"),
                        (slice_timing, realign, [('timecorrected_files', 'in_files')]),
                        (realign, compute_mask, [('mean_image','mean_volume')]),
                        (realign,coregister, [('mean_image', 'target')]),
                        (ds,coregister, [('struc', 'source')]),
                        (coregister,segment, [('coregistered_source', 'data')]),
                        (segment, normalize, [('transformation_mat','parameter_file'),
                                             ('bias_corrected_image', 'source'),]),
                        (realign,normalize, [('realigned_files', 'apply_to_files'),
                                             (('realigned_files', get_vox_dims), 'write_voxel_sizes')]),
                        (normalize, smooth, [('normalized_files', 'in_files')]),
                        (realign,plot_realign, [('realignment_parameters', 'realignment_parameters')]),
                        (normalize, plot_normalization_check, [('normalized_files', 'wra_img')]),
                        (realign, make_covariates, [('realignment_parameters', 'realignment_parameters')]),
                        (art, make_covariates, [('outlier_files', 'spike_id')]),
    return Preprocessed5
Couldn't import dot_parser, loading of dot files will not be possible.


Here is a directed acyclic graph of the preprocessing pipeline. Each processing step in the workflow is a node in the graph. The node names will be directories created in the 'Preprocessed' folder. You can easily run different pipelines on the same data without interfering with other pipelines. All of the files you will need for subsequent analyses will be in each of these folders. If you make changes to a node, nipype will only rerun the thing you changed and all dependent nodes. Simply rerun the code. Also, you can easily iterate over different settings without creating separate pipelines using iterables. Here we are treating the smoothing parameter as an iterable. It is currently set at 8mm, but you can input a vector of different sizes and it will run all of them.

In [4]:
data_dir = '/Users/lukechang/Dropbox/PTSD/Data/Imaging/'
sub = 'subj46153C'
Preprocessed = create_preproc_func_pipeline(data_dir = data_dir, subject_id=sub)
Preprocessed.write_graph(data_dir + sub + "/")
Image(filename=data_dir + sub + '/')
INFO:workflow:Converting dotfile: /Users/lukechang/Dropbox/PTSD/Data/Imaging/subj46153C/ to png format

Run Code

Running the pipeline is really easy and can be run on your local laptop, mac workstation, or blanca. Basically, we create a Pipeline for a given subject and execute the pipeline iterating over runs. This particular code looks for available subjects and runs and automatically distributes them in parallel. It assumes your data is in the standard Canlab format (e.g., Imaging/Subjects/Functional/Raw/Run/func.nii) will excecute the pipeline.
plugin="MultiProc" will run in parallel on a local computer while plugin="PBS" will submit jobs on Blanca

Submitting jobs using PBS is really slow because nipype is waiting to submit jobs while they are in "c" completed state. I still need to figure this out. I have a python script called in my Notebooks folder that I am running using an interactive node on Blanca. It's really fast! (~10min to run a subject's 8 functional runs)

In [160]:
# Create Pipeline for subject

# subject_id = 'subj46153C'
# task_list=['s1_r1Cond','s1_r1Ext','s2_r1Cond','s2_r2Rec','s1_r1Ext','s1_r2Ren','s2_r1Ext','s2_r2Ren']
# Preprocessed = create_preproc_func_pipeline(data_dir=data_dir, subject_id = subject_id, task_list=task_list)

sublist = sorted([x.split('/')[-1] for x in glob.glob(data_dir + '/subj*')])

#Run first half of subjects
for sub in reversed(sublist):
    #Glob Subject runs as they vary
    runlist = [x.split('/')[-1] for x in glob.glob(data_dir + '/' + sub + '/Functional/Raw/*')]
    Preprocessed = create_preproc_func_pipeline(data_dir=data_dir, subject_id = sub, task_list=runlist)

    print  data_dir + '/' + sub

    # Write out pipeline as a DAG
    Preprocessed.write_graph(dotfilename=data_dir + '/' + sub + "/Functional/",format='svg')
#     Preprocessed.write_graph(data_dir + '/' + subject_id + "/")
#     Image(filename=data_dir + '/' + sub + "/Functional/")'MultiProc', plugin_args={'n_procs' : 12}) 
#     Run on workstation in'PBS') #Run on Blanca in parallel
INFO:workflow:Converting dotfile: /Users/lukechang/Dropbox/PTSD/Data/Imaging/subj46153C/ to png format
INFO:workflow:['check', 'execution', 'logging']
INFO:workflow:Running in parallel.
INFO:workflow:Submitting 4 jobs
INFO:workflow:Executing: datasource.bI.b1 ID: 0
INFO:workflow:Executing: datasource.bI.b0 ID: 13
INFO:workflow:Executing: datasource.bI.b2 ID: 26
INFO:workflow:Executing: datasource.bI.b3 ID: 39
INFO:workflow:[Job finished] jobname: datasource.bI.b1 jobid: 0
INFO:workflow:Executing node datasource.bI.b1 in dir: /Users/lukechang/Dropbox/PTSD/Data/Imaging/subj46153C/Functional/Preprocessed/_task_id_s2_r2Rec/datasource
INFO:workflow:Executing node datasource.bI.b0 in dir: /Users/lukechang/Dropbox/PTSD/Data/Imaging/subj46153C/Functional/Preprocessed/_task_id_s2_r1Cond/datasource
INFO:workflow:Executing node datasource.bI.b2 in dir: /Users/lukechang/Dropbox/PTSD/Data/Imaging/subj46153C/Functional/Preprocessed/_task_id_s2_r1Ext/datasource
INFO:workflow:Executing node datasource.bI.b3 in dir: /Users/lukechang/Dropbox/PTSD/Data/Imaging/subj46153C/Functional/Preprocessed/_task_id_s2_r2Ren/datasource
INFO:workflow:[Job finished] jobname: datasource.bI.b0 jobid: 13
INFO:workflow:[Job finished] jobname: datasource.bI.b2 jobid: 26
INFO:workflow:[Job finished] jobname: datasource.bI.b3 jobid: 39
INFO:workflow:Submitting 4 jobs
INFO:workflow:Executing: ta.b1 ID: 1
INFO:workflow:[Job finished] jobname: ta.b1 jobid: 1
INFO:workflow:Executing: ta.b0 ID: 14
INFO:workflow:[Job finished] jobname: ta.b0 jobid: 14
INFO:workflow:Executing: ta.b2 ID: 27
INFO:workflow:[Job finished] jobname: ta.b2 jobid: 27
INFO:workflow:Executing: ta.b3 ID: 40
INFO:workflow:[Job finished] jobname: ta.b3 jobid: 40
INFO:workflow:Submitting 4 jobs
INFO:workflow:Executing: slice_timing.b1 ID: 2
INFO:workflow:[Job finished] jobname: slice_timing.b1 jobid: 2
INFO:workflow:Executing: slice_timing.b0 ID: 15
INFO:workflow:[Job finished] jobname: slice_timing.b0 jobid: 15
INFO:workflow:Executing: slice_timing.b2 ID: 28
INFO:workflow:[Job finished] jobname: slice_timing.b2 jobid: 28
INFO:workflow:Executing: slice_timing.b3 ID: 41
INFO:workflow:[Job finished] jobname: slice_timing.b3 jobid: 41
INFO:workflow:Submitting 4 jobs
INFO:workflow:Executing: realign.b1 ID: 3
INFO:workflow:[Job finished] jobname: realign.b1 jobid: 3
INFO:workflow:Executing: realign.b0 ID: 16
INFO:workflow:[Job finished] jobname: realign.b0 jobid: 16
INFO:workflow:Executing: realign.b2 ID: 29
INFO:workflow:[Job finished] jobname: realign.b2 jobid: 29
INFO:workflow:Executing: realign.b3 ID: 42
INFO:workflow:[Job finished] jobname: realign.b3 jobid: 42
INFO:workflow:Submitting 12 jobs
INFO:workflow:Executing: coregister.b1 ID: 4
INFO:workflow:[Job finished] jobname: coregister.b1 jobid: 4
INFO:workflow:Executing: plot_realign.b1 ID: 6
INFO:workflow:[Job finished] jobname: plot_realign.b1 jobid: 6
INFO:workflow:Executing: compute_mask.b1 ID: 7
INFO:workflow:[Job finished] jobname: compute_mask.b1 jobid: 7
INFO:workflow:Executing: plot_realign.b0 ID: 17
INFO:workflow:[Job finished] jobname: plot_realign.b0 jobid: 17
INFO:workflow:Executing: compute_mask.b0 ID: 18
INFO:workflow:[Job finished] jobname: compute_mask.b0 jobid: 18
INFO:workflow:Executing: coregister.b0 ID: 21
INFO:workflow:[Job finished] jobname: coregister.b0 jobid: 21
INFO:workflow:Executing: coregister.b2 ID: 30
INFO:workflow:[Job finished] jobname: coregister.b2 jobid: 30
INFO:workflow:Executing: plot_realign.b2 ID: 35
INFO:workflow:[Job finished] jobname: plot_realign.b2 jobid: 35
INFO:workflow:Executing: compute_mask.b2 ID: 36
INFO:workflow:[Job finished] jobname: compute_mask.b2 jobid: 36
INFO:workflow:Executing: plot_realign.b3 ID: 43
INFO:workflow:[Job finished] jobname: plot_realign.b3 jobid: 43
INFO:workflow:Executing: coregister.b3 ID: 44
INFO:workflow:[Job finished] jobname: coregister.b3 jobid: 44
INFO:workflow:Executing: compute_mask.b3 ID: 46
INFO:workflow:[Job finished] jobname: compute_mask.b3 jobid: 46
INFO:workflow:Submitting 8 jobs
INFO:workflow:Executing: segment.b1 ID: 5
INFO:workflow:[Job finished] jobname: segment.b1 jobid: 5
INFO:workflow:Executing: art.b1 ID: 10
INFO:workflow:[Job finished] jobname: art.b1 jobid: 10
INFO:workflow:Executing: art.b0 ID: 19
INFO:workflow:[Job finished] jobname: art.b0 jobid: 19
INFO:workflow:Executing: segment.b0 ID: 22
INFO:workflow:[Job finished] jobname: segment.b0 jobid: 22
INFO:workflow:Executing: segment.b2 ID: 31
INFO:workflow:[Job finished] jobname: segment.b2 jobid: 31
INFO:workflow:Executing: art.b2 ID: 37
INFO:workflow:[Job finished] jobname: art.b2 jobid: 37
INFO:workflow:Executing: segment.b3 ID: 45
INFO:workflow:[Job finished] jobname: segment.b3 jobid: 45
INFO:workflow:Executing: art.b3 ID: 50
INFO:workflow:[Job finished] jobname: art.b3 jobid: 50
INFO:workflow:Submitting 8 jobs
INFO:workflow:Executing: normalize.b1 ID: 8
INFO:workflow:[Job finished] jobname: normalize.b1 jobid: 8
INFO:workflow:Executing: make_covariates.b1 ID: 11
INFO:workflow:Executing: make_covariates.b0 ID: 20
INFO:workflow:[Job finished] jobname: make_covariates.b0 jobid: 20
INFO:workflow:Executing node make_covariates.b1 in dir: /Users/lukechang/Dropbox/PTSD/Data/Imaging/subj46153C/Functional/Preprocessed/_task_id_s2_r2Rec/make_covariates
INFO:workflow:Executing: normalize.b0 ID: 23
INFO:workflow:[Job finished] jobname: normalize.b0 jobid: 23
INFO:workflow:Executing: normalize.b2 ID: 32
INFO:workflow:[Job finished] jobname: normalize.b2 jobid: 32
INFO:workflow:Executing: make_covariates.b2 ID: 38
INFO:workflow:[Job finished] jobname: make_covariates.b2 jobid: 38
INFO:workflow:Executing: normalize.b3 ID: 47
INFO:workflow:[Job finished] jobname: normalize.b3 jobid: 47
INFO:workflow:Executing: make_covariates.b3 ID: 51
INFO:workflow:[Job finished] jobname: make_covariates.b3 jobid: 51
INFO:workflow:Submitting 8 jobs
INFO:workflow:Executing: plot_normalization_check.b1 ID: 9
INFO:workflow:[Job finished] jobname: plot_normalization_check.b1 jobid: 9
INFO:workflow:Executing: smooth.aI.a0.b1 ID: 12
INFO:workflow:[Job finished] jobname: smooth.aI.a0.b1 jobid: 12
INFO:workflow:Executing: plot_normalization_check.b0 ID: 24
INFO:workflow:[Job finished] jobname: plot_normalization_check.b0 jobid: 24
INFO:workflow:Executing: smooth.aI.a0.b0 ID: 25
INFO:workflow:[Job finished] jobname: smooth.aI.a0.b0 jobid: 25
INFO:workflow:Executing: smooth.aI.a0.b2 ID: 33
INFO:workflow:[Job finished] jobname: smooth.aI.a0.b2 jobid: 33
INFO:workflow:Executing: plot_normalization_check.b2 ID: 34
INFO:workflow:[Job finished] jobname: plot_normalization_check.b2 jobid: 34
INFO:workflow:Executing: smooth.aI.a0.b3 ID: 48
INFO:workflow:[Job finished] jobname: smooth.aI.a0.b3 jobid: 48
INFO:workflow:Executing: plot_normalization_check.b3 ID: 49
INFO:workflow:[Job finished] jobname: plot_normalization_check.b3 jobid: 49
INFO:workflow:[Job finished] jobname: make_covariates.b1 jobid: 11
<networkx.classes.digraph.DiGraph at 0x10e242590>