__init__.py 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. """
  2. Python Library to Read and Write Surface Files in Freesurfer's TriangularSurface Format
  3. compatible with Freesurfer's MRISwriteTriangularSurface()
  4. https://github.com/freesurfer/freesurfer/blob/release_6_0_0/include/mrisurf.h#L1281
  5. https://github.com/freesurfer/freesurfer/blob/release_6_0_0/utils/mrisurf.c
  6. https://raw.githubusercontent.com/freesurfer/freesurfer/release_6_0_0/utils/mrisurf.c
  7. Freesurfer
  8. https://surfer.nmr.mgh.harvard.edu/
  9. >>> from freesurfer_surface import Surface
  10. >>>
  11. >>> surface = Surface.read_triangular('bert/surf/lh.pial'))
  12. """
  13. import collections
  14. import datetime
  15. import re
  16. import struct
  17. import typing
  18. try:
  19. from freesurfer_surface.version import __version__
  20. except ImportError: # pragma: no cover
  21. __version__ = None
  22. Vertex = collections.namedtuple('Vertex', ['right', 'anterior', 'superior'])
  23. class Surface:
  24. _MAGIC_NUMBER = b'\xff\xff\xfe'
  25. _TAG_CMDLINE = b'\x00\x00\x00\x03'
  26. _TAG_OLD_SURF_GEOM = b'\x00\x00\x00\x14'
  27. _TAG_OLD_USEREALRAS = b'\x00\x00\x00\x02'
  28. # TODO set locale
  29. _DATETIME_FORMAT = '%a %b %d %H:%M:%S %Y'
  30. def __init__(self):
  31. self.creator = None
  32. self.creation_datetime = None
  33. self.vertices = []
  34. self.triangles_vertex_indices = []
  35. # self._triangles = []
  36. self.using_old_real_ras = False
  37. self.volume_geometry_info = None
  38. self.command_lines = []
  39. @classmethod
  40. def _read_cmdlines(cls, stream: typing.BinaryIO) -> typing.Iterator[str]:
  41. while True:
  42. tag = stream.read(4)
  43. if not tag:
  44. return
  45. assert tag == cls._TAG_CMDLINE # might be TAG_GROUP_AVG_SURFACE_AREA
  46. # TAGwrite
  47. # https://github.com/freesurfer/freesurfer/blob/release_6_0_0/utils/tags.c#L94
  48. str_length, = struct.unpack('>Q', stream.read(8))
  49. yield stream.read(str_length - 1)
  50. assert stream.read(1) == b'\x00'
  51. def _read_triangular(self, stream: typing.BinaryIO):
  52. assert stream.read(3) == self._MAGIC_NUMBER
  53. self.creator, creation_dt_str = re.match(rb'^created by (\w+) on (.* \d{4})\n',
  54. stream.readline()).groups()
  55. self.creation_datetime = datetime.datetime.strptime(creation_dt_str.decode(),
  56. self._DATETIME_FORMAT)
  57. assert stream.read(1) == b'\n'
  58. # fwriteInt
  59. # https://github.com/freesurfer/freesurfer/blob/release_6_0_0/utils/fio.c#L290
  60. vertices_num, triangles_num = struct.unpack('>II', stream.read(4 * 2))
  61. self.vertices = [Vertex(*struct.unpack('>fff', stream.read(4 * 3)))
  62. for _ in range(vertices_num)]
  63. self.triangles_vertex_indices = [struct.unpack('>III', stream.read(4 * 3))
  64. for _ in range(triangles_num)]
  65. assert all(vertex_idx < vertices_num
  66. for triangle_vertex_index in self.triangles_vertex_indices
  67. for vertex_idx in triangle_vertex_index)
  68. assert stream.read(4) == self._TAG_OLD_USEREALRAS
  69. using_old_real_ras, = struct.unpack('>I', stream.read(4))
  70. assert using_old_real_ras in [0, 1], using_old_real_ras
  71. self.using_old_real_ras = bool(using_old_real_ras)
  72. assert stream.read(4) == self._TAG_OLD_SURF_GEOM
  73. # writeVolGeom
  74. # https://github.com/freesurfer/freesurfer/blob/release_6_0_0/utils/transform.c#L368
  75. self.volume_geometry_info = [stream.readline() for _ in range(8)]
  76. self.command_lines = list(self._read_cmdlines(stream))
  77. @classmethod
  78. def read_triangular(cls, surface_file_path: str) -> 'Surface':
  79. surface = cls()
  80. with open(surface_file_path, 'rb') as surface_file:
  81. # pylint: disable=protected-access
  82. surface._read_triangular(surface_file)
  83. return surface
  84. def _triangular_creation_datetime_strftime(self) -> bytes:
  85. fmt = self._DATETIME_FORMAT.replace('%d', '{:>2}'.format(self.creation_datetime.day))
  86. return self.creation_datetime.strftime(fmt).encode()
  87. def write_triangular(self, surface_file_path: str,
  88. creation_datetime: typing.Optional[datetime.datetime] = None):
  89. if creation_datetime is None:
  90. creation_datetime = datetime.datetime.now()
  91. with open(surface_file_path, 'wb') as surface_file:
  92. surface_file.write(
  93. self._MAGIC_NUMBER
  94. + b'created by ' + self.creator
  95. + b' on ' + self._triangular_creation_datetime_strftime()
  96. + b'\n\n'
  97. + struct.pack('>II', len(self.vertices), len(self.triangles_vertex_indices))
  98. )
  99. for vertex in self.vertices:
  100. surface_file.write(struct.pack('>fff', *vertex))
  101. for triangle_vertex_indices in self.triangles_vertex_indices:
  102. surface_file.write(struct.pack('>III', *triangle_vertex_indices))
  103. surface_file.write(self._TAG_OLD_USEREALRAS
  104. + struct.pack('>I', 1 if self.using_old_real_ras else 0))
  105. surface_file.write(self._TAG_OLD_SURF_GEOM
  106. + b''.join(self.volume_geometry_info))
  107. for command_line in self.command_lines:
  108. surface_file.write(self._TAG_CMDLINE + struct.pack('>Q', len(command_line) + 1)
  109. + command_line + b'\0')