Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[WIP] 0.7 release #247

Merged
merged 64 commits into from
Oct 6, 2018
Merged
Show file tree
Hide file tree
Changes from 35 commits
Commits
Show all changes
64 commits
Select commit Hold shift + click to select a range
23e70c3
rename entities
tyarkoni Aug 24, 2018
b87a84b
update to reflect entity changes
tyarkoni Aug 25, 2018
84c21f4
fix reports module to reflect entity changes
tyarkoni Aug 25, 2018
cd4c49e
drop grabbids and shorten layout module names
tyarkoni Aug 25, 2018
0723bea
validated files are strings not Files; fixes #222
tyarkoni Aug 25, 2018
a7278f7
return None when merge=True and there's nothing in the list; fixes #202
tyarkoni Aug 25, 2018
f2663cd
add new BIDSFile class; closes #242
tyarkoni Aug 25, 2018
b949bbe
working metadata indexing implementation; closes #243
tyarkoni Aug 26, 2018
14cae64
metadata indexing tests
tyarkoni Aug 26, 2018
40d9ef7
add ability to constrain metadata search to specified file list
tyarkoni Aug 26, 2018
0587658
avoid explicitly passing maxsplit arg--it fails on Python 2
tyarkoni Aug 26, 2018
52110be
force explicit index initialization before metadata search works
tyarkoni Aug 26, 2018
d64f127
exclusions in bids config
tyarkoni Aug 30, 2018
f831297
set validation to True by default; closes #244
tyarkoni Sep 8, 2018
7fe1b61
remove exclusions from config
tyarkoni Sep 11, 2018
a394e73
changes indexing behavior to exclude derivs, code, etc.; ##closes #184
tyarkoni Sep 11, 2018
77ec845
update tests to reflect changes and give dataset fixtures more sensib…
tyarkoni Sep 11, 2018
8e158a1
temporarily store validation rules internally
tyarkoni Sep 11, 2018
4f6067f
disable breaking validation tests
tyarkoni Sep 11, 2018
ca5c697
set validate to True by default
tyarkoni Sep 11, 2018
2878326
turn auto_contrasts into a list
tyarkoni Sep 11, 2018
30662db
update tests
tyarkoni Sep 11, 2018
4dd29f9
allow incremental variable reading and drop _preproc suffix to sync w…
tyarkoni Sep 11, 2018
fe8044e
drop loop_preproc config option
tyarkoni Sep 11, 2018
856e6b0
update auto_contrast handling to reflect spec; closes #234
tyarkoni Sep 12, 2018
750d356
correct analysis tests
tyarkoni Sep 12, 2018
7d61713
temporarily handle preproc properly until derivatives release candida…
tyarkoni Sep 12, 2018
ad7f142
make 'bids' config a dependency of 'derivatives'; closes #246
tyarkoni Sep 12, 2018
571ed2f
remove constraint on passed selectors to _load_time_variables
tyarkoni Sep 12, 2018
8326f39
add explicit space constraint to derivatives dataset
tyarkoni Sep 12, 2018
ae0a583
pin grabbit version
tyarkoni Sep 12, 2018
1a0673e
added path patterns and sample path-building test
tyarkoni Sep 12, 2018
91bb60c
add missing docstrings to metadata indexing methods
tyarkoni Sep 12, 2018
f8557a3
drop unneeded auto_contrasts arg in Block setup
tyarkoni Sep 12, 2018
cf64ae6
drop acq in favor of only acquisitions
tyarkoni Sep 12, 2018
a7f2a72
fix tests; acq --> acquisition
tyarkoni Sep 12, 2018
fe4f6f2
add period before extensions or NPC will get very angry
tyarkoni Sep 12, 2018
6ad17e9
bundle HRF convolution functionality from nistats
bthirion Sep 13, 2018
59d6967
adds convolve_HRF transformation; closes #250; closes #251
tyarkoni Sep 13, 2018
eeb4cc4
updated Changelog in anticipation of 0.7
tyarkoni Sep 14, 2018
d04ea36
add test for metadata indexing at Layout init
tyarkoni Sep 14, 2018
ac3cd55
remove parse_entities in favor of grabbit's parse_file_entities
tyarkoni Sep 14, 2018
0dc5bdb
add test for BIDSLayout.__repr__
tyarkoni Sep 14, 2018
d4a50c8
fix sum transformation and add test
tyarkoni Sep 14, 2018
592aa25
update changelog
tyarkoni Sep 14, 2018
bbfeef7
support F contrasts and improves ContrastInfo tests; closes #249
tyarkoni Sep 14, 2018
2e7e83a
add forgotten path-building test
tyarkoni Sep 14, 2018
968d5b9
add F-contrast test
tyarkoni Sep 14, 2018
df4172c
fix broken test
tyarkoni Sep 14, 2018
1d0c34b
reduce platform-dependence in paths
tyarkoni Sep 14, 2018
f0f084e
add to_df alias for to_data_frame and fix docstring
tyarkoni Sep 14, 2018
52d1cbb
refactor metadata indexing
tyarkoni Oct 5, 2018
e8e0d2c
always use bold as suffix now that derivatives RC is out
tyarkoni Oct 5, 2018
2e003e9
fix metadata indexing
tyarkoni Oct 5, 2018
1dec748
fix metadata indexing tests
tyarkoni Oct 5, 2018
15c0ba4
update synthetic dataset derivatives to match RC1
tyarkoni Oct 5, 2018
f3853fa
simplified initialization interface
tyarkoni Oct 6, 2018
9bc7b74
update tests
tyarkoni Oct 6, 2018
34b8b00
first attempt at fixing bug reported in #258
tyarkoni Oct 6, 2018
0cd8f0e
add 2nd deriv directory to ds005
tyarkoni Oct 6, 2018
29f42dc
variant --> desc
tyarkoni Oct 6, 2018
f3d40e1
add test for layout with multiple deriv dirs
tyarkoni Oct 6, 2018
861eb5a
move ds005 derivs outside of ds005 for better test
tyarkoni Oct 6, 2018
acda6fb
update derivatives entities to reflect RC1
tyarkoni Oct 6, 2018
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
63 changes: 31 additions & 32 deletions bids/analysis/analysis.py
Original file line number Diff line number Diff line change
Expand Up @@ -51,8 +51,7 @@ def _load_blocks(self, blocks):
block = Block(self.layout, index=i, **block_args)
self.blocks.append(block)

def setup(self, blocks=None, agg_func='mean', auto_contrasts=None,
**kwargs):
def setup(self, blocks=None, agg_func='mean', **kwargs):
''' Set up the sequence of blocks for analysis.

Args:
Expand All @@ -69,8 +68,6 @@ def setup(self, blocks=None, agg_func='mean', auto_contrasts=None,
name of a function recognized by apply() in pandas, or a
Callable that takes a DataFrame as input and returns a Series
or a DataFrame. NOTE: CURRENTLY UNIMPLEMENTED.
auto_contrasts (bool): If True, a contrast is automatically
created for each column in the design matrix.
'''

# In the beginning, there was nothing
Expand All @@ -86,7 +83,7 @@ def setup(self, blocks=None, agg_func='mean', auto_contrasts=None,
if blocks is not None and i not in blocks and b.name not in blocks:
continue

b.setup(input_nodes, auto_contrasts, **selectors)
b.setup(input_nodes, **selectors)
input_nodes = b.output_nodes


Expand All @@ -108,14 +105,16 @@ class Block(object):
generated when the model is fit.
input_nodes (list): Optional list of AnalysisNodes to use as input to
this Block (typically, the output from the preceding Block).
auto_contrasts (bool): If True, a contrast is automatically
created for each column in the design matrix. This parameter is
over-written by the setting in setup(). Default is True.
auto_contrasts (list, bool): Optional list of variable names to create
an indicator contrast for. Alternatively, if the boolean value True
is passed, a contrast is automatically generated for _all_
available variables. This parameter is over-written by the setting
in setup() if the latter is passed.
'''

def __init__(self, layout, level, index, name=None, transformations=None,
model=None, contrasts=None, input_nodes=None,
auto_contrasts=True):
auto_contrasts=False):

self.layout = layout
self.level = level
Expand All @@ -125,8 +124,8 @@ def __init__(self, layout, level, index, name=None, transformations=None,
self.model = model or None
self.contrasts = contrasts or []
self.input_nodes = input_nodes or []
self.auto_contrasts = auto_contrasts or []
self.output_nodes = []
self.auto_contrasts = auto_contrasts

def _filter_objects(self, objects, kwargs):
# Keeps only objects that match target entities, and also removes those
Expand Down Expand Up @@ -163,23 +162,18 @@ def _concatenate_input_nodes(self, nodes):
entities = pd.concat(entities, axis=1).T
return BIDSVariableCollection.from_df(data, entities, self.level)

def setup(self, input_nodes=None, auto_contrasts=None, **kwargs):
def setup(self, input_nodes=None, **kwargs):
''' Set up the Block and construct the design matrix.

Args:
input_nodes (list): Optional list of Node objects produced by
the preceding Block in the analysis. If None, uses any inputs
passed in at Block initialization.
auto_contrasts (bool): If True, a contrast is automatically
created for each column in the design matrix.
kwargs: Optional keyword arguments to pass onto load_variables.
'''
self.output_nodes = []
input_nodes = input_nodes or self.input_nodes or []

if auto_contrasts is not None:
self.auto_contrasts = auto_contrasts

# TODO: remove the scan_length argument entirely once we switch tests
# to use the synthetic dataset with image headers.
if self.level != 'run':
Expand All @@ -192,6 +186,14 @@ def setup(self, input_nodes=None, auto_contrasts=None, **kwargs):
objects, kwargs = self._filter_objects(objects, kwargs)
groups = self._group_objects(objects)

# Set up and validate variable lists
model = self.model or {}
variables = model.get('variables', [])
hrf_variables = model.get('HRF_variables', [])
if not set(variables) >= set(hrf_variables):
raise ValueError("HRF_variables must be a subset ",
"of variables in BIDS model.")

for grp in groups:
# Split into separate lists of Collections and Nodes
input_nodes = [o for o in grp if isinstance(o, AnalysisNode)]
Expand All @@ -201,19 +203,11 @@ def setup(self, input_nodes=None, auto_contrasts=None, **kwargs):
node_coll = self._concatenate_input_nodes(input_nodes)
colls.append(node_coll)

model = self.model or {}

variables = set(model.get('variables', []))
hrf_variables = set(model.get('HRF_variables', []))
if not variables >= hrf_variables:
raise ValueError("HRF_variables must be a subset ",
"of variables in BIDS model.")

coll = merge_collections(colls) if len(colls) > 1 else colls[0]

coll = apply_transformations(coll, self.transformations)
if model.get('variables'):
transform.select(coll, model['variables'])
if variables:
transform.select(coll, variables)

node = AnalysisNode(self.level, coll, self.contrasts, input_nodes,
self.auto_contrasts)
Expand Down Expand Up @@ -294,17 +288,21 @@ class AnalysisNode(object):
collection (BIDSVariableCollection): The BIDSVariableCollection
containing variables at this Node.
contrasts (list): A list of contrasts defined in the originating Block.
auto_contrasts (bool): If True, a contrast is automatically
created for each column in the design matrix.
auto_contrasts (list): Optional list of variable names to create
an indicator contrast for. Alternatively, if the boolean value True
is passed, a contrast is automatically generated for _all_
available variables.
'''

def __init__(self, level, collection, contrasts, input_nodes=None,
auto_contrasts=True):
auto_contrasts=None):
self.level = level
self.collection = collection
self._block_contrasts = contrasts
self.input_nodes = input_nodes
self.auto_contrasts = auto_contrasts
if auto_contrasts == True:
auto_contrasts = collection.variables.keys()
self.auto_contrasts = auto_contrasts or []
self._contrasts = None

@property
Expand Down Expand Up @@ -403,8 +401,9 @@ def get_contrasts(self, names=None, entities=False):

### Add ability to set different types of identity contrasts
if self.auto_contrasts:
for col_name in self.collection.variables.keys():
if col_name not in contrast_names:
for col_name in self.auto_contrasts:
if (col_name in self.collection.variables.keys()
and col_name not in contrast_names):
contrasts.append({
'name': col_name,
'condition_list': [col_name],
Expand Down
18 changes: 3 additions & 15 deletions bids/analysis/tests/test_analysis.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,20 +9,12 @@
@pytest.fixture
def analysis():
layout_path = join(get_test_data_path(), 'ds005')
layout = BIDSLayout(layout_path, exclude='derivatives/')
layout = BIDSLayout(layout_path)
json_file = join(layout_path, 'models', 'ds-005_type-test_model.json')
analysis = Analysis(layout, json_file)
analysis.setup(scan_length=480, subject=['01', '02'])
return analysis

@pytest.fixture
def analysis_force_auto_contrasts():
layout_path = join(get_test_data_path(), 'ds005')
layout = BIDSLayout(layout_path, exclude='derivatives/')
json_file = join(layout_path, 'models', 'ds-005_type-test_model.json')
analysis = Analysis(layout, json_file)
analysis.setup(scan_length=480, subject=['01', '02'], auto_contrasts=True)
return analysis

def test_design_matrix_info(analysis):
result = analysis['run'].get_design_matrix(subject=['01', '02', '03'])
Expand Down Expand Up @@ -81,7 +73,7 @@ def test_post_first_level_sparse_design_matrix(analysis):
result = analysis['session'].get_design_matrix(entities=False)
assert len(result) == 2
assert len(result[0]) == 3
assert result[0].sparse.shape == (3, 2)
assert result[0].sparse.shape == (6, 2)
assert result[0].entities == {
'subject': '01',
'task': 'mixedgamblestask'}
Expand All @@ -95,7 +87,7 @@ def test_post_first_level_sparse_design_matrix(analysis):
result = analysis['group'].get_design_matrix()
assert len(result) == 1
data = result[0].sparse
assert len(data) == 6
assert len(data) == 8
assert data['subject'].nunique() == 2

# # Make sure columns from different levels exist
Expand All @@ -114,10 +106,6 @@ def test_contrast_matrix_info(analysis):
assert isinstance(contrasts[0], ContrastMatrixInfo)
assert c._fields == ('data', 'index', 'entities')

def test_force_auto_contrasts(analysis_force_auto_contrasts):
contrasts = analysis_force_auto_contrasts['run'].get_contrasts(subject='01')
assert contrasts[0][0].shape == (8, 9)


# def test_get_contrasts(analysis):
# contrasts = analysis['run'].get_contrasts(subject='01')
2 changes: 1 addition & 1 deletion bids/analysis/tests/test_transformations.py
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@
@pytest.fixture
def collection():
layout_path = join(get_test_data_path(), 'ds005')
layout = BIDSLayout(layout_path, exclude='derivatives/')
layout = BIDSLayout(layout_path)
collection = layout.get_collections('run', types=['events'],
scan_length=480, merge=True,
sampling_rate=10)
Expand Down
4 changes: 1 addition & 3 deletions bids/config.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,9 +10,7 @@

_config_name = 'pybids_config.json'

_default_settings = {
'loop_preproc': False
}
_default_settings = {}


def set_option(key, value):
Expand Down
6 changes: 0 additions & 6 deletions bids/grabbids/__init__.py

This file was deleted.

4 changes: 2 additions & 2 deletions bids/layout/__init__.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,3 @@
from .bids_layout import BIDSLayout
from .bids_validator import BIDSValidator
from .layout import BIDSLayout
from .validation import BIDSValidator
__all__ = ["BIDSLayout", "BIDSValidator"]
Loading