Skip to content

Commit a8d9039

Browse files
committed
tst: ramping up coverage
1 parent 2bda625 commit a8d9039

File tree

5 files changed

+96
-40
lines changed

5 files changed

+96
-40
lines changed

nitransforms/base.py

+4-2
Original file line numberDiff line numberDiff line change
@@ -108,7 +108,7 @@ def __init__(self, image):
108108
self._ndim = 3
109109

110110
self._npoints = getattr(image, 'npoints',
111-
np.prod(image.shape))
111+
np.prod(self._shape))
112112
self._ndindex = None
113113
self._coords = None
114114
self._inverse = getattr(image, 'inverse',
@@ -173,9 +173,11 @@ class TransformBase(object):
173173

174174
__slots__ = ['_reference']
175175

176-
def __init__(self):
176+
def __init__(self, reference=None):
177177
"""Instantiate a transform."""
178178
self._reference = None
179+
if reference:
180+
self.reference = reference
179181

180182
def __call__(self, x, inverse=False):
181183
"""Apply y = f(x)."""

nitransforms/io/fsl.py

+5-6
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
"""Read/write FSL's transforms."""
22
import os
33
import numpy as np
4+
from pathlib import Path
45
from nibabel.affines import voxel_sizes
56

67
from .base import BaseLinearTransformList, LinearParameters, TransformFileError
@@ -63,13 +64,11 @@ class FSLLinearTransformArray(BaseLinearTransformList):
6364

6465
def to_filename(self, filename):
6566
"""Store this transform to a file with the appropriate format."""
66-
if len(self.xforms) == 1:
67-
self.xforms[0].to_filename(filename)
68-
return
69-
67+
output_dir = Path(filename).parent
68+
output_dir.mkdir(exist_ok=True, parents=True)
7069
for i, xfm in enumerate(self.xforms):
71-
with open('%s.%03d' % (filename, i), 'w') as f:
72-
f.write(xfm.to_string())
70+
(output_dir / '.'.join((str(filename), '%03d' % i))).write_text(
71+
xfm.to_string())
7372

7473
def to_ras(self, moving=None, reference=None):
7574
"""Return a nitransforms' internal RAS matrix."""

nitransforms/linear.py

+20-22
Original file line numberDiff line numberDiff line change
@@ -55,19 +55,16 @@ def __init__(self, matrix=None, reference=None):
5555
[0, 0, 0, 1]])
5656
5757
"""
58-
super().__init__()
59-
if matrix is None:
60-
matrix = np.eye(4)
61-
62-
self._matrix = np.array(matrix)
63-
if self._matrix.ndim != 2:
64-
raise TypeError('Affine should be 2D.')
65-
66-
if self._matrix.shape[0] != self._matrix.shape[1]:
67-
raise TypeError('Matrix is not square.')
58+
super().__init__(reference=reference)
59+
self._matrix = np.eye(4)
6860

69-
if reference:
70-
self.reference = reference
61+
if matrix is not None:
62+
matrix = np.array(matrix)
63+
if matrix.ndim != 2:
64+
raise TypeError('Affine should be 2D.')
65+
elif matrix.shape[0] != matrix.shape[1]:
66+
raise TypeError('Matrix is not square.')
67+
self._matrix = matrix
7168

7269
def __eq__(self, other):
7370
"""
@@ -162,7 +159,7 @@ def to_filename(self, filename, fmt='X5', moving=None):
162159
# xform info
163160
lt = io.LinearTransform()
164161
lt['sigma'] = 1.
165-
lt['m_L'] = [self.matrix]
162+
lt['m_L'] = self.matrix
166163
# Just for reference, nitransforms does not write VOX2VOX
167164
lt['src'] = io.VolumeGeometry.from_image(moving)
168165
lt['dst'] = io.VolumeGeometry.from_image(self.reference)
@@ -299,7 +296,7 @@ def map(self, x, inverse=False):
299296

300297
def to_filename(self, filename, fmt='X5', moving=None):
301298
"""Store the transform in BIDS-Transforms HDF5 file format (.x5)."""
302-
if fmt.lower() in ['itk', 'ants', 'elastix']:
299+
if fmt.lower() in ('itk', 'ants', 'elastix'):
303300
itkobj = io.itk.ITKLinearTransformArray.from_ras(self.matrix)
304301
itkobj.to_filename(filename)
305302
return filename
@@ -323,18 +320,19 @@ def to_filename(self, filename, fmt='X5', moving=None):
323320
fslobj.to_filename(filename)
324321
return filename
325322

326-
if fmt.lower() == 'fs':
323+
if fmt.lower() in ('fs', 'lta'):
327324
# xform info
328-
lt = io.LinearTransform()
329-
lt['sigma'] = 1.
330-
lt['m_L'] = self.matrix
331-
# Just for reference, nitransforms does not write VOX2VOX
332-
lt['src'] = io.VolumeGeometry.from_image(moving)
333-
lt['dst'] = io.VolumeGeometry.from_image(self.reference)
334325
# to make LTA file format
335326
lta = io.LinearTransformArray()
336327
lta['type'] = 1 # RAS2RAS
337-
lta['xforms'].append(lt)
328+
for m in self.matrix:
329+
lt = io.LinearTransform()
330+
lt['sigma'] = 1.
331+
lt['m_L'] = m
332+
# Just for reference, nitransforms does not write VOX2VOX
333+
lt['src'] = io.VolumeGeometry.from_image(moving)
334+
lt['dst'] = io.VolumeGeometry.from_image(self.reference)
335+
lta['xforms'].append(lt)
338336

339337
with open(filename, 'w') as f:
340338
f.write(lta.to_string())

nitransforms/tests/test_base.py

+9
Original file line numberDiff line numberDiff line change
@@ -11,13 +11,22 @@ def test_SpatialReference(data_path):
1111
"""Ensure the reference factory is working properly."""
1212
obj1 = data_path / 'someones_anatomy.nii.gz'
1313
obj2 = data_path / 'sub-200148_hemi-R_pial.surf.gii'
14+
obj3 = data_path / 'func.nii.gz'
1415

1516
assert isinstance(SpatialReference.factory(obj1), ImageGrid)
1617
assert isinstance(SpatialReference.factory(str(obj1)), ImageGrid)
1718
assert isinstance(SpatialReference.factory(nb.load(str(obj1))), ImageGrid)
1819
assert isinstance(SpatialReference.factory(obj2), SampledSpatialData)
1920
assert isinstance(SpatialReference.factory(str(obj2)), SampledSpatialData)
2021
assert isinstance(SpatialReference.factory(nb.load(str(obj2))), SampledSpatialData)
22+
assert isinstance(SpatialReference.factory(obj3), ImageGrid)
23+
assert isinstance(SpatialReference.factory(str(obj3)), ImageGrid)
24+
assert isinstance(SpatialReference.factory(nb.load(str(obj3))), ImageGrid)
25+
26+
func_ref = SpatialReference.factory(obj3)
27+
assert func_ref.ndim == 3
28+
assert func_ref.shape == (96, 96, 56)
29+
assert func_ref.npoints == np.prod(func_ref.shape)
2130

2231

2332
@pytest.mark.parametrize('image_orientation', ['RAS', 'LAS', 'LPS', 'oblique'])

nitransforms/tests/test_linear.py

+58-10
Original file line numberDiff line numberDiff line change
@@ -31,28 +31,58 @@
3131
}
3232

3333

34-
def test_linear_typeerrors(data_path):
34+
@pytest.mark.parametrize('matrix', [
35+
[0.0],
36+
np.ones((3, 3, 3)),
37+
np.ones((3, 4)),
38+
])
39+
def test_linear_typeerrors1(matrix):
3540
"""Exercise errors in Affine creation."""
3641
with pytest.raises(TypeError):
37-
ntl.Affine([0.0])
42+
ntl.Affine(matrix)
3843

39-
with pytest.raises(TypeError):
40-
ntl.Affine(np.arange((3, 3, 3)))
41-
42-
with pytest.raises(TypeError):
43-
ntl.Affine(np.arange((3, 4)))
4444

45+
def test_linear_typeerrors2(data_path):
46+
"""Exercise errors in Affine creation."""
4547
with pytest.raises(TypeError):
4648
ntl.Affine.from_filename(data_path / 'itktflist.tfm', fmt='itk')
4749

4850

49-
def test_loadsave(tmp_path, data_path):
51+
def test_loadsave_itk(tmp_path, data_path):
5052
"""Test idempotency."""
53+
ref_file = data_path / 'someones_anatomy.nii.gz'
5154
xfm = ntl.load(data_path / 'itktflist2.tfm', fmt='itk')
52-
xfm.to_filename(tmp_path / 'writtenout.tfm', fmt='itk')
55+
assert isinstance(xfm, ntl.LinearTransformsMapping)
56+
xfm.reference = ref_file
57+
xfm.to_filename(tmp_path / 'transform-mapping.tfm', fmt='itk')
5358

5459
assert (data_path / 'itktflist2.tfm').read_text() \
55-
== (tmp_path / 'writtenout.tfm').read_text()
60+
== (tmp_path / 'transform-mapping.tfm').read_text()
61+
62+
single_xfm = ntl.load(data_path / 'affine-LAS.itk.tfm', fmt='itk')
63+
assert isinstance(single_xfm, ntl.Affine)
64+
assert single_xfm == ntl.Affine.from_filename(
65+
data_path / 'affine-LAS.itk.tfm', fmt='itk')
66+
67+
68+
@pytest.mark.xfail(reason="Not fully implemented")
69+
@pytest.mark.parametrize('fmt', ['itk', 'fsl', 'afni', 'lta'])
70+
def test_loadsave(tmp_path, data_path, fmt):
71+
"""Test idempotency."""
72+
ref_file = data_path / 'someones_anatomy.nii.gz'
73+
xfm = ntl.load(data_path / 'itktflist2.tfm', fmt='itk')
74+
xfm.reference = ref_file
75+
76+
fname = tmp_path / '.'.join(('transform-mapping', fmt))
77+
xfm.to_filename(fname, fmt=fmt)
78+
xfm == ntl.load(fname, fmt=fmt, reference=ref_file)
79+
80+
ref_file = data_path / 'someones_anatomy.nii.gz'
81+
xfm = ntl.load(data_path / 'affine-LAS.itk.tfm', fmt='itk')
82+
xfm.reference = ref_file
83+
fname = tmp_path / '.'.join(('single-transform', fmt))
84+
xfm.to_filename(fname, fmt=fmt)
85+
xfm == ntl.load(fname, fmt=fmt, reference=ref_file)
5686

5787

5888
@pytest.mark.xfail(reason="Not fully implemented")
@@ -146,3 +176,21 @@ def test_concatenation(data_path):
146176
x = [(0., 0., 0.), (1., 1., 1.), (-1., -1., -1.)]
147177
assert np.all((aff + ntl.Affine())(x) == x)
148178
assert np.all((aff + ntl.Affine())(x, inverse=True) == x)
179+
180+
181+
def test_LinearTransformsMapping_apply(tmp_path, data_path):
182+
"""Apply transform mappings."""
183+
hmc = ntl.load(data_path / 'hmc-itk.tfm', fmt='itk',
184+
reference=data_path / 'sbref.nii.gz')
185+
assert isinstance(hmc, ntl.LinearTransformsMapping)
186+
187+
# Test-case: realing functional data on to sbref
188+
nii = hmc.apply(data_path / 'func.nii.gz', order=1)
189+
assert nii.dataobj.shape[-1] == len(hmc)
190+
191+
# Test-case: write out a fieldmap moved with head
192+
hmcinv = ntl.LinearTransformsMapping(
193+
np.linalg.inv(hmc.matrix),
194+
reference=data_path / 'func.nii.gz')
195+
nii = hmcinv.apply(data_path / 'fmap.nii.gz', order=1)
196+
assert nii.dataobj.shape[-1] == len(hmc)

0 commit comments

Comments
 (0)