From e41b5173a8c18ab60b620861d6ba8743bd2bec0f Mon Sep 17 00:00:00 2001 From: Alex Rockhill Date: Thu, 22 Feb 2024 09:54:45 -0800 Subject: [PATCH] Do not zero-pad indices in BIDSPath entities anymore; do not warn about prepending missing dots for extensions (#1215) * [BUG] Allow int subject etc * fix * remove magic padding * fix tests * change position of logic back * fix logic * revert some changes * update changelog * remove zfill again * bring back test --------- Co-authored-by: Stefan Appelhoff --- doc/whats_new.rst | 8 +++++++- mne_bids/path.py | 14 +------------- mne_bids/tests/test_path.py | 36 +++++++++++++++--------------------- 3 files changed, 23 insertions(+), 35 deletions(-) diff --git a/doc/whats_new.rst b/doc/whats_new.rst index 447163df8..07d32b090 100644 --- a/doc/whats_new.rst +++ b/doc/whats_new.rst @@ -21,6 +21,8 @@ The following authors contributed for the first time. Thank you so much! 🤩 The following authors had contributed before. Thank you for sticking around! 🤘 +* `Alex Rockhill`_ +* `Eric Larson`_ * `Laetitia Fesselier`_ * `Richard Höchenberger`_ * `Stefan Appelhoff`_ @@ -38,6 +40,10 @@ Detailed list of changes - The experimental support for running MNE-BIDS examples from your browser using Binder has been removed, by `Stefan Appelhoff`_ (:gh:`1202`) +- MNE-BIDS will no longer zero-pad ("zfill") entity indices passed to :class:`~mne_bids.BIDSPath`. + For example, If ``run=1`` is passed to MNE-BIDS, it will no longer be silently auto-converted to ``run-01``, by `Alex Rockhill`_ (:gh:`1215`) +- MNE-BIDS will no longer warn about missing leading punctuation marks for extensions passed :class:`~mne_bids.BIDSPath`. + For example, MNE-BIDS will now silently auto-convert ``edf`` to ```.edf``, by `Alex Rockhill`_ (:gh:`1215`) 🛠 Requirements ^^^^^^^^^^^^^^^ @@ -52,7 +58,7 @@ Detailed list of changes ^^^^^^^^^^^^ - The datatype in the dataframe returned by :func:`mne_bids.stats.count_events` is now - ``pandas.Int64Dtype`` instead of ``float64``. + ``pandas.Int64Dtype`` instead of ``float64``, by `Eric Larson`_ (:gh:`1227`) ⚕️ Code health ^^^^^^^^^^^^^^ diff --git a/mne_bids/path.py b/mne_bids/path.py index d05f01b47..39e6d5bb1 100644 --- a/mne_bids/path.py +++ b/mne_bids/path.py @@ -966,24 +966,12 @@ def update(self, *, check=None, **kwargs): if isinstance(val, str) and not val.isdigit(): raise ValueError(f"{key} is not an index (Got {val})") elif isinstance(val, int): - kwargs[key] = f"{val:02}" + kwargs[key] = f"{val}" # ensure extension starts with a '.' extension = kwargs.get("extension") if extension is not None and not extension.startswith("."): - warn( - f'extension should start with a period ".", but got: ' - f'"{extension}". Prepending "." to form: ".{extension}". ' - f"This will raise an exception starting with MNE-BIDS 0.12.", - category=FutureWarning, - ) kwargs["extension"] = f".{extension}" - # Uncomment in 0.12, and remove above code: - # - # raise ValueError( - # f'Extension must start wie a period ".", but got: ' - # f'{extension}' - # ) del extension # error check entities diff --git a/mne_bids/tests/test_path.py b/mne_bids/tests/test_path.py index dedf69d68..efe449c7c 100644 --- a/mne_bids/tests/test_path.py +++ b/mne_bids/tests/test_path.py @@ -352,10 +352,10 @@ def test_parse_ext(): @pytest.mark.parametrize( "fname", [ - "sub-01_ses-02_task-test_run-3_split-01_meg.fif", - "sub-01_ses-02_task-test_run-3_split-01", - "/bids_root/sub-01/ses-02/meg/sub-01_ses-02_task-test_run-3_split-01_meg.fif", # noqa: E501 - "sub-01/ses-02/meg/sub-01_ses-02_task-test_run-3_split-01_meg.fif", + "sub-01_ses-02_task-test_run-3_split-1_meg.fif", + "sub-01_ses-02_task-test_run-3_split-1", + "/bids_root/sub-01/ses-02/meg/sub-01_ses-02_task-test_run-3_split-1_meg.fif", # noqa: E501 + "sub-01/ses-02/meg/sub-01_ses-02_task-test_run-3_split-1_meg.fif", ], ) def test_get_bids_path_from_fname(fname): @@ -377,12 +377,12 @@ def test_get_bids_path_from_fname(fname): @pytest.mark.parametrize( "fname", [ - "sub-01_ses-02_task-test_run-3_split-01_desc-filtered_meg.fif", - "sub-01_ses-02_task-test_run-3_split-01_desc-filtered.fif", - "sub-01_ses-02_task-test_run-3_split-01_desc-filtered", + "sub-01_ses-02_task-test_run-3_split-1_desc-filtered_meg.fif", + "sub-01_ses-02_task-test_run-3_split-1_desc-filtered.fif", + "sub-01_ses-02_task-test_run-3_split-1_desc-filtered", ( "/bids_root/sub-01/ses-02/meg/" - + "sub-01_ses-02_task-test_run-3_split-01_desc-filtered_meg.fif" + + "sub-01_ses-02_task-test_run-3_split-1_desc-filtered_meg.fif" ), ], ) @@ -394,7 +394,7 @@ def test_get_entities_from_fname(fname): assert params["run"] == "3" assert params["task"] == "test" assert params["description"] == "filtered" - assert params["split"] == "01" + assert params["split"] == "1" assert list(params.keys()) == [ "subject", "session", @@ -471,7 +471,7 @@ def test_get_entities_from_fname_errors(fname): # First candidate is disqualified (session doesn't match) (["sub-01_ses-01", "sub-01_ses-02"], ["sub-01_ses-02"]), # Multiple equally good candidates - (["sub-01_run-01", "sub-01_run-02"], ["sub-01_run-01", "sub-01_run-02"]), + (["sub-01_run-1", "sub-01_run-2"], ["sub-01_run-1", "sub-01_run-2"]), ], ) def test_find_best_candidates(candidate_list, best_candidates): @@ -796,7 +796,7 @@ def test_bids_path(return_bids_test_dir): # test that split gets properly set bids_path.update(split=1) - assert bids_path.basename == "sub-01_ses-02_task-03_split-01_ieeg.mat" + assert bids_path.basename == "sub-01_ses-02_task-03_split-1_ieeg.mat" # test home dir expansion bids_path = BIDSPath(root="~/foo") @@ -858,7 +858,7 @@ def test_make_filenames(): datatype="ieeg", ) expected_str = ( - "sub-one_ses-two_task-three_acq-four_run-01_proc-six_" "rec-seven_ieeg.json" + "sub-one_ses-two_task-three_acq-four_run-1_proc-six_" "rec-seven_ieeg.json" ) assert BIDSPath(**prefix_data).basename == expected_str assert ( @@ -869,7 +869,7 @@ def test_make_filenames(): # subsets of keys works assert ( BIDSPath(subject="one", task="three", run=4).basename - == "sub-one_task-three_run-04" + == "sub-one_task-three_run-4" ) assert ( BIDSPath(subject="one", task="three", suffix="meg", extension=".json").basename @@ -897,7 +897,7 @@ def test_make_filenames(): basename = BIDSPath(**prefix_data, check=False) assert ( basename.basename - == "sub-one_ses-two_task-three_acq-four_run-01_proc-six_rec-seven_ieeg.h5" + == "sub-one_ses-two_task-three_acq-four_run-1_proc-six_rec-seven_ieeg.h5" ) # noqa # what happens with scans.tsv file @@ -960,7 +960,7 @@ def test_match(return_bids_test_dir): bids_path_01 = BIDSPath(root=bids_root, run="01") paths = bids_path_01.match() assert len(paths) == 3 - assert paths[0].basename == ("sub-01_ses-01_task-testing_run-01_" "channels.tsv") + assert paths[0].basename == ("sub-01_ses-01_task-testing_run-01_channels.tsv") bids_path_01 = BIDSPath(root=bids_root, subject="unknown") paths = bids_path_01.match() @@ -1474,12 +1474,6 @@ def test_setting_entities(): assert getattr(bids_path, entity_name) is None -def test_deprecation(): - """Test deprecated behavior.""" - with pytest.warns(FutureWarning, match="This will raise an exception"): - BIDSPath(extension="vhdr") # no leading period - - def test_dont_create_dirs_on_fpath_access(tmp_path): """Regression test: don't create directories when accessing .fpath.""" bp = BIDSPath(subject="01", datatype="eeg", root=tmp_path)