Skip to content
This repository has been archived by the owner on Dec 20, 2024. It is now read-only.

ENH: Indexes sorting options #150

Merged
merged 47 commits into from
Apr 8, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
47 commits
Select commit Hold shift + click to select a range
3343f34
enh: externalize sorting functions
esavary Apr 4, 2024
1e305d9
fix:import statement
esavary Apr 4, 2024
6a8a585
sty: fix style error
esavary Apr 4, 2024
4e199a9
sty: fix style error
esavary Apr 4, 2024
b13c975
sty: fix style errors
esavary Apr 4, 2024
148bb71
Apply suggestions from code review
esavary Apr 5, 2024
5155f9f
fix: revert stylistic changes
esavary Apr 5, 2024
9420ce5
fix: revert stylistic changes
esavary Apr 5, 2024
3d7afb1
fix: revert stylistic changes
esavary Apr 5, 2024
69611fa
Update src/eddymotion/utils.py
esavary Apr 5, 2024
31fa9d6
Update src/eddymotion/utils.py
esavary Apr 5, 2024
6320f66
Update src/eddymotion/utils.py
esavary Apr 5, 2024
4998afa
fix: revert changes in estimator.py
esavary Apr 5, 2024
a424582
Apply suggestions from code review
esavary Apr 5, 2024
88bea1d
fix: typos
esavary Apr 5, 2024
b58a708
enh: update args and test for bvalue_action
esavary Apr 5, 2024
4366683
fix: docstring
esavary Apr 5, 2024
ee95877
fix: remove unused import
esavary Apr 5, 2024
e84bddd
Apply suggestions from code review
esavary Apr 5, 2024
3c0b36d
fix: add link for new module documentation
esavary Apr 5, 2024
22bbb68
fix: add exeption
esavary Apr 5, 2024
43fe703
fix: bvalue_action implementation + typos
esavary Apr 5, 2024
11f52a4
sty: remove white space
esavary Apr 5, 2024
cdfe20a
fix: typos
esavary Apr 5, 2024
f858a6a
sty: remove white space
esavary Apr 5, 2024
90cf31c
Apply suggestions from code review
esavary Apr 8, 2024
fa43376
sty: change iterator names
esavary Apr 8, 2024
cdeb34f
sty: fix docstring
esavary Apr 8, 2024
965b3c5
Apply suggestions from code review
esavary Apr 8, 2024
a43fdde
fix: random seed and add test
esavary Apr 8, 2024
ae19816
fix: typos
esavary Apr 8, 2024
d78c500
enh: generalize docstring
esavary Apr 8, 2024
6aaa7d4
Update src/eddymotion/utils.py
esavary Apr 8, 2024
4e7a418
Apply suggestions from code review
esavary Apr 8, 2024
4163b0b
Apply suggestions from code review
esavary Apr 8, 2024
ed20ca8
fix: random generator test results
esavary Apr 8, 2024
d966c97
Apply suggestions from code review
esavary Apr 8, 2024
0057079
sty: add typing
esavary Apr 8, 2024
6f0d5fe
fix: remove unused import
esavary Apr 8, 2024
2cc923a
Apply suggestions from code review
esavary Apr 8, 2024
9bda539
fix: import position
esavary Apr 8, 2024
d1c9846
Apply suggestions from code review
esavary Apr 8, 2024
807a468
fix: iterator typing in docstring
esavary Apr 8, 2024
fee865d
sty: fix ruff errors
esavary Apr 8, 2024
5f0837d
Merge branch 'nipreps:main' into index-sorting
esavary Apr 8, 2024
caa72a5
sty: fix sty errors
esavary Apr 8, 2024
911a730
sty: fix sty errors
esavary Apr 8, 2024
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
1 change: 1 addition & 0 deletions docs/developers.rst
Original file line number Diff line number Diff line change
Expand Up @@ -32,4 +32,5 @@ Information on specific functions, classes, and methods.
api/eddymotion.data.dmri
api/eddymotion.estimator
api/eddymotion.model
api/eddymotion.utils
api/eddymotion.viz
172 changes: 172 additions & 0 deletions src/eddymotion/utils.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,172 @@
# emacs: -*- mode: python; py-indent-offset: 4; indent-tabs-mode: nil -*-
# vi: set ft=python sts=4 ts=4 sw=4 et:
#
# Copyright 2024 The NiPreps Developers <[email protected]>
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# We support and encourage derived works from this project, please read
# about our expectations at
#
# https://www.nipreps.org/community/licensing/
#
"""Iterators to traverse the volumes in a 4D image."""

import random
from itertools import chain, zip_longest
from typing import Iterator


oesteban marked this conversation as resolved.
Show resolved Hide resolved
def linear_iterator(size: int = None, **kwargs) -> Iterator[int]:
"""
Traverse the dataset volumes in ascending order.

Parameters
----------
size : :obj:`int`
Number of volumes in the dataset
(for instance, the number of orientations in a DWI)

Returns
-------
:obj:`~typing.Iterator`
The sorted index order.

Examples
--------
>>> list(linear_iterator(10))
[0, 1, 2, 3, 4, 5, 6, 7, 8, 9]

"""
if size is None and "bvals" in kwargs:
size = len(kwargs["bvals"])
if size is None:
raise TypeError("Cannot build iterator without size")

return range(size)
esavary marked this conversation as resolved.
Show resolved Hide resolved


def random_iterator(size: int = None, **kwargs) -> Iterator[int]:
"""
Traverse the dataset volumes randomly.

Parameters
----------
size : :obj:`int`
Number of volumes in the dataset
(for instance, the number of orientations in a DWI)
seed : :obj:`int` or :obj:`bool` or :obj:`bool` or ``None``
If :obj:`int` or :obj:`str` or ``None``, initializes the seed of Python's random generator
with the given value.
If ``False``, the random generator is passed ``None``.
If ``True``, a default seed value is set.

Returns
-------
:obj:`~typing.Iterator`
The sorted index order.
oesteban marked this conversation as resolved.
Show resolved Hide resolved

Examples
--------
>>> list(random_iterator(15, seed=0)) # seed is 0
[1, 10, 9, 5, 11, 2, 3, 7, 8, 4, 0, 14, 12, 6, 13]
>>> # seed is True -> the default value 20210324 is set
>>> list(random_iterator(15, seed=True))
[1, 12, 14, 5, 0, 11, 10, 9, 7, 8, 3, 13, 2, 6, 4]
>>> list(random_iterator(15, seed=20210324))
[1, 12, 14, 5, 0, 11, 10, 9, 7, 8, 3, 13, 2, 6, 4]
>>> list(random_iterator(15, seed=42)) # seed is 42
[8, 13, 7, 6, 14, 12, 5, 2, 9, 3, 4, 11, 0, 1, 10]

"""

if size is None and "bvals" in kwargs:
size = len(kwargs["bvals"])
if size is None:
raise TypeError("Cannot build iterator without size")

_seed = kwargs.get("seed", None)
_seed = 20210324 if _seed is True else _seed

random.seed(None if _seed is False else _seed)

index_order = list(range(size))
random.shuffle(index_order)
return (x for x in index_order)


def bvalue_iterator(size: int = None, **kwargs) -> Iterator[int]:
"""
Traverse the volumes in a DWI dataset by growing b-value.

Parameters
----------
bvalues : :obj:`list`
List of b-values corresponding to all orientations of the dataset.

Returns
-------
:obj:`~typing.Iterator`
The sorted index order.

Examples
--------
>>> list(bvalue_iterator(bvals=[0.0, 0.0, 1000.0, 1000.0, 700.0, 700.0, 2000.0, 2000.0, 0.0]))
[0, 1, 8, 4, 5, 2, 3, 6, 7]

"""
bvals = kwargs.get("bvals", None)
if bvals is None:
raise TypeError("Keyword argument bvals is required")
indexed_bvals = sorted([(round(b, 2), i) for i, b in enumerate(bvals)])
return (index[1] for index in indexed_bvals)


def centralsym_iterator(size: int = None, **kwargs) -> Iterator[int]:
"""
Traverse the dataset starting from the center and alternatingly progressing to the sides.

Parameters
----------
size : :obj:`int`
Number of volumes in the dataset
(for instance, the number of orientations in a DWI)

Returns
-------
:obj:`~typing.Iterator`
The sorted index order.

Examples
--------
>>> list(centralsym_iterator(10))
[5, 4, 6, 3, 7, 2, 8, 1, 9, 0]
>>> list(centralsym_iterator(11))
[5, 4, 6, 3, 7, 2, 8, 1, 9, 0, 10]

esavary marked this conversation as resolved.
Show resolved Hide resolved
"""
if size is None and "bvals" in kwargs:
size = len(kwargs["bvals"])
if size is None:
raise TypeError("Cannot build iterator without size")
linear = list(range(size))
esavary marked this conversation as resolved.
Show resolved Hide resolved
return (
x
for x in chain.from_iterable(
zip_longest(
linear[size // 2 :],
reversed(linear[: size // 2]),
)
)
if x is not None
)