Skip to content

Commit

Permalink
[SCHEMA] Add new MEG files and fix entity tables (#898)
Browse files Browse the repository at this point in the history
* Add new MEG structure.

* Document new rules format.

* Support the new structure in the code.

* Fix entity tables bug.

I definitely made a mistake in #883.

* Clean up code.

* Fix the entity table more.

* Drop unnecessary MEG group.

* Try something.

* Handle arbitrary numbers of rows and just use duplicate name.

* Forgot to revert the import change.
  • Loading branch information
tsalo authored Oct 18, 2021
1 parent 638ae5c commit b4dbd65
Show file tree
Hide file tree
Showing 3 changed files with 179 additions and 68 deletions.
51 changes: 22 additions & 29 deletions src/schema/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -383,58 +383,51 @@ The `extensions` entry is a list of valid file extensions.
The `entities` entry is a dictionary in which the keys are entity names and the values are whether the entity is
required or optional for that suffix.
Any entities that are not present in this dictionary are not allowed in files with any of the suffixes in the group.
In rare occasions, there are restrictions on valid entity values
(for example, some suffixes may only allow an `acq` value of `calibration`).
In those cases, the entity's value will be another object, rather than a string indicating the requirement level.
This object will contain at least two keys: "requirement" and "type".

**NOTE**: The order in which entities appear in these dictionaries does not reflect how they should appear in filenames.
That information is present in `rules/entities.yaml`.

As an example, let us look at part of `func.yaml`:
As an example, let us look at part of `meg.yaml`:

```yaml
- suffixes:
- bold
- cbv
- sbref
- meg
extensions:
- .nii.gz
- .nii
- .json
- .fif
entities:
subject: required
session: optional
task: required
acquisition: optional
ceagent: optional
reconstruction: optional
direction: optional
run: optional
echo: optional
part: optional
processing: optional
split: optional
# Phase (deprecated)
- suffixes:
- phase # deprecated
- meg
extensions:
- .nii.gz
- .nii
- .json
- .fif
entities:
subject: required
session: optional
task: required
acquisition: optional
ceagent: optional
reconstruction: optional
direction: optional
run: optional
echo: optional
acquisition:
requirement: required
type: string
enum:
- crosstalk
```

In this case, the first group has three suffixes: `bold`, `cbv`, and `sbref`.
The second group has one suffix: `phase`.
While the valid extensions are the same for both groups (`.nii.gz`, `.nii`, and `.json`), the entities are not.
In this case, the first group has one suffix: `meg`.
The second group has the same suffix (`meg`), but describes different rules for files with that suffix.
While the valid extension is the same for both groups (`.fif`), the entities are not.

Specifically, files with the `phase` suffix may not have a `part` entity,
while this is an option for files with the `bold`, `cbv`, or `sbref` suffixes.
Specifically, files in the first group may have `task`, `run`, `processing`, and `split` entities,
while files in the second group may not.
Also, when files in the second group have the `acq` entity, the associated value MUST be `crosstalk`.

### `entities.yaml`

Expand Down
38 changes: 36 additions & 2 deletions src/schema/rules/datatypes/meg.yaml
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
---
# MEG data files
# First group
- suffixes:
- meg
Expand All @@ -24,6 +25,36 @@
split: optional

# Second group
# Specifically, it's dat files with "acq-calibration"
- suffixes:
- meg
extensions:
- .dat
entities:
subject: required
session: optional
acquisition:
requirement: required
type: string
enum:
- calibration

# Third group
# fif files with "acq-crosstalk"
- suffixes:
- meg
extensions:
- .fif
entities:
subject: required
session: optional
acquisition:
requirement: required
type: string
enum:
- crosstalk

# Headshape files
- suffixes:
- headshape
extensions:
Expand All @@ -33,7 +64,7 @@
session: optional
acquisition: optional

# Third group
# Marker files
- suffixes:
- markers
extensions:
Expand All @@ -46,7 +77,7 @@
acquisition: optional
space: optional

# Fourth
# Coordinate systems
- suffixes:
- coordsystem
extensions:
Expand All @@ -56,6 +87,7 @@
session: optional
acquisition: optional

# Channel files
- suffixes:
- channels
extensions:
Expand All @@ -69,6 +101,7 @@
run: optional
processing: optional

# Events files
- suffixes:
- events
extensions:
Expand All @@ -81,6 +114,7 @@
acquisition: optional
run: optional

# Head photos
- suffixes:
- photo
extensions:
Expand Down
158 changes: 121 additions & 37 deletions tools/schemacode/schema.py
Original file line number Diff line number Diff line change
Expand Up @@ -208,6 +208,24 @@ def make_entity_definitions(schema):
return text


def _add_entity(filename_template, entity_pattern, requirement_level):
"""Add entity pattern to filename template based on requirement level."""
if requirement_level == "required":
if len(filename_template.strip()):
filename_template += "_" + entity_pattern
else:
# Only the first entity doesn't need an underscore
filename_template += entity_pattern
else:
if len(filename_template.strip()):
filename_template += "[_" + entity_pattern + "]"
else:
# Only the first entity doesn't need an underscore
filename_template += "[" + entity_pattern + "]"

return filename_template


def make_filename_template(schema, **kwargs):
"""Create codeblocks containing example filename patterns for a given
datatype.
Expand Down Expand Up @@ -248,23 +266,35 @@ def make_filename_template(schema, **kwargs):
for group in schema["rules"]["datatypes"][datatype]:
string = "\t\t\t"
for ent in entity_order:
ent_format = "{}-<{}>".format(
schema["objects"]["entities"][ent]["entity"],
schema["objects"]["entities"][ent].get("format", "label")
)
if "enum" in schema["objects"]["entities"][ent].keys():
# Entity key-value pattern with specific allowed values
ent_format = "{}-<{}>".format(
schema["objects"]["entities"][ent]["entity"],
"|".join(schema["objects"]["entities"][ent]["enum"]),
)
else:
# Standard entity key-value pattern with simple label/index
ent_format = "{}-<{}>".format(
schema["objects"]["entities"][ent]["entity"],
schema["objects"]["entities"][ent].get("format", "label"),
)

if ent in group["entities"]:
if group["entities"][ent] == "required":
if len(string.strip()):
string += "_" + ent_format
else:
# Only the first entity doesn't need an underscore
string += ent_format
if isinstance(group["entities"][ent], dict):
if "enum" in group["entities"][ent].keys():
# Overwrite the filename pattern based on the valid values
ent_format = "{}-<{}>".format(
schema["objects"]["entities"][ent]["entity"],
"|".join(group["entities"][ent]["enum"]),
)

string = _add_entity(
string,
ent_format,
group["entities"][ent]["requirement"],
)
else:
if len(string.strip()):
string += "[_" + ent_format + "]"
else:
# Only the first entity doesn't need an underscore
string += "[" + ent_format + "]"
string = _add_entity(string, ent_format, group["entities"][ent])

# In cases of large numbers of suffixes,
# we use the "suffix" variable and expect a table later in the spec
Expand Down Expand Up @@ -332,26 +362,27 @@ def make_entity_table(schema, tablefmt="github", **kwargs):
# import pdb; pdb.set_trace()
header = ["Entity", "DataType"]
formats = ["Format", "DataType"]
entity_to_col = {}
table = [formats]

# Compose header and formats first
for i, (entity, spec) in enumerate(schema["objects"]["entities"].items()):
entity_shorthand = schema["objects"]["entities"][entity]["entity"]
header.append(spec["name"])
all_entities = schema["rules"]["entities"]
for entity in all_entities:
entity_spec = schema["objects"]["entities"][entity]
entity_shorthand = entity_spec["entity"]
header.append(entity_spec["name"])
formats.append(
f'[`{entity_shorthand}-<{spec.get("format", "label")}>`]'
f'[`{entity_shorthand}-<{entity_spec.get("format", "label")}>`]'
f"({ENTITIES_FILE}#{entity_shorthand})"
)
entity_to_col[entity] = i + 1

# Go through data types
for dtype, dtype_specs in schema["rules"]["datatypes"].items():
dtype_rows = {}
duplicate_row_counter = 0

# each dtype could have multiple specs
for spec in dtype_specs:
suffixes = spec.get("suffixes")
for i_dtype_spec, dtype_spec in enumerate(dtype_specs):
suffixes = dtype_spec.get("suffixes")

# Skip this part of the schema if no suffixes are found.
# This is a hack to work around filter_schema's limitations.
Expand All @@ -360,27 +391,60 @@ def make_entity_table(schema, tablefmt="github", **kwargs):

# TODO: <br> is specific for html form
suffixes_str = " ".join(suffixes) if suffixes else ""
dtype_row = [dtype] + ([""] * len(entity_to_col))
for ent, req in spec.get("entities", []).items():
dtype_row[entity_to_col[ent]] = req.upper()
dtype_row = [dtype] + ([""] * len(all_entities))
for ent, ent_info in dtype_spec.get("entities", {}).items():
if isinstance(ent_info, dict):
requirement_level = ent_info["requirement"]
else:
requirement_level = ent_info

dtype_row[all_entities.index(ent) + 1] = requirement_level.upper()

# Merge specs within dtypes if they share all of the same entities
if dtype_row in dtype_rows.values():
for k, v in dtype_rows.items():
if dtype_row == v:
dtype_rows.pop(k)
new_k = k + " " + suffixes_str
new_k = new_k.strip()
dtype_rows[new_k] = v
# Merge specs within dtypes if they share all of the same entities
for existing_suffixes_str, existing_entities in dtype_rows.items():
if dtype_row == existing_entities:
# Combine suffixes from the existing row with ones from the new row
dtype_rows.pop(existing_suffixes_str)
old_suffix_list = existing_suffixes_str.split(" ")
new_suffix_list = suffixes_str.split(" ")
comb_suffix_list = sorted(list(set(new_suffix_list + old_suffix_list)))

# Identify if the list of suffixes comes from an existing alternate row
number_suffixes = list(filter(str.isnumeric, comb_suffix_list))
if len(number_suffixes) == 1:
# Suffixes come from an existing alternate row
number = number_suffixes[0]
comb_suffix_list.remove(number)
new_suffixes_str = " ".join(comb_suffix_list)
# Retain the old number
new_suffixes_str = number + " " + new_suffixes_str
elif len(number_suffixes) > 1:
# The row exists already, but contains multiple numbers
raise Exception("Something's wrong here.")
else:
# It's a new row
new_suffixes_str = " ".join(comb_suffix_list)

dtype_rows[new_suffixes_str] = existing_entities
break

elif suffixes_str in dtype_rows.keys():
# Create new lines for multiple specs with the same dtype and suffix,
# but different entities
# Unfortunately, the keys need to be unique, so we include a number
# NOTE: This assumes that no suffix in BIDS will ever be purely numeric.
dtype_rows[str(duplicate_row_counter) + " " + suffixes_str] = dtype_row
duplicate_row_counter += 1

else:
# Otherwise, just add the new suffix group
dtype_rows[suffixes_str] = dtype_row

# Reformat first column
dtype_rows = {
dtype + "<br>({})".format(k): v for k, v in dtype_rows.items()
}
# Add datatype to first column and reformat it
dtype_rows = {dtype + "<br>({})".format(k): v for k, v in dtype_rows.items()}
dtype_rows = [[k] + v for k, v in dtype_rows.items()]

table += dtype_rows

# Create multi-level index because first two rows are headers
Expand All @@ -393,6 +457,26 @@ def make_entity_table(schema, tablefmt="github", **kwargs):
table = utils.drop_unused_entities(table)
table = utils.flatten_multiindexed_columns(table)

# Remove fake numeric suffixes from first column
def _remove_numeric_suffixes(string):
import re

suffix_str = re.findall("\((.+)\)", string)
# The "Format" row should be skipped
if not suffix_str:
return string

suffix_str = suffix_str[0] # Only one parenthesis should appear
suffixes = suffix_str.split(" ")
suffixes = list(filter(lambda v: not str.isnumeric(v), suffixes))
suffix_str2 = " ".join(suffixes)
new_string = string.replace(f"({suffix_str})", f"({suffix_str2})")
return new_string

table[table.index.name] = table.index
table[table.index.name] = table[table.index.name].apply(_remove_numeric_suffixes)
table = table.set_index(table.index.name, drop=True)

# Print it as markdown
table_str = tabulate(table, headers="keys", tablefmt=tablefmt)
return table_str
Expand Down

0 comments on commit b4dbd65

Please sign in to comment.