forked from danielabler/label-map-warping
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathimage_registration_utils.py
157 lines (145 loc) · 7.52 KB
/
image_registration_utils.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
import os
import shlex
import subprocess
import shutil
import file_utils as fu
def ants_apply_transforms(input_img, reference_img, output_file, transforms, interpolation='Linear', dim=3, shorten_paths=True):
print(" - Starting ANTS Apply Transforms:")
print(" - INPUT IMG : %s"%input_img)
print(" - REFERENCE IMG : %s" % reference_img)
print(" - OUTPUT : %s" % output_file)
if shorten_paths:
img_paths = [input_img, reference_img, output_file]
transform_paths = transforms
#print(img_paths + transform_paths)
common_prefix = os.path.commonprefix(img_paths + transform_paths)
if not len(common_prefix)==0:
common_prefix = os.path.dirname(common_prefix)
input_img, reference_img, output_file = [os.path.relpath(path, common_prefix) for path in img_paths]
transforms = [os.path.relpath(path, common_prefix) for path in transform_paths]
ants_cmd = build_ants_apply_transforms_command(input_img, reference_img, output_file, transforms, interpolation, dim)
print("ANTS command: %s"%ants_cmd)
fu.ensure_dir_exists(os.path.dirname(output_file))
args = shlex.split(ants_cmd)
if shorten_paths:
process = subprocess.Popen(args, env=os.environ.copy(), cwd=common_prefix)
else:
process = subprocess.Popen(args, env=os.environ.copy())
process.wait()
#return process.returncode
print("ANTS apply transforms terminated with return code: '%s'"%process.returncode)
def build_ants_apply_transforms_command(input_img, reference_img, output_file, transforms, interpolation='Linear', dim=3):
ants_params_dict = {'verbose': 0,
'dimensionality': dim,
'input-image-type': 'scalar',
'input' : input_img,
'reference-image': reference_img,
'output': output_file,
'interpolation': interpolation}
ants_params_str = ' --'.join([' '.join([key, str(value)]) for key, value in ants_params_dict.items()])
for transform in transforms:
ants_params_str = ants_params_str + ' --transform %s'%transform
ants_cmd = 'antsApplyTransforms --'+ants_params_str
return ants_cmd
def build_ants_registration_command(fixed_img, moving_img, output_prefix, registration_type='Rigid', image_ext='mha',
fixed_mask=None, moving_mask=None, verbose=0, dim=3):
if registration_type=='Syn':
reg_params = [0.1, 3, 0]
metric = 'CC'+str([fixed_img, moving_img, 1, 4])
convergence = '[100x70x50x20,1e-8,10]'
else:
reg_params = [0.1]
metric = 'MI'+str([fixed_img, moving_img, 1, 32, 'Regular', 0.25])
convergence = '[1000x500x250x100,1e-6,10]'
ants_params_dict = { 'verbose' : verbose,
'dimensionality' : dim,
'output' : [ output_prefix, output_prefix+'.'+image_ext ],
'interpolation' : 'Linear',
'winsorize-image-intensities': [ 0.005 , 0.995 ],
'use-histogram-matching': 1,
'initial-moving-transform': [fixed_img, moving_img, 1],
'transform' : registration_type+str(reg_params),
'metric' : metric,
'convergence' : convergence,
'shrink-factors' : '8x4x2x1',
'smoothing-sigmas': '3x2x1x0vox'}
if fixed_mask and moving_mask:
ants_params_dict['x']='['+fixed_mask+','+moving_mask+']'
elif fixed_mask:
ants_params_dict['x'] = fixed_mask
elif moving_mask:
ants_params_dict['x'] = '[,'+moving_mask+']'
ants_params_str = ' --'.join([' '.join([key, str(value)]) for key, value in ants_params_dict.items()])
ants_cmd = 'antsRegistration --'+ants_params_str
return ants_cmd
def register_ants(fixed_img, moving_img, output_prefix, path_to_transform=None, registration_type='Rigid',
image_ext='mha', fixed_mask=None, moving_mask=None, verbose=0, dim=3, shorten_paths=True):
"""
Note that ANTs cannot handle filepaths beyond a certain length; the 'shorten_paths' flag is meant to take care of
that by shortening all paths relative to the highest level common directory.
If you experience an exitcode '-6' this may be an indication of excessively long filepaths!
"""
print(" - Starting ANTS registration:")
print(" - FIXED IMG : %s"%fixed_img)
print(" - MOVING IMG: %s" % moving_img)
print(" - OUTPUT : %s" % output_prefix)
if shorten_paths:
img_paths = [fixed_img, moving_img, output_prefix]
mask_paths = [path for path in [fixed_mask, moving_mask] if path is not None]
common_prefix = os.path.commonprefix(img_paths + mask_paths)
if not len(common_prefix) == 0:
common_prefix = os.path.dirname(common_prefix)
fixed_img, moving_img, output_prefix_rel = [os.path.relpath(path, common_prefix) for path in img_paths]
if fixed_mask:
fixed_mask = os.path.relpath(fixed_mask, common_prefix)
if moving_mask:
moving_mask = os.path.relpath(moving_mask, common_prefix)
else:
output_prefix_rel = output_prefix
print(fixed_img, moving_img, output_prefix_rel)
print(fixed_mask, moving_mask)
ants_cmd = build_ants_registration_command(fixed_img, moving_img, output_prefix_rel, registration_type, image_ext,
fixed_mask, moving_mask, verbose, dim=dim)
print("ANTS command: %s"%ants_cmd)
fu.ensure_dir_exists(os.path.dirname(output_prefix))
args = shlex.split(ants_cmd)
if shorten_paths:
process = subprocess.Popen(args, env=os.environ.copy(), cwd=common_prefix)
else:
process = subprocess.Popen(args, env=os.environ.copy())
process.wait()
if process.returncode==0 and path_to_transform!=None:
#-- rename trafo file
if registration_type=='Rigid' or registration_type=='Affine':
path_to_transform_ants = output_prefix+'0GenericAffine.mat'
shutil.move(path_to_transform_ants, path_to_transform)
if registration_type=='Syn':
path_to_transform_ants = output_prefix + '1Warp.nii.gz'
shutil.move(path_to_transform_ants, path_to_transform)
print("Registration terminated with return code: '%s'"%process.returncode)
return process.returncode
def register_ants_synquick(fixed_img, moving_img, output_prefix, registration='s', fixed_mask=None, dim=3):
"""
registration:
- r -> rigid
- a -> rigid, affine
- s -> rigid, affine, syn
"""
ants_params_dict = {'d' : dim,
'f': fixed_img,
'm': moving_img,
't': registration,
'o': output_prefix,
'n': 4,
'j': 1,
'z': 0 }
if fixed_mask:
ants_params_dict['x'] = fixed_mask
ants_params_str = ' -'.join([' '.join([key, str(value)]) for key, value in ants_params_dict.items()])
ants_cmd = "%s -"%'antsRegistrationSyNQuick.sh' + ants_params_str
print("ANTS command SYNquick: %s" % ants_cmd)
fu.ensure_dir_exists(os.path.dirname(output_prefix))
args = shlex.split(ants_cmd)
process = subprocess.Popen(args, env=os.environ.copy())
process.wait()
print("ANTS terminated with return code: '%s'" % process.returncode)