- 
                Notifications
    You must be signed in to change notification settings 
- Fork 268
RF: Simplify MGHImage and add footer fields #569
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Conversation
5241b1b    to
    832f584      
    Compare
  
    | Codecov Report
 @@            Coverage Diff             @@
##           master     #569      +/-   ##
==========================================
+ Coverage   94.32%   94.41%   +0.09%     
==========================================
  Files         177      177              
  Lines       24697    24884     +187     
  Branches     2640     2656      +16     
==========================================
+ Hits        23295    23495     +200     
+ Misses        925      913      -12     
+ Partials      477      476       -1
 Continue to review full report at Codecov. 
 | 
| This is ready for review, if anybody's got a few minutes. | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Main question - is it really worth renaming Mdc and the other fields?
        
          
                nibabel/freesurfer/mghformat.py
              
                Outdated
          
        
      |  | ||
| def __init__(self, | ||
| binaryblock=None, | ||
| endianness='>', | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Does it make sense to allow passing this flag? It also changes the API.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I did this because of places where LabeledWrapStruct calls __init__ assuming the type signature of the base class. I can instead make sure to override any methods that make such calls, just figured this would be a little shorter.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I played with the idea of making WrapStruct have a parent or similar that had forced endianness, but I seem to remember it got a bit messy.
| return klass(hdr_str + ftr_str, check=check) | ||
|  | ||
| def get_affine(self): | ||
| ''' Get the affine transform from the header information. | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Blank line after first line (sorry, I know that wasn't present before either).
        
          
                nibabel/freesurfer/mghformat.py
              
                Outdated
          
        
      | MdcD = np.hstack((hdr['x_ras'], hdr['y_ras'], hdr['z_ras'])) * hdr['voxelsize'] | ||
| vol_center = MdcD.dot(hdr['dims'][:3].reshape(-1, 1)) / 2 | ||
| affine[:3, :3] = MdcD | ||
| affine[:3, [3]] = hdr['c_ras'] - vol_center | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
3 rather than [3] ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The RHS is a column, so the [3] keeps the LHS expecting a column. Is there a trick I don't know here?
| hdr = self._header_data | ||
| zooms = hdr['delta'] | ||
| return tuple(zooms[:]) | ||
| # Do not return time zoom (TR) if 3D image | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Maybe worth a note in the docstring about the time zoom and 3D?
        
          
                nibabel/freesurfer/mghformat.py
              
                Outdated
          
        
      | ('dims', '>i4', (4,)), # 4; width, height, depth, nframes | ||
| ('type', '>i4'), # 20; data type | ||
| ('dof', '>i4'), # 24; degrees of freedom | ||
| ('ras_good', '>i2'), # 28; *_ras fields valid | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I guess these renamings are API changes - our our Freesurfer colleagues happy with these name changes? Any way to offer the old names as alternatives for a while?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah, I can re-add a _header_data structure (which I've replaced with LabeledWrapStruct._structarr) that will expose the old fields and raise a DeprecationWarning.
        
          
                nibabel/freesurfer/mghformat.py
              
                Outdated
          
        
      | ('dof', '>i4'), # 24; degrees of freedom | ||
| ('ras_good', '>i2'), # 28; *_ras fields valid | ||
| ('voxelsize', '>f4', (3,)), # 30; zooms (X, Y, Z) | ||
| ('x_ras', '>f4', (3, 1)), # 42; X direction cosine column | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
(3, 1) rather than (3,)?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is this split into three columns really useful? Compared to the native 3, 3 size?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
By including all three columns as a single, row-ordered Mdc matrix, we're exposing a transposed matrix. It was being transposed correctly in and out, but if someone does decide to use it directly, they need to know this. I prefer explicit to silently misleading...
I could use (3,) if that's important. But it will mean users knowing that they're columns and not row vectors (can add to the docs).
For context, this is the comparable portion of the C data structure, which is just 16 individual float fields: https://github.com/freesurfer/freesurfer/blob/5367eec84621b331271e97f4e1f1502c3f6d0d87/include/mri.h#L184-L191
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
OK - fine to leave as is (3,1) etc.
        
          
                nibabel/freesurfer/mghformat.py
              
                Outdated
          
        
      | hdr = self._structarr | ||
| MdcD = np.hstack((hdr['x_ras'], hdr['y_ras'], hdr['z_ras'])) * hdr['voxelsize'] | ||
| vol_center = MdcD.dot(hdr['dims'][:3].reshape(-1, 1)) / 2 | ||
| affine[:3, :3] = MdcD | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
return nib.affines.from_matvec(MdcD, hdr['c_ras'] - vol_center)
?
        
          
                nibabel/freesurfer/mghformat.py
              
                Outdated
          
        
      | Ignores byte order; always big endian | ||
| ''' | ||
| structarr = super(MGHHeader, klass).default_structarr() | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Here might be worth checking and error if endian not big.
        
          
                nibabel/freesurfer/mghformat.py
              
                Outdated
          
        
      |  | ||
| # Assign after we've had a chance to raise exceptions | ||
| hdr['voxelsize'] = voxelsize | ||
| hdr['x_ras'] = Mdc[:, [0]] | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
hdr['x_ras'], hdr['y_ras'], hdr['z_ras'] = Mdc.T
?
| Re: Renaming, the biggest motivation is that a lot of the field names seem only to align with one slideshow that documents the spaces/transforms, and not at all with either the FreeSurfer source code or the outputs of  So my strong preference is to move to a more consistent naming scheme (I'm open to alternative suggestions), but I'm also perfectly happy to provide a backwards-compatible interface to access the data using the old field names to get back the old shapes. | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I guess it would be good to get more feedback from Freesurferers out there.
        
          
                nibabel/freesurfer/mghformat.py
              
                Outdated
          
        
      | ('dof', '>i4'), # 24; degrees of freedom | ||
| ('ras_good', '>i2'), # 28; *_ras fields valid | ||
| ('voxelsize', '>f4', (3,)), # 30; zooms (X, Y, Z) | ||
| ('x_ras', '>f4', (3, 1)), # 42; X direction cosine column | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
OK - fine to leave as is (3,1) etc.
        
          
                nibabel/freesurfer/mghformat.py
              
                Outdated
          
        
      |  | ||
| def __init__(self, | ||
| binaryblock=None, | ||
| endianness='>', | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I played with the idea of making WrapStruct have a parent or similar that had forced endianness, but I seem to remember it got a bit messy.
| Okay. I've pinged the Python neuroimaging list and the FreeSurfer support list. Hopefully we can find somebody willing to share their opinions on the Internet. :-) | 
| I don't have a strong opinion here, but I would guess it's more likely that a user has encountered the fscoordinates slide deck than read through the freesurfer source code. | 
| @effigies - perhaps chatting with doug would be a good idea as well. also folks at corticometrics may have some suggestions as well. however, i do not have a strong opinion either. since i rarely do much with those files than reading into an array, doing some processing, and writing. or just visualization. | 
| @satra Can I expect them to respond to the post on the FreeSurfer list, or is there a preferred way of contacting them more directly? | 
| @effigies - i hadn't seen that the mail was on the freesurfer list. hopefully they respond - otherwise it's your call :) | 
| Okay, it sounds like there's a slight preference from @mwaskom to follow the FS documentary slideshow and a preference (of unknown strength) from @matthew-brett to avoid changing names without a Very Good Reason. I'll defer to this for now and revert  However, I do think I would like to move to more "standard" FreeSurfer naming, because this construct ("volume geometry", or dimensions + x/y/z/c_ras) shows up in several places, including  nibabel/nibabel/freesurfer/io.py Lines 50 to 73 in 2139ce0 
 nibabel/nibabel/freesurfer/io.py Lines 462 to 488 in 2139ce0 
 But I think it does make sense to postpone the discussion of field naming to an actual attempt to provide a common interface to volume geometries (which is actually re-implemented in several FreeSurfer data structures). | 
| Chris - sorry - I should have said - I don't care too much about the name changes, especially if there's a deprecation route. I should have said before, but if you think these names are more canonical, that's OK - we can always explain in the docstring and comments what the relationship is to the slide deck. | 
| Okay. I think I'll still push it off for now. If/when we add support for LTA/M3Z etc, it will be time enough to consider a consistent naming scheme, rather than deprecating the old one now and finding later that we would have preferred something slightly different. I will add docs though noting that  | 
| @matthew-brett Got back to this. I think I've addressed all of your comments, and I've fleshed out some more tests. Ready for another review. | 
| Hi @matthew-brett, just bugging you about this. No worries if you're busy. | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Just a few small comments.
        
          
                nibabel/freesurfer/mghformat.py
              
                Outdated
          
        
      | ''' Set zooms into header fields | ||
| See docstring for ``get_zooms`` for examples | ||
| Sets the spaing of voxels in the x, y, and z dimensions. | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Typo 'spacing'
| See docstring for ``get_zooms`` for examples | ||
| Sets the spaing of voxels in the x, y, and z dimensions. | ||
| For four-dimensional files, a temporal zoom (repetition time, or TR, in | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ouch - it's really in milliseconds? That's unfortunate (compared to nifti).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Annoyingly. Fortunately #567 proposes a way to get normalized zooms. (It was working on that that led to this PR in the first place.)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Add reference to docstring?
| Thanks for your patience, in it goes. | 
| Oops - a couple of PEP8 failures : https://travis-ci.org/nipy/nibabel/builds/318192252 | 
This PR includes a number of simplifications to the
MGHHeaderandMGHImagedata structures.The most significant is subclassing
MGHHeaderfromLabeledWrapStruct. It appeared to have been a reimplementation ofLabeledWrapStructin many places without theendiannessoption. I have simply raised an exception on attempts to instantiate as little-endian.The next most significant change is moving from
Mdc(matrix of direction cosines) andPxyz_c(center point in world coordinates) to the more explicitx/y/z/c_rascolumn vectors. These are consistent with most FreeSurfer literature and program outputs, and correctly encode the shape of the direction cosines as column-major rather than row-major (needing to be transposed when reading/writing).Somewhat confusingly, there are two ways of referring to the voxel and world coordinates, and
XYZmeans world in one, and voxel in the other, and in both cases the alternative includes "R" and "S".As CRS/XYZ appears to be FreeSurfer-unique, I'd prefer to settle on XYZ/RAS, and perhaps consider non-XYZ variable names when they can reduce confusion.
TODO:
get_data_shapehandling (see ENH: Add image slicing #550)