Skip to content

Commit

Permalink
REL: v0.11.1 (#1089)
Browse files Browse the repository at this point in the history
* avoid creating many Annotations objects in for loop (#1079)

* ENH: Better error message (#1080)

* ENH: Better error message

* FIX: Beautiful infrastructure

* FIX: Spacing

* ENH: Improve error message (#1081)

* ENH: Improve error message

* FIX: Use working openneuro-py

* FIX: Install

* FIX: Name

* ENH: Make suggestion (#1087)

* MRG: Enforce specification of all annotation descriptions in event_id if event_id is passed to write_raw_bids() (#1086)

* Write all annotations even if event_id was passed

* Implement new logic

* Update changelog

* Fix tests

* Style

* REL: v0.11.1

Co-authored-by: Alexandre Gramfort <[email protected]>
Co-authored-by: Eric Larson <[email protected]>
Co-authored-by: Richard Höchenberger <[email protected]>
  • Loading branch information
4 people authored Oct 21, 2022
1 parent c31854a commit b9316bd
Show file tree
Hide file tree
Showing 8 changed files with 160 additions and 29 deletions.
11 changes: 11 additions & 0 deletions doc/whats_new.rst
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,17 @@
What's new?
===========

.. _changes_0_11_1:

Version 0.11.1 (2022-10-21)
---------------------------

Version 0.11.1 is a patch release, all changes are listed below.
For more complete information on the 0.11 version, see changelog below.

- Speed up :func:`mne_bids.read_raw_bids` when lots of events are present by `Alexandre Gramfort`_ (:gh:`1079`)
- When writing data containing :class:`mne.Annotations` **and** passing events to :func:`~mne_bids.write_raw_bids`, previously, annotations whose description did not appear in ``event_id`` were silently dropped. We now raise an exception and request users to specify mappings between descriptions and event codes in this case. It is still possible to omit ``event_id`` if no ``events`` are passed, by `Richard Höchenberger`_ (:gh:`1084`)

.. _changes_0_11:

Version 0.11 (2022-10-08)
Expand Down
2 changes: 1 addition & 1 deletion mne_bids/__init__.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
"""MNE software for easily interacting with BIDS compatible datasets."""

__version__ = '0.11'
__version__ = '0.11.1'
from mne_bids import commands
from mne_bids.report import make_report
from mne_bids.path import (BIDSPath, get_datatypes, get_entity_vals,
Expand Down
12 changes: 8 additions & 4 deletions mne_bids/path.py
Original file line number Diff line number Diff line change
Expand Up @@ -953,8 +953,11 @@ def find_empty_room(self, use_sidecar_only=False, verbose=None):
)
er_bids_path = _find_matched_empty_room(self)

if er_bids_path is not None:
assert er_bids_path.fpath.exists()
if er_bids_path is not None and not er_bids_path.fpath.exists():
raise FileNotFoundError(
f'Empty-room BIDS path resolved but not found:\n'
f'{er_bids_path}\n'
'Check your BIDS dataset for completeness.')

return er_bids_path

Expand Down Expand Up @@ -1510,9 +1513,10 @@ def _find_matching_sidecar(bids_path, suffix=None,
f'associated with {bids_path.basename}.')
elif len(best_candidates) > 1:
# More than one candidates were tied for best match
msg = (f'Expected to find a single {suffix} file '
msg = (f'Expected to find a single {search_suffix} file '
f'associated with {bids_path.basename}, '
f'but found {len(candidate_list)}: "{candidate_list}".')
f'but found {len(candidate_list)}:\n\n' +
"\n".join(candidate_list))
msg += f'\n\nThe search_str was "{search_str_complete}"'
if on_error == 'raise':
raise RuntimeError(msg)
Expand Down
44 changes: 37 additions & 7 deletions mne_bids/read.py
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@
import re
from datetime import datetime, timezone
from difflib import get_close_matches
import os

import numpy as np
import mne
Expand Down Expand Up @@ -123,8 +124,29 @@ def _read_events(events, event_id, raw, bids_path=None):
else:
events = read_events(events).astype(int)

if raw.annotations:
if event_id is None:
logger.info(
'The provided raw data contains annotations, but you did not '
'pass an "event_id" mapping from annotation descriptions to '
'event codes. We will generate arbitrary event codes. '
'To specify custom event codes, please pass "event_id".'
)
else:
desc_without_id = sorted(
set(raw.annotations.description) - set(event_id.keys())
)
if desc_without_id:
raise ValueError(
f'The provided raw data contains annotations, but '
f'"event_id" does not contain entries for all annotation '
f'descriptions. The following entries are missing: '
f'{", ".join(desc_without_id)}'
)

# If we have events, convert them to Annotations so they can be easily
# merged with existing Annotations.
if events.size > 0:
# Only keep events for which we have an ID <> description mapping.
ids_without_desc = set(events[:, 2]) - set(event_id.values())
if ids_without_desc:
raise ValueError(
Expand All @@ -133,9 +155,6 @@ def _read_events(events, event_id, raw, bids_path=None):
f'Please add them to the event_id dictionary, or drop them '
f'from the events array.'
)
del ids_without_desc
mask = [e in list(event_id.values()) for e in events[:, 2]]
events = events[mask]

# Append events to raw.annotations. All event onsets are relative to
# measurement beginning.
Expand Down Expand Up @@ -491,8 +510,9 @@ def _handle_events_reading(events_fname, raw):
description=descriptions)
raw.set_annotations(annot_from_events)

annot_idx_to_keep = [idx for idx, annot in enumerate(annot_from_raw)
if annot['description'] in ANNOTATIONS_TO_KEEP]
annot_idx_to_keep = [idx for idx, descr
in enumerate(annot_from_raw.description)
if descr in ANNOTATIONS_TO_KEEP]
annot_to_keep = annot_from_raw[annot_idx_to_keep]

if len(annot_to_keep):
Expand Down Expand Up @@ -719,7 +739,17 @@ def read_raw_bids(bids_path, extra_params=None, verbose=None):
break

if not raw_path.exists():
raise FileNotFoundError(f'File does not exist: {raw_path}')
options = os.listdir(bids_path.directory)
matches = get_close_matches(bids_path.basename, options)
msg = f'File does not exist:\n{raw_path}'
if matches:
msg += (
'\nDid you mean one of:\n' +
'\n'.join(matches) +
'\ninstead of:\n' +
bids_path.basename
)
raise FileNotFoundError(msg)
if config_path is not None and not config_path.exists():
raise FileNotFoundError(f'config directory not found: {config_path}')

Expand Down
7 changes: 7 additions & 0 deletions mne_bids/tests/test_path.py
Original file line number Diff line number Diff line change
Expand Up @@ -960,11 +960,18 @@ def test_find_empty_room(return_bids_test_dir, tmp_path):

# Retrieve empty-room BIDSPath
assert bids_path.find_empty_room() == er_associated_bids_path
assert bids_path.find_empty_room(
use_sidecar_only=True) == er_associated_bids_path

# Should only work for MEG
with pytest.raises(ValueError, match='only supported for MEG'):
bids_path.copy().update(datatype='eeg').find_empty_room()

# Raises an error if the file is missing
os.remove(er_associated_bids_path.fpath)
with pytest.raises(FileNotFoundError, match='Empty-room BIDS .* not foun'):
bids_path.find_empty_room(use_sidecar_only=True)

# Don't create `AssociatedEmptyRoom` entry in sidecar – we should now
# retrieve the empty-room recording closer in time
write_raw_bids(raw, bids_path=bids_path, empty_room=None, overwrite=True)
Expand Down
10 changes: 10 additions & 0 deletions mne_bids/tests/test_read.py
Original file line number Diff line number Diff line change
Expand Up @@ -1297,6 +1297,7 @@ def test_channels_tsv_raw_mismatch(tmp_path):
read_raw_bids(bids_path)


@testing.requires_testing_data
def test_file_not_found(tmp_path):
"""Check behavior if the requested file cannot be found."""
# First a path with a filename extension.
Expand All @@ -1313,6 +1314,15 @@ def test_file_not_found(tmp_path):
with pytest.raises(FileNotFoundError, match='File does not exist'):
read_raw_bids(bids_path=bp)

bp.update(extension='.fif')
_read_raw_fif(raw_fname, verbose=False).save(bp.fpath)
with pytest.warns(RuntimeWarning, match=r'channels\.tsv'):
read_raw_bids(bp) # smoke test

bp.update(task=None)
with pytest.raises(FileNotFoundError, match='Did you mean'):
read_raw_bids(bp)


@requires_version('mne', '1.2')
@pytest.mark.filterwarnings(warning_str['channel_unit_changed'])
Expand Down
84 changes: 76 additions & 8 deletions mne_bids/tests/test_write.py
Original file line number Diff line number Diff line change
Expand Up @@ -1272,17 +1272,10 @@ def test_eegieeg(dir_name, fname, reader, _bids_validate, tmp_path):
match='Encountered data in "double" format'):
bids_output_path = write_raw_bids(**kwargs)

event_id = {'Auditory/Left': 1, 'Auditory/Right': 2, 'Visual/Left': 3,
'Visual/Right': 4, 'Smiley': 5, 'Button': 32}

with pytest.raises(ValueError,
match='You passed events, but no event_id '):
write_raw_bids(raw, bids_path, events=events)

with pytest.raises(ValueError,
match='You passed event_id, but no events'):
write_raw_bids(raw, bids_path, event_id=event_id)

# check events.tsv is written
events_tsv_fname = bids_output_path.copy().update(suffix='events',
extension='.tsv')
Expand Down Expand Up @@ -2704,14 +2697,89 @@ def test_annotations(_bids_validate, bad_segments, tmp_path):
_bids_validate(bids_root)


@pytest.mark.parametrize(
'write_events', [True, False] # whether to pass "events" to write_raw_bids
)
@pytest.mark.filterwarnings(warning_str['channel_unit_changed'])
@testing.requires_testing_data
def test_annotations_and_events(_bids_validate, tmp_path, write_events):
"""Test combined writing of Annotations and events."""
bids_root = tmp_path / 'bids'
bids_path = _bids_path.copy().update(root=bids_root, datatype='meg')
raw_fname = data_path / 'MEG' / 'sample' / 'sample_audvis_trunc_raw.fif'
events_fname = (
data_path / 'MEG' / 'sample' / 'sample_audvis_trunc_raw-eve.fif'
)
events_tsv_fname = bids_path.copy().update(
suffix='events',
extension='.tsv',
)

events = mne.read_events(events_fname)
events = events[events[:, 2] != 0] # drop unknown "0" events
event_id = {'Auditory/Left': 1, 'Auditory/Right': 2, 'Visual/Left': 3,
'Visual/Right': 4, 'Smiley': 5, 'Button': 32}
raw = _read_raw_fif(raw_fname)
annotations = mne.Annotations(
# Try to avoid rounding errors.
onset=(
1 / raw.info['sfreq'] * 600,
1 / raw.info['sfreq'] * 600, # intentional
1 / raw.info['sfreq'] * 3000
),
duration=(
1 / raw.info['sfreq'],
1 / raw.info['sfreq'],
1 / raw.info['sfreq'] * 200
),
description=('BAD_segment', 'EDGE_segment', 'custom'),
)
raw.set_annotations(annotations)

# Write annotations while passing event_id
# Should raise since annotations descriptions are missing from event_id
with pytest.raises(ValueError, match='The following entries are missing'):
write_raw_bids(
raw,
bids_path=bids_path,
event_id=event_id,
events=events if write_events else None,
)

# Passing a complete mapping should work
event_id_with_annots = event_id.copy()
event_id_with_annots.update({
'BAD_segment': 9999,
'EDGE_segment': 10000,
'custom': 2000
})
write_raw_bids(
raw,
bids_path=bids_path,
event_id=event_id_with_annots,
events=events if write_events else None,
)
_bids_validate(bids_root)

# Ensure all events + annotations were written
events_tsv = _from_tsv(events_tsv_fname)

if write_events:
n_events_expected = len(events) + len(raw.annotations)
else:
n_events_expected = len(raw.annotations)

assert len(events_tsv['trial_type']) == n_events_expected


@pytest.mark.parametrize(
'drop_undescribed_events',
[True, False]
)
@pytest.mark.filterwarnings(warning_str['channel_unit_changed'])
@testing.requires_testing_data
def test_undescribed_events(_bids_validate, drop_undescribed_events, tmp_path):
"""Test we're behaving correctly if event descriptions are missing."""
"""Test we're raising if event descriptions are missing."""
bids_root = tmp_path / 'bids1'
bids_path = _bids_path.copy().update(root=bids_root, datatype='meg')
raw_fname = op.join(data_path, 'MEG', 'sample',
Expand Down
19 changes: 10 additions & 9 deletions mne_bids/write.py
Original file line number Diff line number Diff line change
Expand Up @@ -1337,14 +1337,16 @@ def write_raw_bids(
If an array, the MNE events array (shape: ``(n_events, 3)``).
If a path or an array and ``raw.annotations`` exist, the union of
``events`` and ``raw.annotations`` will be written.
Corresponding descriptions for all event codes (listed in the third
Mappings from event names to event codes (listed in the third
column of the MNE events array) must be specified via the ``event_id``
parameter; otherwise, an exception is raised.
parameter; otherwise, an exception is raised. If
:class:`~mne.Annotations` are present, their descriptions must be
included in ``event_id`` as well.
If ``None``, events will only be inferred from the raw object's
:class:`~mne.Annotations`.
.. note::
If ``not None``, writes the union of ``events`` and
If specified, writes the union of ``events`` and
``raw.annotations``. If you wish to **only** write
``raw.annotations``, pass ``events=None``. If you want to
**exclude** the events in ``raw.annotations`` from being written,
Expand All @@ -1359,7 +1361,10 @@ def write_raw_bids(
``events``. The descriptions will be written to the ``trial_type``
column in ``*_events.tsv``. The dictionary keys correspond to the event
description,s and the values to the event codes. You must specify a
description for all event codes appearing in ``events``.
description for all event codes appearing in ``events``. If your data
contains :class:`~mne.Annotations`, you can use this parameter to
assign event codes to each unique annotation description (mapping from
description to event code).
anonymize : dict | None
If `None` (default), no anonymization is performed.
If a dictionary, data will be anonymized depending on the dictionary
Expand Down Expand Up @@ -1575,11 +1580,7 @@ def write_raw_bids(

if events is not None and event_id is None:
raise ValueError('You passed events, but no event_id '
'dictionary. You need to pass both, or neither.')

if event_id is not None and events is None:
raise ValueError('You passed event_id, but no events. '
'You need to pass both, or neither.')
'dictionary.')

_validate_type(item=empty_room, item_name='empty_room',
types=(mne.io.BaseRaw, BIDSPath, None))
Expand Down

0 comments on commit b9316bd

Please sign in to comment.