Skip to content

Commit d353f0d

Browse files
authored
Merge pull request #2878 from rciric/master
[ENH] CompCor enhancement
2 parents 3b21e66 + b80a3d7 commit d353f0d

File tree

8 files changed

+304
-95
lines changed

8 files changed

+304
-95
lines changed

.zenodo.json

+5
Original file line numberDiff line numberDiff line change
@@ -330,6 +330,11 @@
330330
"name": "Liem, Franz",
331331
"orcid": "0000-0003-0646-4810"
332332
},
333+
{
334+
"affiliation": "Stanford University",
335+
"name": "Ciric, Rastko",
336+
"orcid": "0000-0001-6347-7939"
337+
},
333338
{
334339
"affiliation": "The Centre for Addiction and Mental Health",
335340
"name": "Joseph, Michael",

nipype/algorithms/confounds.py

+220-72
Large diffs are not rendered by default.

nipype/algorithms/tests/test_CompCor.py

+57-9
Original file line numberDiff line numberDiff line change
@@ -48,20 +48,48 @@ def test_compcor(self):
4848

4949
self.run_cc(
5050
CompCor(
51+
num_components=6,
5152
realigned_file=self.realigned_file,
5253
mask_files=self.mask_files,
5354
mask_index=0), expected_components)
5455

5556
self.run_cc(
5657
ACompCor(
58+
num_components=6,
5759
realigned_file=self.realigned_file,
5860
mask_files=self.mask_files,
5961
mask_index=0,
6062
components_file='acc_components_file'), expected_components,
6163
'aCompCor')
6264

65+
def test_compcor_variance_threshold_and_metadata(self):
66+
expected_components = [['-0.2027150345', '-0.4954813834'],
67+
['0.2565929051', '0.7866217875'],
68+
['-0.3550986008', '-0.0089784905'],
69+
['0.7512786244', '-0.3599828482'],
70+
['-0.4500578942', '0.0778209345']]
71+
expected_metadata = {
72+
'component': 'CompCor00',
73+
'mask': 'mask',
74+
'singular_value': '4.0720553036',
75+
'variance_explained': '0.5527211465',
76+
'cumulative_variance_explained': '0.5527211465',
77+
'retained': 'True',
78+
}
79+
ccinterface = CompCor(
80+
variance_threshold=0.7,
81+
realigned_file=self.realigned_file,
82+
mask_files=self.mask_files,
83+
mask_names=['mask'],
84+
mask_index=1,
85+
save_metadata=True)
86+
self.run_cc(ccinterface=ccinterface,
87+
expected_components=expected_components,
88+
expected_n_components=2,
89+
expected_metadata=expected_metadata)
90+
6391
def test_tcompcor(self):
64-
ccinterface = TCompCor(
92+
ccinterface = TCompCor(num_components=6,
6593
realigned_file=self.realigned_file, percentile_threshold=0.75)
6694
self.run_cc(ccinterface, [['-0.1114536190', '-0.4632908609'], [
6795
'0.4566907310', '0.6983205193'
@@ -70,7 +98,8 @@ def test_tcompcor(self):
7098
], ['-0.1342351356', '0.1407855119']], 'tCompCor')
7199

72100
def test_tcompcor_no_percentile(self):
73-
ccinterface = TCompCor(realigned_file=self.realigned_file)
101+
ccinterface = TCompCor(num_components=6,
102+
realigned_file=self.realigned_file)
74103
ccinterface.run()
75104

76105
mask = nb.load('mask_000.nii.gz').get_data()
@@ -80,6 +109,7 @@ def test_tcompcor_no_percentile(self):
80109
def test_compcor_no_regress_poly(self):
81110
self.run_cc(
82111
CompCor(
112+
num_components=6,
83113
realigned_file=self.realigned_file,
84114
mask_files=self.mask_files,
85115
mask_index=0,
@@ -151,7 +181,9 @@ def test_tcompcor_multi_mask_no_index(self):
151181
def run_cc(self,
152182
ccinterface,
153183
expected_components,
154-
expected_header='CompCor'):
184+
expected_header='CompCor',
185+
expected_n_components=None,
186+
expected_metadata=None):
155187
# run
156188
ccresult = ccinterface.run()
157189

@@ -160,13 +192,14 @@ def run_cc(self,
160192
assert ccresult.outputs.components_file == expected_file
161193
assert os.path.exists(expected_file)
162194
assert os.path.getsize(expected_file) > 0
163-
assert ccinterface.inputs.num_components == 6
164195

165196
with open(ccresult.outputs.components_file, 'r') as components_file:
166-
expected_n_components = min(ccinterface.inputs.num_components,
167-
self.fake_data.shape[3])
197+
if expected_n_components is None:
198+
expected_n_components = min(ccinterface.inputs.num_components,
199+
self.fake_data.shape[3])
168200

169-
components_data = [line.split('\t') for line in components_file]
201+
components_data = [line.rstrip().split('\t')
202+
for line in components_file]
170203

171204
# the first item will be '#', we can throw it out
172205
header = components_data.pop(0)
@@ -180,9 +213,24 @@ def run_cc(self,
180213
num_got_timepoints = len(components_data)
181214
assert num_got_timepoints == self.fake_data.shape[3]
182215
for index, timepoint in enumerate(components_data):
183-
assert (len(timepoint) == ccinterface.inputs.num_components
184-
or len(timepoint) == self.fake_data.shape[3])
216+
assert (len(timepoint) == expected_n_components)
185217
assert timepoint[:2] == expected_components[index]
218+
219+
if ccinterface.inputs.save_metadata:
220+
expected_metadata_file = (
221+
ccinterface._list_outputs()['metadata_file'])
222+
assert ccresult.outputs.metadata_file == expected_metadata_file
223+
assert os.path.exists(expected_metadata_file)
224+
assert os.path.getsize(expected_metadata_file) > 0
225+
226+
with open(ccresult.outputs.metadata_file, 'r') as metadata_file:
227+
components_metadata = [line.rstrip().split('\t')
228+
for line in metadata_file]
229+
components_metadata = {i: j for i, j in
230+
zip(components_metadata[0],
231+
components_metadata[1])}
232+
assert components_metadata == expected_metadata
233+
186234
return ccresult
187235

188236
@staticmethod

nipype/algorithms/tests/test_auto_ACompCor.py

+5-1
Original file line numberDiff line numberDiff line change
@@ -15,20 +15,23 @@ def test_ACompCor_inputs():
1515
requires=['mask_files'],
1616
xor=['merge_method'],
1717
),
18+
mask_names=dict(),
1819
merge_method=dict(
1920
requires=['mask_files'],
2021
xor=['mask_index'],
2122
),
22-
num_components=dict(usedefault=True, ),
23+
num_components=dict(xor=['variance_threshold'], ),
2324
pre_filter=dict(usedefault=True, ),
2425
realigned_file=dict(mandatory=True, ),
2526
regress_poly_degree=dict(usedefault=True, ),
2627
repetition_time=dict(),
28+
save_metadata=dict(),
2729
save_pre_filter=dict(),
2830
use_regress_poly=dict(
2931
deprecated='0.15.0',
3032
new_name='pre_filter',
3133
),
34+
variance_threshold=dict(xor=['num_components'], ),
3235
)
3336
inputs = ACompCor.input_spec()
3437

@@ -38,6 +41,7 @@ def test_ACompCor_inputs():
3841
def test_ACompCor_outputs():
3942
output_map = dict(
4043
components_file=dict(),
44+
metadata_file=dict(),
4145
pre_filter_file=dict(),
4246
)
4347
outputs = ACompCor.output_spec()

nipype/algorithms/tests/test_auto_TCompCor.py

+5-1
Original file line numberDiff line numberDiff line change
@@ -15,21 +15,24 @@ def test_TCompCor_inputs():
1515
requires=['mask_files'],
1616
xor=['merge_method'],
1717
),
18+
mask_names=dict(),
1819
merge_method=dict(
1920
requires=['mask_files'],
2021
xor=['mask_index'],
2122
),
22-
num_components=dict(usedefault=True, ),
23+
num_components=dict(xor=['variance_threshold'], ),
2324
percentile_threshold=dict(usedefault=True, ),
2425
pre_filter=dict(usedefault=True, ),
2526
realigned_file=dict(mandatory=True, ),
2627
regress_poly_degree=dict(usedefault=True, ),
2728
repetition_time=dict(),
29+
save_metadata=dict(),
2830
save_pre_filter=dict(),
2931
use_regress_poly=dict(
3032
deprecated='0.15.0',
3133
new_name='pre_filter',
3234
),
35+
variance_threshold=dict(xor=['num_components'], ),
3336
)
3437
inputs = TCompCor.input_spec()
3538

@@ -40,6 +43,7 @@ def test_TCompCor_outputs():
4043
output_map = dict(
4144
components_file=dict(),
4245
high_variance_masks=dict(),
46+
metadata_file=dict(),
4347
pre_filter_file=dict(),
4448
)
4549
outputs = TCompCor.output_spec()

nipype/info.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -141,7 +141,7 @@ def get_nipype_gitversion():
141141
'numpy>=%s ; python_version >= "3.7"' % NUMPY_MIN_VERSION_37,
142142
'python-dateutil>=%s' % DATEUTIL_MIN_VERSION,
143143
'scipy>=%s' % SCIPY_MIN_VERSION,
144-
'traits>=%s' % TRAITS_MIN_VERSION,
144+
'traits>=%s,!=5.0' % TRAITS_MIN_VERSION,
145145
'future>=%s' % FUTURE_MIN_VERSION,
146146
'simplejson>=%s' % SIMPLEJSON_MIN_VERSION,
147147
'prov>=%s' % PROV_VERSION,

nipype/workflows/rsfmri/fsl/resting.py

-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@
33
# vi: set ft=python sts=4 ts=4 sw=4 et:
44
from __future__ import (print_function, division, unicode_literals,
55
absolute_import)
6-
from builtins import str
76

87
from ....interfaces import fsl as fsl # fsl
98
from ....interfaces import utility as util # utility

nipype/workflows/rsfmri/fsl/tests/test_resting.py

+11-10
Original file line numberDiff line numberDiff line change
@@ -89,16 +89,17 @@ def test_create_resting_preproc(self, mock_node, mock_realign_wf):
8989
# assert
9090
expected_file = os.path.abspath(self.out_filenames['components_file'])
9191
with open(expected_file, 'r') as components_file:
92-
components_data = [line.split() for line in components_file]
93-
num_got_components = len(components_data)
94-
assert (num_got_components == self.num_noise_components
95-
or num_got_components == self.fake_data.shape[3])
96-
first_two = [row[:2] for row in components_data[1:]]
97-
assert first_two == [['-0.5172356654', '-0.6973053243'], [
98-
'0.2574722644', '0.1645270737'
99-
], ['-0.0806469590',
100-
'0.5156853779'], ['0.7187176051', '-0.3235820287'],
101-
['-0.3783072450', '0.3406749013']]
92+
components_data = [line.rstrip().split()
93+
for line in components_file]
94+
num_got_components = len(components_data)
95+
assert (num_got_components == self.num_noise_components or
96+
num_got_components == self.fake_data.shape[3])
97+
first_two = [row[:2] for row in components_data[1:]]
98+
assert first_two == [['-0.5172356654', '-0.6973053243'],
99+
['0.2574722644', '0.1645270737'],
100+
['-0.0806469590', '0.5156853779'],
101+
['0.7187176051', '-0.3235820287'],
102+
['-0.3783072450', '0.3406749013']]
102103

103104
fake_data = np.array([[[[2, 4, 3, 9, 1], [3, 6, 4, 7, 4]],
104105
[[8, 3, 4, 6, 2], [4, 0, 4, 4, 2]]],

0 commit comments

Comments
 (0)