From 7c7de1662f62586a894e6198f39cd205e1dda99d Mon Sep 17 00:00:00 2001 From: Chris Markiewicz Date: Tue, 24 Sep 2024 10:05:24 -0400 Subject: [PATCH] FIX: Normalize BIDS-URIs to subject-relative --- sdcflows/utils/tests/test_wrangler.py | 13 +++++++++-- sdcflows/utils/wrangler.py | 33 +++++++++++++++++++++++++-- 2 files changed, 42 insertions(+), 4 deletions(-) diff --git a/sdcflows/utils/tests/test_wrangler.py b/sdcflows/utils/tests/test_wrangler.py index c41a4297ff..f5e7c6cdd5 100644 --- a/sdcflows/utils/tests/test_wrangler.py +++ b/sdcflows/utils/tests/test_wrangler.py @@ -1,10 +1,13 @@ -import pytest +import re +from pathlib import Path from shutil import rmtree +import pytest + from niworkflows.utils.testing import generate_bids_skeleton from sdcflows.utils.wrangler import find_estimators -from sdcflows.fieldmaps import clear_registry +from sdcflows.fieldmaps import clear_registry, get_identifier def gen_layout(bids_dir, database_dir=None): @@ -339,6 +342,12 @@ def test_wrangler_URIs(tmpdir, name, skeleton, session, estimations, total_estim sessions=[session] if session else None, ) assert len(est) == estimations or total_estimations + if session: + bold = layout.get(session=session, suffix="bold", extension=".nii.gz")[0] + intended_rel = re.sub(r'^sub-[a-zA-Z0-9]*/', '', str(Path(bold).relative_to(layout.root))) + b0_id = get_identifier(intended_rel) + assert b0_id == ('auto_00000',) + clear_registry() diff --git a/sdcflows/utils/wrangler.py b/sdcflows/utils/wrangler.py index c0086acefe..21f28d65fc 100644 --- a/sdcflows/utils/wrangler.py +++ b/sdcflows/utils/wrangler.py @@ -34,6 +34,23 @@ from .. import fieldmaps as fm +def _normalize_intent( + intent: str, + layout: BIDSLayout, + subject: str +) -> str | None: + """Convert BIDS-URI intent to subject-relative intent + + SDCFlows currently makes strong assumptions about old-style intents, + and a change to that needs to be carefully considered and tested. + """ + if intent.startswith("bids::"): + # bids::sub-/ + # ^- 10 ^- 11 + return intent[11 + len(subject):] + return intent + + def _resolve_intent( intent: str, layout: BIDSLayout, @@ -47,6 +64,18 @@ def _resolve_intent( return intent +def _filter_metadata( + metadata: Dict[str, Any], + layout: BIDSLayout, + subject: str +) -> Dict[str, Any]: + intents = metadata.get("IntendedFor") + if intents: + updated = [_normalize_intent(intent, layout, subject) for intent in listify(intents)] + return {**metadata, "IntendedFor": updated} + return metadata + + def find_estimators( *, layout: BIDSLayout, @@ -377,7 +406,7 @@ def find_estimators( ): try: e = fm.FieldmapEstimation( - fm.FieldmapFile(fmap.path, metadata=fmap.get_metadata()) + fm.FieldmapFile(fmap.path, metadata=_filter_metadata(fmap.get_metadata(), layout, subject)) ) except (ValueError, TypeError) as err: _log_debug_estimator_fail( @@ -405,7 +434,7 @@ def find_estimators( if len(dirs) > 1: by_intent = {} for fmap in layout.get(**{**entities, **{'direction': dirs}}): - fmapfile = fm.FieldmapFile(fmap.path, metadata=fmap.get_metadata()) + fmapfile = fm.FieldmapFile(fmap.path, metadata=_filter_metadata(fmap.get_metadata(), layout, subject)) by_intent.setdefault( tuple(fmapfile.metadata.get('IntendedFor', ())), [] ).append(fmapfile)