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

[ENH] Update config to support microscopy, qMRI, PET, ASL #840

Merged
merged 36 commits into from
Apr 24, 2022
Merged
Show file tree
Hide file tree
Changes from 29 commits
Commits
Show all changes
36 commits
Select commit Hold shift + click to select a range
282f287
add Pipfile to gitingore
Remi-Gau Apr 14, 2022
a469300
add missing datatypes to config
Remi-Gau Apr 14, 2022
a13adac
add microscopy entities to config
Remi-Gau Apr 14, 2022
dd96e71
add draft path patterns microscopy and qMRI
Remi-Gau Apr 14, 2022
a035f16
clean up and refactor
Remi-Gau Apr 14, 2022
4ed8753
add extension micr
Remi-Gau Apr 14, 2022
d6c1cb8
add suffixes micr
Remi-Gau Apr 14, 2022
fcb2e8f
add photo files for micr
Remi-Gau Apr 14, 2022
899bc81
add parametric parts of qMRI
Remi-Gau Apr 14, 2022
3d27292
add PET tracer entity to config
Remi-Gau Apr 14, 2022
7417813
add path patterns for PET
Remi-Gau Apr 14, 2022
0167134
add path patterns for ASL
Remi-Gau Apr 14, 2022
6a95de5
refactor some patterns
Remi-Gau Apr 14, 2022
b48f7a5
add patterns for json files in root of dataset
Remi-Gau Apr 14, 2022
4044359
update bids nodot config and use {suffix<>} everywhere
Remi-Gau Apr 14, 2022
ab039b6
add tracer entity to no dot config
Remi-Gau Apr 14, 2022
b361f21
fix entity order in filename pattern for no dot config
Remi-Gau Apr 14, 2022
8aff655
fix typo
Remi-Gau Apr 15, 2022
f44cdd7
remove problematic path pattern
Remi-Gau Apr 15, 2022
5d96b0e
readd sample entity to micr filename pattern in root of dataset
Remi-Gau Apr 15, 2022
de1e61f
sample is actually required for micr
Remi-Gau Apr 15, 2022
5b5e805
update test layout on bids-example
Remi-Gau Apr 15, 2022
8a470f9
Update bids/layout/config/bids.json
Remi-Gau Apr 15, 2022
ec69fbd
Update bids/layout/config/bids.json
Remi-Gau Apr 15, 2022
debedda
fix separator position for session
Remi-Gau Apr 15, 2022
c4dc192
Merge branch 'master' of https://github.com/bids-standard/pybids into…
Remi-Gau Apr 16, 2022
c0aea8b
add more bids example layout tests and flag MPM as xfail
Remi-Gau Apr 18, 2022
196cb0f
flag qmri_qsm as failure expected too
Remi-Gau Apr 18, 2022
bf95917
make xfail strict to detect tests when they start passing
Remi-Gau Apr 19, 2022
d3539a8
Update bids/layout/config/bids.json
Remi-Gau Apr 19, 2022
b83b2b5
Update bids/layout/config/bids.json
Remi-Gau Apr 19, 2022
abfc9b2
fix more typos
Remi-Gau Apr 19, 2022
2360b82
Merge branch 'master' of https://github.com/bids-standard/pybids into…
Remi-Gau Apr 24, 2022
173c2a1
use bids-example submodule and fixture for testing
Remi-Gau Apr 24, 2022
4c5e504
remove makefile recipe for bids-examles
Remi-Gau Apr 24, 2022
669200a
MNT: Restore makefile
effigies Apr 24, 2022
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
7 changes: 6 additions & 1 deletion .gitignore
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
historical/

#
# clone of bids-example repo for testing
bids/tests/data/bids-examples

# Byte-compiled / optimized / DLL files
__pycache__/
*.py[cod]
Expand Down Expand Up @@ -86,6 +88,9 @@ celerybeat-schedule
venv/
ENV/

# Pipenv
Pipfile

# Spyder project settings
.spyderproject

Expand Down
5 changes: 4 additions & 1 deletion Makefile
Original file line number Diff line number Diff line change
@@ -1,10 +1,13 @@
.PHONY: doc tutorial travis_tests

ci_tests:
ci_tests: bids/tests/data/bids-examples
Remi-Gau marked this conversation as resolved.
Show resolved Hide resolved
pytest --doctest-modules -n 2 -v --cov bids --cov-config .coveragerc --cov-report xml:cov.xml bids

tutorial:
jupyter nbconvert --execute examples/pybids_tutorial.ipynb --to html

doc:
$(MAKE) -C doc html

bids/tests/data/bids-examples:
git clone https://github.com/bids-standard/bids-examples.git --depth 1 bids/tests/data/bids-examples
56 changes: 52 additions & 4 deletions bids/layout/config/bids.json
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,10 @@
"mandatory": false,
"directory": "{subject}{session}"
},
{
"name": "sample",
"pattern": "[_/\\\\]+sample-([a-zA-Z0-9]+)"
},
{
"name": "task",
"pattern": "[_/\\\\]+task-([a-zA-Z0-9]+)"
Expand All @@ -24,6 +28,14 @@
"name": "ceagent",
"pattern": "[_/\\\\]+ce-([a-zA-Z0-9]+)"
},
{
"name": "staining",
"pattern": "[_/\\\\]+stain-([a-zA-Z0-9]+)"
},
{
"name": "tracer",
"pattern": "[_/\\\\]+stain-([a-zA-Z0-9]+)"
Remi-Gau marked this conversation as resolved.
Show resolved Hide resolved
},
{
"name": "reconstruction",
"pattern": "[_/\\\\]+rec-([a-zA-Z0-9]+)"
Expand Down Expand Up @@ -73,6 +85,10 @@
"name": "space",
"pattern": "[_/\\\\]+space-([a-zA-Z0-9]+)"
},
{
"name": "chunk",
"pattern": "[_/\\\\]+chunk-([0-9]+)"
},
{
"name": "suffix",
"pattern": "[._]*([a-zA-Z0-9]*?)\\.[^/\\\\]+$"
Expand All @@ -87,7 +103,7 @@
},
{
"name": "datatype",
"pattern": "[/\\\\]+(anat|beh|dwi|eeg|fmap|func|ieeg|meg|perf)[/\\\\]+"
"pattern": "[/\\\\]+(anat|beh|dwi|eeg|fmap|func|ieeg|meg|micr|perf|pet)[/\\\\]+"
},
{
"name": "extension",
Expand All @@ -98,8 +114,8 @@
"default_path_patterns": [
"sub-{subject}[/ses-{session}]/{datatype<anat>|anat}/sub-{subject}[_ses-{session}][_acq-{acquisition}][_ce-{ceagent}][_rec-{reconstruction}][_part-{part}]_{suffix<T1w|T2w|T1rho|T1map|T2map|T2star|FLAIR|FLASH|PDmap|PD|PDT2|inplaneT[12]|angio>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
"sub-{subject}[/ses-{session}]/{datatype<anat>|anat}/sub-{subject}[_ses-{session}][_acq-{acquisition}][_ce-{ceagent}][_rec-{reconstruction}][_mod-{modality}]_{suffix<defacemask>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
"sub-{subject}[/ses-{session}]/{datatype<func>|func}/sub-{subject}[_ses-{session}]_task-{task}[_acq-{acquisition}][_ce-{ceagent}][_dir-{direction}][_rec-{reconstruction}][_run-{run}][_echo-{echo}][_part-{part}]_{suffix<bold|cbv|sbref>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
"sub-{subject}[/ses-{session}]/{datatype<func>|func}/sub-{subject}[_ses-{session}]_task-{task}[_acq-{acquisition}][_ce-{ceagent}][_dir-{direction}][_rec-{reconstruction}][_run-{run}][_echo-{echo}]_{suffix<phase>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
"sub-{subject}[/ses-{session}]/{datatype<func>|func}/sub-{subject}[_ses-{session}]_task-{task}[_acq-{acquisition}][_ce-{ceagent}][_rec-{reconstruction}][_dir-{direction}][_run-{run}][_echo-{echo}][_part-{part}]_{suffix<bold|cbv|sbref>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
"sub-{subject}[/ses-{session}]/{datatype<func>|func}/sub-{subject}[_ses-{session}]_task-{task}[_acq-{acquisition}][_ce-{ceagent}][_rec-{reconstruction}][_dir-{direction}][_run-{run}][_echo-{echo}]_{suffix<phase>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
effigies marked this conversation as resolved.
Show resolved Hide resolved
"sub-{subject}[/ses-{session}]/{datatype<dwi>|dwi}/sub-{subject}[_ses-{session}][_acq-{acquisition}][_part-{part}]_{suffix<dwi>}{extension<.bval|.bvec|.json|.nii.gz|.nii>|.nii.gz}",
"sub-{subject}[/ses-{session}]/{datatype<fmap>|fmap}/sub-{subject}[_ses-{session}][_acq-{acquisition}][_dir-{direction}][_run-{run}]_{fmap<phasediff|magnitude[12]|phase[12]|fieldmap>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
"sub-{subject}[/ses-{session}]/{datatype<fmap>|fmap}/sub-{subject}[_ses-{session}][_acq-{acquisition}][_ce-{ceagent}]_dir-{direction}[_run-{run}]_{fmap<epi>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
Expand All @@ -109,13 +125,45 @@
"sub-{subject}[/ses-{session}]/{datatype<meg>|meg}/sub-{subject}[_ses-{session}]_task-{task}[_acq-{acquisition}][_run-{run}][_proc-{proc}]_{suffix<channels>}{extension<.tsv|.json>|.tsv}",
"sub-{subject}[/ses-{session}]/{datatype<meg>|meg}/sub-{subject}[_ses-{session}]_task-{task}[_acq-{acquisition}]_{suffix<coordsystem>}{extension<.json>|.json}",
"sub-{subject}[/ses-{session}]/{datatype<meg>|meg}/sub-{subject}[_ses-{session}]_task-{task}[_acq-{acquisition}]_{suffix<photo>}{extension<.jpg>|.jpg}",
"sub-{subject}[/ses-{session}]/{datatype<micr>|micr}/sub-{subject}[_ses-{session}]_sample-{sample}[_acq-{acquisition}][_stain-{label}][_run-{run}][_chunk-{index}]_{suffix<TEM|SEM|uCT|BF|DF|PC|DIC|FLUO|CONF|PLI|CARS|2PE|MPE|SR|NLO|OCT|SPIM>}{extension<.png|.tif|.ome.tif|.ome.btf|.json>}",
Remi-Gau marked this conversation as resolved.
Show resolved Hide resolved
"sub-{subject}[/ses-{session}]/{datatype<micr>|micr}/sub-{subject}[_ses-{session}]_sample-{sample}[_acq-{acquisition}]_{suffix<photo>|photo}{extension<.jpg|.png|.tif|.json>}",
Remi-Gau marked this conversation as resolved.
Show resolved Hide resolved
"sub-{subject}[/ses-{session}]/{datatype<anat>|anat}|anat/sub-{subject}[_ses-{session}][_acq-{acquisition}][_ce-{ceagent}][_rec-{label}][_run-{run}]_echo-{echo}[_part-{<mag|phase|real|imag>}]_{suffix<MEGRE|MESE>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
"sub-{subject}[/ses-{session}]/{datatype<anat>|anat}|anat/sub-{subject}[_ses-{session}][_acq-{acquisition}][_ce-{ceagent}][_rec-{label}][_run-{run}][_echo-{echo}]_flip-{flip}[_part-{<mag|phase|real|imag>}]_{suffix<VFA>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
"sub-{subject}[/ses-{session}]/{datatype<anat>|anat}|anat/sub-{subject}[_ses-{session}][_acq-{acquisition}][_ce-{ceagent}][_rec-{label}][_run-{run}]_inv-{inv}[_part-{<mag|phase|real|imag>}]_{suffix<IRT1>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
"sub-{subject}[/ses-{session}]/{datatype<anat>|anat}|anat/sub-{subject}[_ses-{session}][_acq-{acquisition}][_ce-{ceagent}][_rec-{label}][_run-{run}][_echo-{echo}][_flip-{flip}]_inv-{inv}[_part-{<mag|phase|real|imag>}]_{suffix<MP2RAGE>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
"sub-{subject}[/ses-{session}]/{datatype<anat>|anat}|anat/sub-{subject}[_ses-{session}][_acq-{acquisition}][_ce-{ceagent}][_rec-{label}][_run-{run}][_echo-{echo}]_flip-{flip}_mt-{on|off}[_part-{<mag|phase|real|imag>}]_{suffix<MPM|MTS>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
"sub-{subject}[/ses-{session}]/{datatype<anat>|anat}|anat/sub-{subject}[_ses-{session}][_acq-{acquisition}][_ce-{ceagent}][_rec-{label}][_run-{run}]_mt-{on|off}[_part-{<mag|phase|real|imag>}]_{suffix<MTR>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
"sub-{subject}[/ses-{session}]/{datatype<anat>|anat}|anat/sub-{subject}[_ses-{session}][_acq-{acquisition}][_ce-{ceagent}][_rec-{label}][_run-{run}]_{suffix<T1map|T2map|T2starmap|R1map|R2map|R2starmap|PDmap|MTRmap|MTsat|UNIT1|T1rho|MWFmap|MTVmap|PDT2map|Chimap|S0map|M0map>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
"sub-{subject}[/ses-{session}]/{datatype<pet>|pet}/sub-{subject}[_ses-{session}][_task-{task}][trc-{tracer}][_rec-{reconstruction}][_run-{run}]_{suffix<pet>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
"sub-{subject}[/ses-{session}]/{datatype<pet>|pet}/sub-{subject}[_ses-{session}][_task-{task}][trc-{tracer}][_rec-{reconstruction}][_run-{run}]_recording-{recording}_{suffix<blood>}{extension<.tsv|.json>}",
"sub-{subject}[/ses-{session}]/{datatype<pet>|pet}/sub-{subject}[_ses-{session}]_task-{task}[trc-{tracer}][_rec-{reconstruction}][_run-{run}]_{suffix<events>}{extension<.tsv|.json>}",
"sub-{subject}[/ses-{session}]/{datatype<pet>|pet}/sub-{subject}[_ses-{session}][_task-{task}][trc-{tracer}][_rec-{reconstruction}][_run-{run}][_recording-{recording}]_{suffix<physio|stim>}{extension<.tsv.gz|.json>}",
"sub-{subject}[/ses-{session}]/{datatype<perf>|perf}/sub-{subject}[_ses-{session}][_acq-{acquisition}][_rec-{reconstruction}][_dir{direction}][_run-{run}]_{suffix<asl|m0scan>}{extension<.nii|.nii.gz|.json>|.nii.gz}",
"sub-{subject}[/ses-{session}]/{datatype<perf>|perf}/sub-{subject}[_ses-{session}][_acq-{acquisition}][_rec-{reconstruction}][_dir{direction}][_run-{run}]_{suffix<aslcontext>}{extension<.tsv|.json>}",
"sub-{subject}[/ses-{session}]/{datatype<perf>|perf}/sub-{subject}[_ses-{session}][_acq-{acquisition}][_rec-{reconstruction}][_run-{run}]_{suffix<asllabeling>}{extension<.jpg>}",
"sub-{subject}[/ses-{session}]/{datatype<perf>|perf}/sub-{subject}[_ses-{session}][_acq-{acquisition}][_rec-{reconstruction}][_dir{direction}][_run-{run}][_recording{recording}]_{suffix<physio|stim>}{extension<.tsv|.json>}",
"[acq-{acquisition}_][ce-{ceagent}_][rec-{reconstruction}_]{suffix<T1w|T2w|T1rho|T1map|T2map|T2star|FLAIR|FLASH|PDmap|PD|PDT2|inplaneT[12]|angio>}{extension<.json>|.json}",
"[acq-{acquisition}_][ce-{ceagent}_][rec-{reconstruction}_][mod-{modality}_]{suffix<defacemask>}{extension<.json>|.json}",
"task-{task}[_acq-{acquisition}][_ce-{ceagent}][_dir-{direction}][_rec-{reconstruction}][_run-{run}][_echo-{echo}]_{suffix<bold|cbv|phase|sbref>}{extension<.json>|.json}",
"[acq-{acquisition}_]{suffix<dwi>}{extension<.json>|.json}",
"[acq-{acquisition}_][dir-{direction}_][run-{run}_]{fmap<phasediff|magnitude[1-2]|phase[1-2]|fieldmap>}{extension<.json>|.json}",
"[acq-{acquisition}_][ce-{ceagent}_]dir-{direction}[_run-{run}]_{fmap<epi>}{extension<.json>|.json}",
"task-{task}[_acq-{acquisition}][_rec-{reconstruction}][_run-{run}][_echo-{echo}][_recording-{recording}]_{suffix<events>}{extension<.json>|.json}",
"task-{task}[_acq-{acquisition}][_rec-{reconstruction}][_run-{run}][_echo-{echo}][_recording-{recording}]_{suffix<physio|stim>}{extension<.json>}"
"task-{task}[_acq-{acquisition}][_rec-{reconstruction}][_run-{run}][_echo-{echo}][_recording-{recording}]_{suffix<physio|stim>}{extension<.json>}",
"sample-{sample}[_acq-{acquisition}][_stain-{label}][_run-{run}][_chunk-{index}]_{suffix<TEM|SEM|uCT|BF|DF|PC|DIC|FLUO|CONF|PLI|CARS|2PE|MPE|SR|NLO|OCT|SPIM>}{extension<.json>|.json}",
"[acq-{acquisition}][_ce-{ceagent}][_rec-{label}][_run-{run}]_echo-{echo}[_part-{<mag|phase|real|imag>}]_{suffix<MEGRE|MESE>}{extension<.json>|.json}",
"[acq-{acquisition}][_ce-{ceagent}][_rec-{label}][_run-{run}][_echo-{echo}]_flip-{flip}[_part-{<mag|phase|real|imag>}]_{suffix<VFA>}{extension<.json>|.json}",
"[acq-{acquisition}][_ce-{ceagent}][_rec-{label}][_run-{run}]_inv-{inv}[_part-{<mag|phase|real|imag>}]_{suffix<IRT1>}{extension<.json>|.json}",
"[acq-{acquisition}][_ce-{ceagent}][_rec-{label}][_run-{run}][_echo-{echo}][_flip-{flip}]_inv-{inv}[_part-{<mag|phase|real|imag>}]_{suffix<MP2RAGE>}{extension<.json>|.json}",
"[acq-{acquisition}][_ce-{ceagent}][_rec-{label}][_run-{run}][_echo-{echo}]_flip-{flip}_mt-{on|off}[_part-{<mag|phase|real|imag>}]_{suffix<MPM|MTS>}{extension<.json>|.json}",
"[acq-{acquisition}][_ce-{ceagent}][_rec-{label}][_run-{run}]_mt-{on|off}[_part-{<mag|phase|real|imag>}]_{suffix<MTR>}{extension<.json>|.json}",
"[acq-{acquisition}][_ce-{ceagent}][_rec-{label}][_run-{run}]_{suffix<T1map|T2map|T2starmap|R1map|R2map|R2starmap|PDmap|MTRmap|MTsat|UNIT1|T1rho|MWFmap|MTVmap|PDT2map|Chimap|S0map|M0map>}{extension<.json>|.json}",
"[task-{task}][trc-{tracer}][_rec-{reconstruction}][_run-{run}]_{suffix<pet>}{extension<.json>|.json}",
"[task-{task}][trc-{tracer}][_rec-{reconstruction}][_run-{run}]_recording-{recording}_{suffix<blood>}{extension<.json>|.json}",
"task-{task}[trc-{tracer}][_rec-{reconstruction}][_run-{run}]_{suffix<events>}{extension<.json>|.json}",
"[task-{task}][trc-{tracer}][_rec-{reconstruction}][_run-{run}][_recording-{recording}]_{suffix<physio|stim>}{extension<.json>|.json}",
"[acq-{acquisition}][_rec-{reconstruction}][_dir{direction}][_run-{run}]_{suffix<asl|m0scan>}{extension<.json>|.json}",
"[acq-{acquisition}][_rec-{reconstruction}][_dir{direction}][_run-{run}]_{suffix<aslcontext>}{extension<.json>|.json}",
"[acq-{acquisition}][_rec-{reconstruction}][_dir{direction}][_run-{run}][_recording{recording}]_{suffix<physio|stim>}{extension<.json>|.json}"
]
}
89 changes: 89 additions & 0 deletions bids/layout/tests/test_layout_on_examples.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,89 @@
""" Tests runs layout on bids examples and make sure all files are caught"""

""" TODO
- add more 'vanilla' datasets
- missing files in micr?
"""

from os.path import join

import pytest

from bids.layout import BIDSLayout
from bids.tests import get_test_data_path

# Values for the number of file got from a:
#
# find ds_name -type f | wc -l
#


@pytest.mark.parametrize(
"dataset, nb_files",
[
("qmri_irt1", 15),
("qmri_mese", 73),
("qmri_mp2rage", 14),
("qmri_mp2rageme", 28),
("qmri_mtsat", 23),
("qmri_sa2rage", 9),
("qmri_vfa", 17),
],
)
def test_layout_on_examples_with_derivatives(dataset, nb_files):
ds = join(get_test_data_path(), "bids-examples", dataset)
layout = BIDSLayout(ds, derivatives=True)
files = layout.get()
assert len(files) == nb_files


@pytest.mark.parametrize(
"dataset, nb_files",
[
("micr_SEM", 12),
("micr_SPIM", 22),
("asl001", 8),
("asl002", 10),
("asl003", 10),
("asl004", 12),
("asl005", 10),
("pet001", 12),
("pet002", 20),
("pet003", 9),
("pet004", 10),
("pet005", 14),
("qmri_megre", 18),
("qmri_tb1tfl", 6),
],
)
def test_layout_on_examples_no_derivatives(dataset, nb_files):
ds = join(get_test_data_path(), "bids-examples", dataset)
layout = BIDSLayout(ds)
files = layout.get()
assert len(files) == nb_files


@pytest.mark.parametrize(
"dataset, nb_files",
[
pytest.param(
"qmri_qsm",
8,
marks=pytest.mark.xfail(strict=True,
reason="https://github.com/bids-standard/bids-examples/issues/311"
),
),
pytest.param(
"qmri_mpm",
125,
marks=pytest.mark.xfail(strict=True,
reason="https://github.com/bids-standard/bids-examples/issues/310"
),
),
],
)
def test_layout_on_examples_invalid_ds_description(dataset, nb_files):
ds = join(get_test_data_path(), "bids-examples", dataset)
layout = BIDSLayout(ds, derivatives=True)
files = layout.get()
assert len(files) == nb_files