annotate tomo.py @ 4:7405057bcb29 draft default tip

Uploaded
author rv43
date Thu, 24 Mar 2022 17:02:54 +0000
parents 2718cbde04fe
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
1
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1 #!/usr/bin/env python3
2718cbde04fe Uploaded
rv43
parents:
diff changeset
2
2718cbde04fe Uploaded
rv43
parents:
diff changeset
3 # -*- coding: utf-8 -*-
2718cbde04fe Uploaded
rv43
parents:
diff changeset
4 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
5 Created on Fri Dec 10 09:54:37 2021
2718cbde04fe Uploaded
rv43
parents:
diff changeset
6
2718cbde04fe Uploaded
rv43
parents:
diff changeset
7 @author: rv43
2718cbde04fe Uploaded
rv43
parents:
diff changeset
8 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
9
2718cbde04fe Uploaded
rv43
parents:
diff changeset
10 import logging
2718cbde04fe Uploaded
rv43
parents:
diff changeset
11
2718cbde04fe Uploaded
rv43
parents:
diff changeset
12 import os
2718cbde04fe Uploaded
rv43
parents:
diff changeset
13 import sys
2718cbde04fe Uploaded
rv43
parents:
diff changeset
14 import getopt
2718cbde04fe Uploaded
rv43
parents:
diff changeset
15 import re
2718cbde04fe Uploaded
rv43
parents:
diff changeset
16 import io
2718cbde04fe Uploaded
rv43
parents:
diff changeset
17 import pyinputplus as pyip
2718cbde04fe Uploaded
rv43
parents:
diff changeset
18 import numpy as np
2718cbde04fe Uploaded
rv43
parents:
diff changeset
19 import numexpr as ne
2718cbde04fe Uploaded
rv43
parents:
diff changeset
20 import multiprocessing as mp
2718cbde04fe Uploaded
rv43
parents:
diff changeset
21 import scipy.ndimage as spi
2718cbde04fe Uploaded
rv43
parents:
diff changeset
22 import tomopy
2718cbde04fe Uploaded
rv43
parents:
diff changeset
23 from time import time
2718cbde04fe Uploaded
rv43
parents:
diff changeset
24 from skimage.transform import iradon
2718cbde04fe Uploaded
rv43
parents:
diff changeset
25 from skimage.restoration import denoise_tv_chambolle
2718cbde04fe Uploaded
rv43
parents:
diff changeset
26
2718cbde04fe Uploaded
rv43
parents:
diff changeset
27 import msnc_tools as msnc
2718cbde04fe Uploaded
rv43
parents:
diff changeset
28
2718cbde04fe Uploaded
rv43
parents:
diff changeset
29 class set_numexpr_threads:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
30
2718cbde04fe Uploaded
rv43
parents:
diff changeset
31 def __init__(self, nthreads):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
32 cpu_count = mp.cpu_count()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
33 if nthreads is None or nthreads > cpu_count:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
34 self.n = cpu_count
2718cbde04fe Uploaded
rv43
parents:
diff changeset
35 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
36 self.n = nthreads
2718cbde04fe Uploaded
rv43
parents:
diff changeset
37
2718cbde04fe Uploaded
rv43
parents:
diff changeset
38 def __enter__(self):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
39 self.oldn = ne.set_num_threads(self.n)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
40
2718cbde04fe Uploaded
rv43
parents:
diff changeset
41 def __exit__(self, exc_type, exc_value, traceback):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
42 ne.set_num_threads(self.oldn)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
43
2718cbde04fe Uploaded
rv43
parents:
diff changeset
44 class ConfigTomo(msnc.Config):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
45 """Class for processing a config file.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
46 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
47
2718cbde04fe Uploaded
rv43
parents:
diff changeset
48 def __init__(self, config_file=None, config_dict=None):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
49 super().__init__(config_file, config_dict)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
50
2718cbde04fe Uploaded
rv43
parents:
diff changeset
51 def _validate_txt(self):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
52 """Returns False if any required config parameter is illegal or missing.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
53 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
54 is_valid = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
55
2718cbde04fe Uploaded
rv43
parents:
diff changeset
56 # Check for required first-level keys
2718cbde04fe Uploaded
rv43
parents:
diff changeset
57 pars_required = ['tdf_data_path', 'tbf_data_path', 'detector_id']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
58 pars_missing = []
2718cbde04fe Uploaded
rv43
parents:
diff changeset
59 is_valid = super().validate(pars_required, pars_missing)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
60 if len(pars_missing) > 0:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
61 logging.error(f'Missing item(s) in config file: {", ".join(pars_missing)}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
62 self.detector_id = self.config.get('detector_id')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
63
2718cbde04fe Uploaded
rv43
parents:
diff changeset
64 # Find tomography dark field images file/folder
2718cbde04fe Uploaded
rv43
parents:
diff changeset
65 self.tdf_data_path = self.config.get('tdf_data_path')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
66
2718cbde04fe Uploaded
rv43
parents:
diff changeset
67 # Find tomography bright field images file/folder
2718cbde04fe Uploaded
rv43
parents:
diff changeset
68 self.tbf_data_path = self.config.get('tbf_data_path')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
69
2718cbde04fe Uploaded
rv43
parents:
diff changeset
70 # Check number of tomography image stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
71 self.num_tomo_stacks = self.config.get('num_tomo_stacks', 1)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
72 if not msnc.is_int(self.num_tomo_stacks, 1):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
73 self.num_tomo_stacks = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
74 msnc.illegal_value('num_tomo_stacks', self.num_tomo_stacks, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
75 return False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
76 logging.info(f'num_tomo_stacks = {self.num_tomo_stacks}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
77
2718cbde04fe Uploaded
rv43
parents:
diff changeset
78 # Find tomography images file/folders and stack parameters
2718cbde04fe Uploaded
rv43
parents:
diff changeset
79 tomo_data_paths_indices = sorted({key:value for key,value in self.config.items()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
80 if 'tomo_data_path' in key}.items())
2718cbde04fe Uploaded
rv43
parents:
diff changeset
81 if len(tomo_data_paths_indices) != self.num_tomo_stacks:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
82 logging.error(f'Incorrect number of tomography data path names in config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
83 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
84 self.tomo_data_paths = [tomo_data_paths_indices[i][1] for i in range(self.num_tomo_stacks)]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
85 self.tomo_data_indices = [msnc.get_trailing_int(tomo_data_paths_indices[i][0])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
86 if msnc.get_trailing_int(tomo_data_paths_indices[i][0]) else None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
87 for i in range(self.num_tomo_stacks)]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
88 tomo_ref_height_indices = sorted({key:value for key,value in self.config.items()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
89 if 'z_pos' in key}.items())
2718cbde04fe Uploaded
rv43
parents:
diff changeset
90 if self.num_tomo_stacks > 1 and len(tomo_ref_height_indices) != self.num_tomo_stacks:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
91 logging.error(f'Incorrect number of tomography reference heights in config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
92 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
93 if len(tomo_ref_height_indices):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
94 self.tomo_ref_heights = [
2718cbde04fe Uploaded
rv43
parents:
diff changeset
95 tomo_ref_height_indices[i][1] for i in range(self.num_tomo_stacks)]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
96 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
97 self.tomo_ref_heights = [0.0]*self.num_tomo_stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
98
2718cbde04fe Uploaded
rv43
parents:
diff changeset
99 # Check tomo angle (theta) range
2718cbde04fe Uploaded
rv43
parents:
diff changeset
100 self.start_theta = self.config.get('start_theta', 0.)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
101 if not msnc.is_num(self.start_theta, 0.):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
102 msnc.illegal_value('start_theta', self.start_theta, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
103 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
104 logging.debug(f'start_theta = {self.start_theta}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
105 self.end_theta = self.config.get('end_theta', 180.)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
106 if not msnc.is_num(self.end_theta, self.start_theta):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
107 msnc.illegal_value('end_theta', self.end_theta, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
108 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
109 logging.debug(f'end_theta = {self.end_theta}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
110 self.num_thetas = self.config.get('num_thetas')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
111 if not (self.num_thetas is None or msnc.is_int(self.num_thetas, 1)):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
112 msnc.illegal_value('num_thetas', self.num_thetas, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
113 self.num_thetas = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
114 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
115 logging.debug(f'num_thetas = {self.num_thetas}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
116
2718cbde04fe Uploaded
rv43
parents:
diff changeset
117 return is_valid
2718cbde04fe Uploaded
rv43
parents:
diff changeset
118
2718cbde04fe Uploaded
rv43
parents:
diff changeset
119 def _validate_yaml(self):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
120 """Returns False if any required config parameter is illegal or missing.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
121 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
122 is_valid = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
123
2718cbde04fe Uploaded
rv43
parents:
diff changeset
124 # Check for required first-level keys
2718cbde04fe Uploaded
rv43
parents:
diff changeset
125 pars_required = ['dark_field', 'bright_field', 'stack_info', 'detector']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
126 pars_missing = []
2718cbde04fe Uploaded
rv43
parents:
diff changeset
127 is_valid = super().validate(pars_required, pars_missing)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
128 if len(pars_missing) > 0:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
129 logging.error(f'Missing item(s) in config file: {", ".join(pars_missing)}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
130 self.detector_id = self.config['detector'].get('id')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
131
2718cbde04fe Uploaded
rv43
parents:
diff changeset
132 # Find tomography dark field images file/folder
2718cbde04fe Uploaded
rv43
parents:
diff changeset
133 self.tdf_data_path = self.config['dark_field'].get('data_path')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
134
2718cbde04fe Uploaded
rv43
parents:
diff changeset
135 # Find tomography bright field images file/folder
2718cbde04fe Uploaded
rv43
parents:
diff changeset
136 self.tbf_data_path = self.config['bright_field'].get('data_path')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
137
2718cbde04fe Uploaded
rv43
parents:
diff changeset
138 # Check number of tomography image stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
139 stack_info = self.config['stack_info']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
140 self.num_tomo_stacks = stack_info.get('num', 1)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
141 if not msnc.is_int(self.num_tomo_stacks, 1):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
142 self.num_tomo_stacks = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
143 msnc.illegal_value('stack_info:num', self.num_tomo_stacks, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
144 return False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
145 logging.info(f'num_tomo_stacks = {self.num_tomo_stacks}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
146
2718cbde04fe Uploaded
rv43
parents:
diff changeset
147 # Find tomography images file/folders and stack parameters
2718cbde04fe Uploaded
rv43
parents:
diff changeset
148 stacks = stack_info.get('stacks')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
149 if stacks is None or len(stacks) is not self.num_tomo_stacks:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
150 msnc.illegal_value('stack_info:stacks', stacks, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
151 return False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
152 self.tomo_data_paths = []
2718cbde04fe Uploaded
rv43
parents:
diff changeset
153 self.tomo_data_indices = []
2718cbde04fe Uploaded
rv43
parents:
diff changeset
154 self.tomo_ref_heights = []
2718cbde04fe Uploaded
rv43
parents:
diff changeset
155 for stack in stacks:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
156 self.tomo_data_paths.append(stack.get('data_path'))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
157 self.tomo_data_indices.append(stack.get('index'))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
158 self.tomo_ref_heights.append(stack.get('ref_height'))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
159
2718cbde04fe Uploaded
rv43
parents:
diff changeset
160 # Check tomo angle (theta) range
2718cbde04fe Uploaded
rv43
parents:
diff changeset
161 theta_range = self.config.get('theta_range')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
162 if theta_range is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
163 self.start_theta = 0.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
164 self.end_theta = 180.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
165 self.num_thetas = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
166 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
167 self.start_theta = theta_range.get('start', 0.)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
168 if not msnc.is_num(self.start_theta, 0.):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
169 msnc.illegal_value('theta_range:start', self.start_theta, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
170 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
171 logging.debug(f'start_theta = {self.start_theta}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
172 self.end_theta = theta_range.get('end', 180.)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
173 if not msnc.is_num(self.end_theta, self.start_theta):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
174 msnc.illegal_value('theta_range:end', self.end_theta, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
175 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
176 logging.debug(f'end_theta = {self.end_theta}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
177 self.num_thetas = theta_range.get('num')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
178 if self.num_thetas and not msnc.is_int(self.num_thetas, 1):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
179 msnc.illegal_value('theta_range:num', self.num_thetas, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
180 self.num_thetas = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
181 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
182 logging.debug(f'num_thetas = {self.num_thetas}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
183
2718cbde04fe Uploaded
rv43
parents:
diff changeset
184 return is_valid
2718cbde04fe Uploaded
rv43
parents:
diff changeset
185
2718cbde04fe Uploaded
rv43
parents:
diff changeset
186 def validate(self):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
187 """Returns False if any required config parameter is illegal or missing.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
188 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
189 is_valid = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
190
2718cbde04fe Uploaded
rv43
parents:
diff changeset
191 # Check work_folder (shared by both file formats)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
192 work_folder = os.path.abspath(self.config.get('work_folder', ''))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
193 if not os.path.isdir(work_folder):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
194 msnc.illegal_value('work_folder', work_folder, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
195 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
196 logging.info(f'work_folder: {work_folder}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
197
2718cbde04fe Uploaded
rv43
parents:
diff changeset
198 # Check data filetype (shared by both file formats)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
199 self.data_filetype = self.config.get('data_filetype', 'tif')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
200 if not isinstance(self.data_filetype, str) or (self.data_filetype != 'tif' and
2718cbde04fe Uploaded
rv43
parents:
diff changeset
201 self.data_filetype != 'h5'):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
202 msnc.illegal_value('data_filetype', self.data_filetype, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
203
2718cbde04fe Uploaded
rv43
parents:
diff changeset
204 if self.suffix == '.yml' or self.suffix == '.yaml':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
205 is_valid = self._validate_yaml()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
206 elif self.suffix == '.txt':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
207 is_valid = self._validate_txt()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
208 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
209 logging.error(f'Undefined or illegal config file extension: {self.suffix}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
210
2718cbde04fe Uploaded
rv43
parents:
diff changeset
211 # Find tomography bright field images file/folder
2718cbde04fe Uploaded
rv43
parents:
diff changeset
212 if self.tdf_data_path:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
213 if self.data_filetype == 'h5':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
214 if isinstance(self.tdf_data_path, str):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
215 if not os.path.isabs(self.tdf_data_path):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
216 self.tdf_data_path = os.path.abspath(
2718cbde04fe Uploaded
rv43
parents:
diff changeset
217 f'{work_folder}/{self.tdf_data_path}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
218 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
219 msnc.illegal_value('tdf_data_path', tdf_data_fil, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
220 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
221 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
222 if isinstance(self.tdf_data_path, int):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
223 self.tdf_data_path = os.path.abspath(
2718cbde04fe Uploaded
rv43
parents:
diff changeset
224 f'{work_folder}/{self.tdf_data_path}/nf')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
225 elif isinstance(self.tdf_data_path, str):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
226 if not os.path.isabs(self.tdf_data_path):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
227 self.tdf_data_path = os.path.abspath(
2718cbde04fe Uploaded
rv43
parents:
diff changeset
228 f'{work_folder}/{self.tdf_data_path}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
229 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
230 msnc.illegal_value('tdf_data_path', self.tdf_data_path, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
231 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
232 logging.info(f'dark field images path = {self.tdf_data_path}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
233
2718cbde04fe Uploaded
rv43
parents:
diff changeset
234 # Find tomography bright field images file/folder
2718cbde04fe Uploaded
rv43
parents:
diff changeset
235 if self.tbf_data_path:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
236 if self.data_filetype == 'h5':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
237 if isinstance(self.tbf_data_path, str):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
238 if not os.path.isabs(self.tbf_data_path):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
239 self.tbf_data_path = os.path.abspath(
2718cbde04fe Uploaded
rv43
parents:
diff changeset
240 f'{work_folder}/{self.tbf_data_path}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
241 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
242 msnc.illegal_value('tbf_data_path', tbf_data_fil, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
243 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
244 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
245 if isinstance(self.tbf_data_path, int):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
246 self.tbf_data_path = os.path.abspath(
2718cbde04fe Uploaded
rv43
parents:
diff changeset
247 f'{work_folder}/{self.tbf_data_path}/nf')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
248 elif isinstance(self.tbf_data_path, str):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
249 if not os.path.isabs(self.tbf_data_path):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
250 self.tbf_data_path = os.path.abspath(
2718cbde04fe Uploaded
rv43
parents:
diff changeset
251 f'{work_folder}/{self.tbf_data_path}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
252 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
253 msnc.illegal_value('tbf_data_path', self.tbf_data_path, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
254 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
255 logging.info(f'bright field images path = {self.tbf_data_path}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
256
2718cbde04fe Uploaded
rv43
parents:
diff changeset
257 # Find tomography images file/folders and stack parameters
2718cbde04fe Uploaded
rv43
parents:
diff changeset
258 tomo_data_paths = []
2718cbde04fe Uploaded
rv43
parents:
diff changeset
259 tomo_data_indices = []
2718cbde04fe Uploaded
rv43
parents:
diff changeset
260 tomo_ref_heights = []
2718cbde04fe Uploaded
rv43
parents:
diff changeset
261 for data_path, index, ref_height in zip(self.tomo_data_paths, self.tomo_data_indices,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
262 self.tomo_ref_heights):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
263 if self.data_filetype == 'h5':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
264 if isinstance(data_path, str):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
265 if not os.path.isabs(data_path):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
266 data_path = os.path.abspath(f'{work_folder}/{data_path}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
267 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
268 msnc.illegal_value(f'stack_info:stacks:data_path', data_path, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
269 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
270 data_path = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
271 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
272 if isinstance(data_path, int):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
273 data_path = os.path.abspath(f'{work_folder}/{data_path}/nf')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
274 elif isinstance(data_path, str):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
275 if not os.path.isabs(data_path):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
276 data_path = os.path.abspath(f'{work_folder}/{data_path}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
277 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
278 msnc.illegal_value(f'stack_info:stacks:data_path', data_path, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
279 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
280 data_path = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
281 tomo_data_paths.append(data_path)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
282 if index is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
283 if self.num_tomo_stacks > 1:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
284 logging.error('Missing stack_info:stacks:index in config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
285 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
286 index = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
287 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
288 index = 1
2718cbde04fe Uploaded
rv43
parents:
diff changeset
289 elif not isinstance(index, int):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
290 msnc.illegal_value(f'stack_info:stacks:index', index, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
291 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
292 index = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
293 tomo_data_indices.append(index)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
294 if ref_height is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
295 if self.num_tomo_stacks > 1:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
296 logging.error('Missing stack_info:stacks:ref_height in config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
297 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
298 ref_height = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
299 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
300 ref_height = 0.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
301 elif not msnc.is_num(ref_height):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
302 msnc.illegal_value(f'stack_info:stacks:ref_height', ref_height, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
303 is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
304 ref_height = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
305 # Set reference heights relative to first stack
2718cbde04fe Uploaded
rv43
parents:
diff changeset
306 if (len(tomo_ref_heights) and msnc.is_num(ref_height) and
2718cbde04fe Uploaded
rv43
parents:
diff changeset
307 msnc.is_num(tomo_ref_heights[0])):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
308 ref_height = (round(ref_height-tomo_ref_heights[0], 3))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
309 tomo_ref_heights.append(ref_height)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
310 tomo_ref_heights[0] = 0.0
2718cbde04fe Uploaded
rv43
parents:
diff changeset
311 logging.info('tomography data paths:')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
312 for i in range(self.num_tomo_stacks):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
313 logging.info(f' {tomo_data_paths[i]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
314 logging.info(f'tomography data path indices: {tomo_data_indices}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
315 logging.info(f'tomography reference heights: {tomo_ref_heights}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
316
2718cbde04fe Uploaded
rv43
parents:
diff changeset
317 # Update config in memory
2718cbde04fe Uploaded
rv43
parents:
diff changeset
318 if self.suffix == '.txt':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
319 self.config = {}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
320 dark_field = self.config.get('dark_field')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
321 if dark_field is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
322 self.config['dark_field'] = {'data_path' : self.tdf_data_path}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
323 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
324 self.config['dark_field']['data_path'] = self.tdf_data_path
2718cbde04fe Uploaded
rv43
parents:
diff changeset
325 bright_field = self.config.get('bright_field')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
326 if bright_field is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
327 self.config['bright_field'] = {'data_path' : self.tbf_data_path}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
328 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
329 self.config['bright_field']['data_path'] = self.tbf_data_path
2718cbde04fe Uploaded
rv43
parents:
diff changeset
330 detector = self.config.get('detector')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
331 if detector is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
332 self.config['detector'] = {'id' : self.detector_id}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
333 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
334 detector['id'] = self.detector_id
2718cbde04fe Uploaded
rv43
parents:
diff changeset
335 self.config['work_folder'] = work_folder
2718cbde04fe Uploaded
rv43
parents:
diff changeset
336 self.config['data_filetype'] = self.data_filetype
2718cbde04fe Uploaded
rv43
parents:
diff changeset
337 stack_info = self.config.get('stack_info')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
338 if stack_info is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
339 stacks = []
2718cbde04fe Uploaded
rv43
parents:
diff changeset
340 for i in range(self.num_tomo_stacks):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
341 stacks.append({'data_path' : tomo_data_paths[i], 'index' : tomo_data_indices[i],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
342 'ref_height' : tomo_ref_heights[i]})
2718cbde04fe Uploaded
rv43
parents:
diff changeset
343 self.config['stack_info'] = {'num' : self.num_tomo_stacks, 'stacks' : stacks}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
344 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
345 stack_info['num'] = self.num_tomo_stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
346 stacks = stack_info.get('stacks')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
347 for i,stack in enumerate(stacks):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
348 stack['data_path'] = tomo_data_paths[i]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
349 stack['index'] = tomo_data_indices[i]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
350 stack['ref_height'] = tomo_ref_heights[i]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
351 if self.num_thetas:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
352 theta_range = {'start' : self.start_theta, 'end' : self.end_theta,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
353 'num' : self.num_thetas}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
354 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
355 theta_range = {'start' : self.start_theta, 'end' : self.end_theta}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
356 self.config['theta_range'] = theta_range
2718cbde04fe Uploaded
rv43
parents:
diff changeset
357
2718cbde04fe Uploaded
rv43
parents:
diff changeset
358 # Cleanup temporary validation variables
2718cbde04fe Uploaded
rv43
parents:
diff changeset
359 del self.tdf_data_path
2718cbde04fe Uploaded
rv43
parents:
diff changeset
360 del self.tbf_data_path
2718cbde04fe Uploaded
rv43
parents:
diff changeset
361 del self.detector_id
2718cbde04fe Uploaded
rv43
parents:
diff changeset
362 del self.data_filetype
2718cbde04fe Uploaded
rv43
parents:
diff changeset
363 del self.num_tomo_stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
364 del self.tomo_data_paths
2718cbde04fe Uploaded
rv43
parents:
diff changeset
365 del self.tomo_data_indices
2718cbde04fe Uploaded
rv43
parents:
diff changeset
366 del self.tomo_ref_heights
2718cbde04fe Uploaded
rv43
parents:
diff changeset
367 del self.start_theta
2718cbde04fe Uploaded
rv43
parents:
diff changeset
368 del self.end_theta
2718cbde04fe Uploaded
rv43
parents:
diff changeset
369 del self.num_thetas
2718cbde04fe Uploaded
rv43
parents:
diff changeset
370
2718cbde04fe Uploaded
rv43
parents:
diff changeset
371 return is_valid
2718cbde04fe Uploaded
rv43
parents:
diff changeset
372
2718cbde04fe Uploaded
rv43
parents:
diff changeset
373 class Tomo:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
374 """Processing tomography data with misalignment.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
375 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
376
2718cbde04fe Uploaded
rv43
parents:
diff changeset
377 def __init__(self, config_file=None, config_dict=None, config_out=None, output_folder='.',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
378 log_level='INFO', log_stream='tomo.log', galaxy_flag=False, test_mode=False):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
379 """Initialize with optional config input file or dictionary
2718cbde04fe Uploaded
rv43
parents:
diff changeset
380 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
381 self.ncore = mp.cpu_count()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
382 self.config_out = config_out
2718cbde04fe Uploaded
rv43
parents:
diff changeset
383 self.output_folder = output_folder
2718cbde04fe Uploaded
rv43
parents:
diff changeset
384 self.galaxy_flag = galaxy_flag
2718cbde04fe Uploaded
rv43
parents:
diff changeset
385 self.test_mode = test_mode
2718cbde04fe Uploaded
rv43
parents:
diff changeset
386 self.save_plots = True # Make input argument?
2718cbde04fe Uploaded
rv43
parents:
diff changeset
387 self.save_plots_only = True # Make input argument?
2718cbde04fe Uploaded
rv43
parents:
diff changeset
388 self.cf = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
389 self.config = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
390 self.is_valid = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
391 self.tdf = np.array([])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
392 self.tbf = np.array([])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
393 self.tomo_stacks = []
2718cbde04fe Uploaded
rv43
parents:
diff changeset
394 self.tomo_recon_stacks = []
2718cbde04fe Uploaded
rv43
parents:
diff changeset
395
2718cbde04fe Uploaded
rv43
parents:
diff changeset
396 # Set log configuration
2718cbde04fe Uploaded
rv43
parents:
diff changeset
397 logging_format = '%(asctime)s : %(levelname)s - %(module)s : %(funcName)s - %(message)s'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
398 if self.test_mode:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
399 self.save_plots_only = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
400 if isinstance(log_stream, str):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
401 logging.basicConfig(filename=f'{log_stream}', filemode='w',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
402 format=logging_format, level=logging.WARNING, force=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
403 elif isinstance(log_stream, io.TextIOWrapper):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
404 logging.basicConfig(filemode='w', format=logging_format, level=logging.WARNING,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
405 stream=log_stream, force=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
406 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
407 raise ValueError(f'Invalid log_stream: {log_stream}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
408 logging.warning('Ignoring log_level argument in test mode')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
409 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
410 level = getattr(logging, log_level.upper(), None)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
411 if not isinstance(level, int):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
412 raise ValueError(f'Invalid log_level: {log_level}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
413 if log_stream is sys.stdout:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
414 logging.basicConfig(format=logging_format, level=level, force=True,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
415 handlers=[logging.StreamHandler()])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
416 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
417 if isinstance(log_stream, str):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
418 logging.basicConfig(filename=f'{log_stream}', filemode='w',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
419 format=logging_format, level=level, force=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
420 elif isinstance(log_stream, io.TextIOWrapper):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
421 logging.basicConfig(filemode='w', format=logging_format, level=level,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
422 stream=log_stream, force=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
423 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
424 raise ValueError(f'Invalid log_stream: {log_stream}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
425 stream_handler = logging.StreamHandler()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
426 logging.getLogger().addHandler(stream_handler)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
427 stream_handler.setLevel(logging.WARNING)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
428 stream_handler.setFormatter(logging.Formatter(logging_format))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
429
2718cbde04fe Uploaded
rv43
parents:
diff changeset
430 # Set output config file name
2718cbde04fe Uploaded
rv43
parents:
diff changeset
431 if self.config_out is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
432 if self.config is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
433 self.config_out = 'config.yaml'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
434 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
435 self.config_out = config_file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
436
2718cbde04fe Uploaded
rv43
parents:
diff changeset
437 logging.info(f'ncore = {self.ncore}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
438 logging.debug(f'config_file = {config_file}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
439 logging.debug(f'config_dict = {config_dict}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
440 logging.debug(f'config_out = {self.config_out}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
441 logging.debug(f'output_folder = {self.output_folder}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
442 logging.debug(f'log_stream = {log_stream}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
443 logging.debug(f'log_level = {log_level}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
444 logging.debug(f'galaxy_flag = {self.galaxy_flag}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
445 logging.debug(f'test_mode = {self.test_mode}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
446
2718cbde04fe Uploaded
rv43
parents:
diff changeset
447 # Create config object and load config file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
448 self.cf = ConfigTomo(config_file, config_dict)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
449 if not self.cf.load_flag:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
450 self.is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
451 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
452
2718cbde04fe Uploaded
rv43
parents:
diff changeset
453 if self.galaxy_flag:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
454 self.ncore = 1 #RV can I set this? mp.cpu_count()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
455 assert(self.output_folder == '.')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
456 assert(self.test_mode is False)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
457 self.save_plots = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
458 self.save_plots_only = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
459 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
460 # Input validation is already performed during link_data_to_galaxy
2718cbde04fe Uploaded
rv43
parents:
diff changeset
461
2718cbde04fe Uploaded
rv43
parents:
diff changeset
462 # Check config file parameters
2718cbde04fe Uploaded
rv43
parents:
diff changeset
463 self.is_valid = self.cf.validate()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
464
2718cbde04fe Uploaded
rv43
parents:
diff changeset
465 # Load detector info file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
466 df = msnc.Detector(self.cf.config['detector']['id'])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
467
2718cbde04fe Uploaded
rv43
parents:
diff changeset
468 # Check detector info file parameters
2718cbde04fe Uploaded
rv43
parents:
diff changeset
469 if df.validate():
2718cbde04fe Uploaded
rv43
parents:
diff changeset
470 pixel_size = df.getPixelSize()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
471 num_rows, num_columns = df.getDimensions()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
472 if not pixel_size or not num_rows or not num_columns:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
473 self.is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
474 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
475 pixel_size = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
476 num_rows = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
477 num_columns = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
478 self.is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
479
2718cbde04fe Uploaded
rv43
parents:
diff changeset
480 # Update config
2718cbde04fe Uploaded
rv43
parents:
diff changeset
481 self.cf.config['detector']['pixel_size'] = pixel_size
2718cbde04fe Uploaded
rv43
parents:
diff changeset
482 self.cf.config['detector']['rows'] = num_rows
2718cbde04fe Uploaded
rv43
parents:
diff changeset
483 self.cf.config['detector']['columns'] = num_columns
2718cbde04fe Uploaded
rv43
parents:
diff changeset
484 logging.debug(f'pixel_size = self.cf.config["detector"]["pixel_size"]')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
485 logging.debug(f'num_rows: {self.cf.config["detector"]["rows"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
486 logging.debug(f'num_columns: {self.cf.config["detector"]["columns"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
487
2718cbde04fe Uploaded
rv43
parents:
diff changeset
488 # Safe config to file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
489 if self.is_valid:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
490 self.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
491
2718cbde04fe Uploaded
rv43
parents:
diff changeset
492 # Initialize shortcut to config
2718cbde04fe Uploaded
rv43
parents:
diff changeset
493 self.config = self.cf.config
2718cbde04fe Uploaded
rv43
parents:
diff changeset
494
2718cbde04fe Uploaded
rv43
parents:
diff changeset
495 # Initialize tomography stack
2718cbde04fe Uploaded
rv43
parents:
diff changeset
496 num_tomo_stacks = self.config['stack_info']['num']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
497 if num_tomo_stacks:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
498 self.tomo_stacks = [np.array([]) for _ in range(num_tomo_stacks)]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
499 self.tomo_recon_stacks = [np.array([]) for _ in range(num_tomo_stacks)]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
500
2718cbde04fe Uploaded
rv43
parents:
diff changeset
501 logging.debug(f'save_plots = {self.save_plots}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
502 logging.debug(f'save_plots_only = {self.save_plots_only}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
503
2718cbde04fe Uploaded
rv43
parents:
diff changeset
504 def findImageFiles(self):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
505 """Find all available image files.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
506 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
507 self.is_valid = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
508
2718cbde04fe Uploaded
rv43
parents:
diff changeset
509 # Find dark field images
2718cbde04fe Uploaded
rv43
parents:
diff changeset
510 dark_field = self.config['dark_field']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
511 img_start, num_imgs, dark_files = msnc.findImageFiles(
2718cbde04fe Uploaded
rv43
parents:
diff changeset
512 dark_field['data_path'], self.config['data_filetype'], 'dark field')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
513 if img_start < 0 or num_imgs < 1:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
514 logging.error('Unable to find suitable dark field images')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
515 if dark_field['data_path']:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
516 self.is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
517 dark_field['num'] = num_imgs
2718cbde04fe Uploaded
rv43
parents:
diff changeset
518 dark_field['img_start'] = img_start
2718cbde04fe Uploaded
rv43
parents:
diff changeset
519 logging.info(f'Number of dark field images = {dark_field["num"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
520 logging.info(f'Dark field image start index = {dark_field["img_start"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
521
2718cbde04fe Uploaded
rv43
parents:
diff changeset
522 # Find bright field images
2718cbde04fe Uploaded
rv43
parents:
diff changeset
523 bright_field = self.config['bright_field']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
524 img_start, num_imgs, bright_files = msnc.findImageFiles(
2718cbde04fe Uploaded
rv43
parents:
diff changeset
525 bright_field['data_path'], self.config['data_filetype'], 'bright field')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
526 if img_start < 0 or num_imgs < 1:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
527 logging.error('Unable to find suitable bright field images')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
528 self.is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
529 bright_field['num'] = num_imgs
2718cbde04fe Uploaded
rv43
parents:
diff changeset
530 bright_field['img_start'] = img_start
2718cbde04fe Uploaded
rv43
parents:
diff changeset
531 logging.info(f'Number of bright field images = {bright_field["num"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
532 logging.info(f'Bright field image start index = {bright_field["img_start"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
533
2718cbde04fe Uploaded
rv43
parents:
diff changeset
534 # Find tomography images
2718cbde04fe Uploaded
rv43
parents:
diff changeset
535 tomo_stack_files = []
2718cbde04fe Uploaded
rv43
parents:
diff changeset
536 for stack in self.config['stack_info']['stacks']:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
537 index = stack['index']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
538 img_start, num_imgs, tomo_files = msnc.findImageFiles(
2718cbde04fe Uploaded
rv43
parents:
diff changeset
539 stack['data_path'], self.config['data_filetype'], f'tomography set {index}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
540 if img_start < 0 or num_imgs < 1:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
541 logging.error('Unable to find suitable tomography images')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
542 self.is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
543 stack['num'] = num_imgs
2718cbde04fe Uploaded
rv43
parents:
diff changeset
544 stack['img_start'] = img_start
2718cbde04fe Uploaded
rv43
parents:
diff changeset
545 logging.info(f'Number of tomography images for set {index} = {stack["num"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
546 logging.info(f'Tomography set {index} image start index = {stack["img_start"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
547 tomo_stack_files.append(tomo_files)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
548 del tomo_files
2718cbde04fe Uploaded
rv43
parents:
diff changeset
549
2718cbde04fe Uploaded
rv43
parents:
diff changeset
550 # Safe updated config
2718cbde04fe Uploaded
rv43
parents:
diff changeset
551 if self.is_valid:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
552 self.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
553
2718cbde04fe Uploaded
rv43
parents:
diff changeset
554 return dark_files, bright_files, tomo_stack_files
2718cbde04fe Uploaded
rv43
parents:
diff changeset
555
2718cbde04fe Uploaded
rv43
parents:
diff changeset
556 def selectImageRanges(self, available_stacks=None):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
557 """Find and check all required image files.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
558 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
559 self.is_valid = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
560 stack_info = self.config['stack_info']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
561 if available_stacks is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
562 available_stacks = [False]*stack_info['num']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
563 elif len(available_stacks) != stack_info['num']:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
564 logging.warning('Illegal dimension of available_stacks in getImageFiles '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
565 f'({len(available_stacks)}');
2718cbde04fe Uploaded
rv43
parents:
diff changeset
566 available_stacks = [False]*stack_info['num']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
567
2718cbde04fe Uploaded
rv43
parents:
diff changeset
568 # Check number of tomography angles/thetas
2718cbde04fe Uploaded
rv43
parents:
diff changeset
569 num_thetas = self.config['theta_range'].get('num')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
570 if num_thetas is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
571 num_thetas = pyip.inputInt('\nEnter the number of thetas (>0): ', greaterThan=0)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
572 elif not msnc.is_int(num_thetas, 0):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
573 msnc.illegal_value('num_thetas', num_thetas, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
574 self.is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
575 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
576 self.config['theta_range']['num'] = num_thetas
2718cbde04fe Uploaded
rv43
parents:
diff changeset
577 logging.debug(f'num_thetas = {self.config["theta_range"]["num"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
578
2718cbde04fe Uploaded
rv43
parents:
diff changeset
579 # Find tomography dark field images
2718cbde04fe Uploaded
rv43
parents:
diff changeset
580 dark_field = self.config['dark_field']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
581 img_start = dark_field.get('img_start', -1)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
582 img_offset = dark_field.get('img_offset', -1)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
583 num_imgs = dark_field.get('num', 0)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
584 if not self.test_mode:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
585 img_start, img_offset, num_imgs = msnc.selectImageRange(img_start, img_offset,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
586 num_imgs, 'dark field')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
587 if img_start < 0 or num_imgs < 1:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
588 logging.error('Unable to find suitable dark field images')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
589 if dark_field['data_path']:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
590 self.is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
591 dark_field['img_start'] = img_start
2718cbde04fe Uploaded
rv43
parents:
diff changeset
592 dark_field['img_offset'] = img_offset
2718cbde04fe Uploaded
rv43
parents:
diff changeset
593 dark_field['num'] = num_imgs
2718cbde04fe Uploaded
rv43
parents:
diff changeset
594 logging.debug(f'Dark field image start index: {dark_field["img_start"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
595 logging.debug(f'Dark field image offset: {dark_field["img_offset"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
596 logging.debug(f'Number of dark field images: {dark_field["num"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
597
2718cbde04fe Uploaded
rv43
parents:
diff changeset
598 # Find tomography bright field images
2718cbde04fe Uploaded
rv43
parents:
diff changeset
599 bright_field = self.config['bright_field']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
600 img_start = bright_field.get('img_start', -1)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
601 img_offset = bright_field.get('img_offset', -1)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
602 num_imgs = bright_field.get('num', 0)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
603 if not self.test_mode:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
604 img_start, img_offset, num_imgs = msnc.selectImageRange(img_start, img_offset,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
605 num_imgs, 'bright field')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
606 if img_start < 0 or num_imgs < 1:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
607 logging.error('Unable to find suitable bright field images')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
608 if bright_field['data_path']:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
609 self.is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
610 bright_field['img_start'] = img_start
2718cbde04fe Uploaded
rv43
parents:
diff changeset
611 bright_field['img_offset'] = img_offset
2718cbde04fe Uploaded
rv43
parents:
diff changeset
612 bright_field['num'] = num_imgs
2718cbde04fe Uploaded
rv43
parents:
diff changeset
613 logging.debug(f'Bright field image start index: {bright_field["img_start"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
614 logging.debug(f'Bright field image offset: {bright_field["img_offset"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
615 logging.debug(f'Number of bright field images: {bright_field["num"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
616
2718cbde04fe Uploaded
rv43
parents:
diff changeset
617 # Find tomography images
2718cbde04fe Uploaded
rv43
parents:
diff changeset
618 for i,stack in enumerate(stack_info['stacks']):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
619 # Check if stack is already loaded or available
2718cbde04fe Uploaded
rv43
parents:
diff changeset
620 if self.tomo_stacks[i].size or available_stacks[i]:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
621 continue
2718cbde04fe Uploaded
rv43
parents:
diff changeset
622 index = stack['index']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
623 img_start = stack.get('img_start', -1)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
624 img_offset = stack.get('img_offset', -1)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
625 num_imgs = num_thetas
2718cbde04fe Uploaded
rv43
parents:
diff changeset
626 if not self.test_mode:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
627 img_start, img_offset, num_imgs = msnc.selectImageRange(img_start, img_offset,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
628 num_imgs, f'tomography stack {index}', num_thetas)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
629 if img_start < 0 or num_imgs != num_thetas:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
630 logging.error('Unable to find suitable tomography images')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
631 self.is_valid = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
632 stack['img_start'] = img_start
2718cbde04fe Uploaded
rv43
parents:
diff changeset
633 stack['img_offset'] = img_offset
2718cbde04fe Uploaded
rv43
parents:
diff changeset
634 stack['num'] = num_imgs
2718cbde04fe Uploaded
rv43
parents:
diff changeset
635 logging.debug(f'Tomography stack {index} image start index: {stack["img_start"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
636 logging.debug(f'Tomography stack {index} image offset: {stack["img_offset"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
637 logging.debug(f'Number of tomography images for stack {index}: {stack["num"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
638 available_stacks[i] = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
639
2718cbde04fe Uploaded
rv43
parents:
diff changeset
640 # Safe updated config to file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
641 if self.is_valid:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
642 self.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
643
2718cbde04fe Uploaded
rv43
parents:
diff changeset
644 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
645
2718cbde04fe Uploaded
rv43
parents:
diff changeset
646 def _genDark(self, tdf_files, dark_field_pngname):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
647 """Generate dark field.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
648 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
649 # Load the dark field images
2718cbde04fe Uploaded
rv43
parents:
diff changeset
650 logging.debug('Loading dark field...')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
651 dark_field = self.config['dark_field']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
652 tdf_stack = msnc.loadImageStack(tdf_files, self.config['data_filetype'],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
653 dark_field['img_offset'], dark_field['num'])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
654
2718cbde04fe Uploaded
rv43
parents:
diff changeset
655 # Take median
2718cbde04fe Uploaded
rv43
parents:
diff changeset
656 self.tdf = np.median(tdf_stack, axis=0)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
657 del tdf_stack
2718cbde04fe Uploaded
rv43
parents:
diff changeset
658
2718cbde04fe Uploaded
rv43
parents:
diff changeset
659 # RV make input of some kind (not always needed)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
660 tdf_cutoff = 21
2718cbde04fe Uploaded
rv43
parents:
diff changeset
661 self.tdf[self.tdf > tdf_cutoff] = np.nan
2718cbde04fe Uploaded
rv43
parents:
diff changeset
662 tdf_mean = np.nanmean(self.tdf)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
663 logging.debug(f'tdf_cutoff = {tdf_cutoff}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
664 logging.debug(f'tdf_mean = {tdf_mean}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
665 np.nan_to_num(self.tdf, copy=False, nan=tdf_mean, posinf=tdf_mean, neginf=0.)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
666 if not self.test_mode and not self.galaxy_flag:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
667 msnc.quickImshow(self.tdf, title='dark field', path=self.output_folder,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
668 save_fig=self.save_plots, save_only=self.save_plots_only)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
669 elif self.galaxy_flag:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
670 msnc.quickImshow(self.tdf, title='dark field', name=dark_field_pngname,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
671 save_fig=True, save_only=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
672
2718cbde04fe Uploaded
rv43
parents:
diff changeset
673 def _genBright(self, tbf_files, bright_field_pngname):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
674 """Generate bright field.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
675 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
676 # Load the bright field images
2718cbde04fe Uploaded
rv43
parents:
diff changeset
677 logging.debug('Loading bright field...')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
678 bright_field = self.config['bright_field']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
679 tbf_stack = msnc.loadImageStack(tbf_files, self.config['data_filetype'],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
680 bright_field['img_offset'], bright_field['num'])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
681
2718cbde04fe Uploaded
rv43
parents:
diff changeset
682 # Take median
2718cbde04fe Uploaded
rv43
parents:
diff changeset
683 """Median or mean: It may be best to try the median because of some image
2718cbde04fe Uploaded
rv43
parents:
diff changeset
684 artifacts that arise due to crinkles in the upstream kapton tape windows
2718cbde04fe Uploaded
rv43
parents:
diff changeset
685 causing some phase contrast images to appear on the detector.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
686 One thing that also may be useful in a future implementation is to do a
2718cbde04fe Uploaded
rv43
parents:
diff changeset
687 brightfield adjustment on EACH frame of the tomo based on a ROI in the
2718cbde04fe Uploaded
rv43
parents:
diff changeset
688 corner of the frame where there is no sample but there is the direct X-ray
2718cbde04fe Uploaded
rv43
parents:
diff changeset
689 beam because there is frame to frame fluctuations from the incoming beam.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
690 We don’t typically account for them but potentially could.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
691 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
692 self.tbf = np.median(tbf_stack, axis=0)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
693 del tbf_stack
2718cbde04fe Uploaded
rv43
parents:
diff changeset
694
2718cbde04fe Uploaded
rv43
parents:
diff changeset
695 # Subtract dark field
2718cbde04fe Uploaded
rv43
parents:
diff changeset
696 if self.tdf.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
697 self.tbf -= self.tdf
2718cbde04fe Uploaded
rv43
parents:
diff changeset
698 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
699 logging.warning('Dark field unavailable')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
700 if not self.test_mode and not self.galaxy_flag:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
701 msnc.quickImshow(self.tbf, title='bright field', path=self.output_folder,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
702 save_fig=self.save_plots, save_only=self.save_plots_only)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
703 elif self.galaxy_flag:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
704 msnc.quickImshow(self.tbf, title='bright field', name=bright_field_pngname,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
705 save_fig=True, save_only=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
706
2718cbde04fe Uploaded
rv43
parents:
diff changeset
707 def _setDetectorBounds(self, tomo_stack_files, tomo_field_pngname, detectorbounds_pngname):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
708 """Set vertical detector bounds for image stack.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
709 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
710 preprocess = self.config.get('preprocess')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
711 if preprocess is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
712 img_x_bounds = [0, self.tbf.shape[0]]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
713 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
714 img_x_bounds = preprocess.get('img_x_bounds', [0, self.tbf.shape[0]])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
715 if self.test_mode:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
716 # Update config and save to file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
717 if preprocess is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
718 self.cf.config['preprocess'] = {'img_x_bounds' : img_x_bounds}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
719 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
720 preprocess['img_x_bounds'] = img_x_bounds
2718cbde04fe Uploaded
rv43
parents:
diff changeset
721 self.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
722 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
723
2718cbde04fe Uploaded
rv43
parents:
diff changeset
724 # Check reference heights
2718cbde04fe Uploaded
rv43
parents:
diff changeset
725 pixel_size = self.config['detector']['pixel_size']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
726 if pixel_size is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
727 raise ValueError('Detector pixel size unavailable')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
728 if not self.tbf.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
729 raise ValueError('Bright field unavailable')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
730 num_x_min = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
731 num_tomo_stacks = self.config['stack_info']['num']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
732 stacks = self.config['stack_info']['stacks']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
733 if num_tomo_stacks > 1:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
734 delta_z = stacks[1]['ref_height']-stacks[0]['ref_height']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
735 for i in range(2, num_tomo_stacks):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
736 delta_z = min(delta_z, stacks[i]['ref_height']-stacks[i-1]['ref_height'])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
737 logging.debug(f'delta_z = {delta_z}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
738 num_x_min = int(delta_z/pixel_size)+1
2718cbde04fe Uploaded
rv43
parents:
diff changeset
739 logging.debug(f'num_x_min = {num_x_min}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
740 if num_x_min > self.tbf.shape[0]:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
741 logging.warning('Image bounds and pixel size prevent seamless stacking')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
742 num_x_min = self.tbf.shape[0]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
743
2718cbde04fe Uploaded
rv43
parents:
diff changeset
744 # Select image bounds
2718cbde04fe Uploaded
rv43
parents:
diff changeset
745 if self.galaxy_flag:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
746 if num_x_min is None or num_x_min >= self.tbf.shape[0]:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
747 img_x_bounds = [0, self.tbf.shape[0]]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
748 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
749 margin = int(num_x_min/10)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
750 offset = min(0, int((self.tbf.shape[0]-num_x_min)/2-margin))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
751 img_x_bounds = [offset, num_x_min+offset+2*margin]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
752 tomo_stack = msnc.loadImageStack(tomo_stack_files[0], self.config['data_filetype'],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
753 stacks[0]['img_offset'], 1)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
754 x_sum = np.sum(tomo_stack[0,:,:], 1)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
755 title = f'tomography image at theta={self.config["theta_range"]["start"]}'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
756 msnc.quickImshow(tomo_stack[0,:,:], title=title, name=tomo_field_pngname,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
757 save_fig=True, save_only=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
758 msnc.quickPlot((range(x_sum.size), x_sum),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
759 ([img_x_bounds[0], img_x_bounds[0]], [x_sum.min(), x_sum.max()], 'r-'),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
760 ([img_x_bounds[1], img_x_bounds[1]], [x_sum.min(), x_sum.max()], 'r-'),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
761 title='sum over theta and y', name=detectorbounds_pngname,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
762 save_fig=True, save_only=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
763
2718cbde04fe Uploaded
rv43
parents:
diff changeset
764 # Update config and save to file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
765 if preprocess is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
766 self.cf.config['preprocess'] = {'img_x_bounds' : img_x_bounds}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
767 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
768 preprocess['img_x_bounds'] = img_x_bounds
2718cbde04fe Uploaded
rv43
parents:
diff changeset
769 self.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
770 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
771
2718cbde04fe Uploaded
rv43
parents:
diff changeset
772 # For one tomography stack only: load the first image
2718cbde04fe Uploaded
rv43
parents:
diff changeset
773 title = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
774 msnc.quickImshow(self.tbf, title='bright field')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
775 if num_tomo_stacks == 1:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
776 tomo_stack = msnc.loadImageStack(tomo_stack_files[0], self.config['data_filetype'],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
777 stacks[0]['img_offset'], 1)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
778 title = f'tomography image at theta={self.config["theta_range"]["start"]}'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
779 msnc.quickImshow(tomo_stack[0,:,:], title=title)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
780 tomo_or_bright = pyip.inputNum('\nSelect image bounds from bright field (0) or '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
781 'from first tomography image (1): ', min=0, max=1)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
782 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
783 print('\nSelect image bounds from bright field')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
784 tomo_or_bright = 0
2718cbde04fe Uploaded
rv43
parents:
diff changeset
785 if tomo_or_bright:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
786 x_sum = np.sum(tomo_stack[0,:,:], 1)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
787 use_bounds = 'no'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
788 if img_x_bounds[0] is not None and img_x_bounds[1] is not None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
789 if img_x_bounds[0] < 0:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
790 msnc.illegal_value('img_x_bounds[0]', img_x_bounds[0], 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
791 img_x_bounds[0] = 0
2718cbde04fe Uploaded
rv43
parents:
diff changeset
792 if not img_x_bounds[0] < img_x_bounds[1] <= x_sum.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
793 msnc.illegal_value('img_x_bounds[1]', img_x_bounds[1], 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
794 img_x_bounds[1] = x_sum.size
2718cbde04fe Uploaded
rv43
parents:
diff changeset
795 tomo_tmp = tomo_stack[0,:,:]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
796 tomo_tmp[img_x_bounds[0],:] = tomo_stack[0,:,:].max()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
797 tomo_tmp[img_x_bounds[1],:] = tomo_stack[0,:,:].max()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
798 title = f'tomography image at theta={self.config["theta_range"]["start"]}'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
799 msnc.quickImshow(tomo_stack[0,:,:], title=title)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
800 msnc.quickPlot((range(x_sum.size), x_sum),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
801 ([img_x_bounds[0], img_x_bounds[0]], [x_sum.min(), x_sum.max()], 'r-'),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
802 ([img_x_bounds[1], img_x_bounds[1]], [x_sum.min(), x_sum.max()], 'r-'),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
803 title='sum over theta and y')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
804 print(f'lower bound = {img_x_bounds[0]} (inclusive)\n'+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
805 f'upper bound = {img_x_bounds[1]} (exclusive)]')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
806 use_bounds = pyip.inputYesNo('Accept these bounds ([y]/n)?: ', blank=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
807 if use_bounds == 'no':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
808 img_x_bounds = msnc.selectImageBounds(tomo_stack[0,:,:], 0,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
809 img_x_bounds[0], img_x_bounds[1], num_x_min,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
810 f'tomography image at theta={self.config["theta_range"]["start"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
811 if num_x_min is not None and img_x_bounds[1]-img_x_bounds[0]+1 < num_x_min:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
812 logging.warning('Image bounds and pixel size prevent seamless stacking')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
813 tomo_tmp = tomo_stack[0,:,:]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
814 tomo_tmp[img_x_bounds[0],:] = tomo_stack[0,:,:].max()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
815 tomo_tmp[img_x_bounds[1],:] = tomo_stack[0,:,:].max()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
816 title = f'tomography image at theta={self.config["theta_range"]["start"]}'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
817 msnc.quickImshow(tomo_stack[0,:,:], title=title, path=self.output_folder,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
818 save_fig=self.save_plots, save_only=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
819 msnc.quickPlot(range(img_x_bounds[0], img_x_bounds[1]),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
820 x_sum[img_x_bounds[0]:img_x_bounds[1]],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
821 title='sum over theta and y', path=self.output_folder,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
822 save_fig=self.save_plots, save_only=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
823 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
824 x_sum = np.sum(self.tbf, 1)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
825 use_bounds = 'no'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
826 if img_x_bounds[0] is not None and img_x_bounds[1] is not None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
827 if img_x_bounds[0] < 0:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
828 msnc.illegal_value('img_x_bounds[0]', img_x_bounds[0], 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
829 img_x_bounds[0] = 0
2718cbde04fe Uploaded
rv43
parents:
diff changeset
830 if not img_x_bounds[0] < img_x_bounds[1] <= x_sum.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
831 msnc.illegal_value('img_x_bounds[1]', img_x_bounds[1], 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
832 img_x_bounds[1] = x_sum.size
2718cbde04fe Uploaded
rv43
parents:
diff changeset
833 msnc.quickPlot((range(x_sum.size), x_sum),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
834 ([img_x_bounds[0], img_x_bounds[0]], [x_sum.min(), x_sum.max()], 'r-'),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
835 ([img_x_bounds[1], img_x_bounds[1]], [x_sum.min(), x_sum.max()], 'r-'),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
836 title='sum over theta and y')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
837 print(f'lower bound = {img_x_bounds[0]} (inclusive)\n'+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
838 f'upper bound = {img_x_bounds[1]} (exclusive)]')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
839 use_bounds = pyip.inputYesNo('Accept these bounds ([y]/n)?: ', blank=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
840 if use_bounds == 'no':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
841 fit = msnc.fitStep(y=x_sum, model='rectangle', form='atan')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
842 x_low = fit.get('center1', None)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
843 x_upp = fit.get('center2', None)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
844 if x_low is not None and x_low >= 0 and x_upp is not None and x_low < x_upp < x_sum.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
845 x_low = int(x_low-(x_upp-x_low)/10)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
846 if x_low < 0:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
847 x_low = 0
2718cbde04fe Uploaded
rv43
parents:
diff changeset
848 x_upp = int(x_upp+(x_upp-x_low)/10)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
849 if x_upp >= x_sum.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
850 x_upp = x_sum.size
2718cbde04fe Uploaded
rv43
parents:
diff changeset
851 msnc.quickPlot((range(x_sum.size), x_sum),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
852 ([x_low, x_low], [x_sum.min(), x_sum.max()], 'r-'),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
853 ([x_upp, x_upp], [x_sum.min(), x_sum.max()], 'r-'),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
854 title='sum over theta and y')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
855 print(f'lower bound = {x_low} (inclusive)\nupper bound = {x_upp} (exclusive)]')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
856 use_fit = pyip.inputYesNo('Accept these bounds ([y]/n)?: ', blank=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
857 if use_fit == 'no':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
858 img_x_bounds = msnc.selectArrayBounds(x_sum, img_x_bounds[0], img_x_bounds[1],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
859 num_x_min, 'sum over theta and y')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
860 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
861 img_x_bounds = [x_low, x_upp]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
862 if num_x_min is not None and img_x_bounds[1]-img_x_bounds[0]+1 < num_x_min:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
863 logging.warning('Image bounds and pixel size prevent seamless stacking')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
864 msnc.quickPlot(range(img_x_bounds[0], img_x_bounds[1]),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
865 x_sum[img_x_bounds[0]:img_x_bounds[1]],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
866 title='sum over theta and y', path=self.output_folder,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
867 save_fig=self.save_plots, save_only=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
868 logging.debug(f'img_x_bounds: {img_x_bounds}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
869
2718cbde04fe Uploaded
rv43
parents:
diff changeset
870 if self.save_plots_only:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
871 msnc.clearFig('bright field')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
872 msnc.clearFig('sum over theta and y')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
873 if title:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
874 msnc.clearFig(title)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
875
2718cbde04fe Uploaded
rv43
parents:
diff changeset
876 # Update config and save to file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
877 if preprocess is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
878 self.cf.config['preprocess'] = {'img_x_bounds' : img_x_bounds}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
879 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
880 preprocess['img_x_bounds'] = img_x_bounds
2718cbde04fe Uploaded
rv43
parents:
diff changeset
881 self.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
882
2718cbde04fe Uploaded
rv43
parents:
diff changeset
883 def _setZoomOrSkip(self):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
884 """Set zoom and/or theta skip to reduce memory the requirement for the analysis.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
885 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
886 preprocess = self.config.get('preprocess')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
887 zoom_perc = 100
2718cbde04fe Uploaded
rv43
parents:
diff changeset
888 if not self.galaxy_flag:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
889 if preprocess is None or 'zoom_perc' not in preprocess:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
890 if pyip.inputYesNo(
2718cbde04fe Uploaded
rv43
parents:
diff changeset
891 '\nDo you want to zoom in to reduce memory requirement (y/[n])? ',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
892 blank=True) == 'yes':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
893 zoom_perc = pyip.inputInt(' Enter zoom percentage [1, 100]: ',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
894 min=1, max=100)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
895 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
896 zoom_perc = preprocess['zoom_perc']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
897 if msnc.is_num(zoom_perc, 1., 100.):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
898 zoom_perc = int(zoom_perc)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
899 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
900 msnc.illegal_value('zoom_perc', zoom_perc, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
901 zoom_perc = 100
2718cbde04fe Uploaded
rv43
parents:
diff changeset
902 num_theta_skip = 0
2718cbde04fe Uploaded
rv43
parents:
diff changeset
903 if not self.galaxy_flag:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
904 if preprocess is None or 'num_theta_skip' not in preprocess:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
905 if pyip.inputYesNo(
2718cbde04fe Uploaded
rv43
parents:
diff changeset
906 'Do you want to skip thetas to reduce memory requirement (y/[n])? ',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
907 blank=True) == 'yes':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
908 num_theta_skip = pyip.inputInt(' Enter the number skip theta interval'+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
909 f' [0, {self.num_thetas-1}]: ', min=0, max=self.num_thetas-1)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
910 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
911 num_theta_skip = preprocess['num_theta_skip']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
912 if not msnc.is_int(num_theta_skip, 0):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
913 msnc.illegal_value('num_theta_skip', num_theta_skip, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
914 num_theta_skip = 0
2718cbde04fe Uploaded
rv43
parents:
diff changeset
915 logging.info(f'zoom_perc = {zoom_perc}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
916 logging.info(f'num_theta_skip = {num_theta_skip}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
917
2718cbde04fe Uploaded
rv43
parents:
diff changeset
918 # Update config and save to file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
919 if preprocess is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
920 self.cf.config['preprocess'] = {'zoom_perc' : zoom_perc,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
921 'num_theta_skip' : num_theta_skip}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
922 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
923 preprocess['zoom_perc'] = zoom_perc
2718cbde04fe Uploaded
rv43
parents:
diff changeset
924 preprocess['num_theta_skip'] = num_theta_skip
2718cbde04fe Uploaded
rv43
parents:
diff changeset
925 self.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
926
2718cbde04fe Uploaded
rv43
parents:
diff changeset
927 def _loadTomo(self, base_name, index, required=False):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
928 """Load a tomography stack.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
929 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
930 # stack order: row,theta,column
2718cbde04fe Uploaded
rv43
parents:
diff changeset
931 zoom_perc = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
932 preprocess = self.config.get('preprocess')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
933 if preprocess:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
934 zoom_perc = preprocess.get('zoom_perc')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
935 if zoom_perc is None or zoom_perc == 100:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
936 title = f'{base_name} fullres'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
937 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
938 title = f'{base_name} {zoom_perc}p'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
939 title += f'_{index}'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
940 tomo_file = re.sub(r"\s+", '_', f'{self.output_folder}/{title}.npy')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
941 load_flag = 'no'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
942 available = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
943 if os.path.isfile(tomo_file):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
944 available = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
945 if required:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
946 load_flag = 'yes'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
947 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
948 load_flag = pyip.inputYesNo(f'\nDo you want to load {tomo_file} (y/n)? ')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
949 stack = np.array([])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
950 if load_flag == 'yes':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
951 t0 = time()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
952 logging.info(f'Loading {tomo_file} ...')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
953 try:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
954 stack = np.load(tomo_file)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
955 except IOError or ValueError:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
956 stack = np.array([])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
957 logging.error(f'Error loading {tomo_file}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
958 logging.info(f'... done in {time()-t0:.2f} seconds!')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
959 if stack.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
960 msnc.quickImshow(stack[:,0,:], title=title, path=self.output_folder,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
961 save_fig=self.save_plots, save_only=self.save_plots_only)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
962 return stack, available
2718cbde04fe Uploaded
rv43
parents:
diff changeset
963
2718cbde04fe Uploaded
rv43
parents:
diff changeset
964 def _saveTomo(self, base_name, stack, index=None):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
965 """Save a tomography stack.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
966 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
967 zoom_perc = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
968 preprocess = self.config.get('preprocess')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
969 if preprocess:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
970 zoom_perc = preprocess.get('zoom_perc')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
971 if zoom_perc is None or zoom_perc == 100:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
972 title = f'{base_name} fullres'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
973 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
974 title = f'{base_name} {zoom_perc}p'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
975 if index:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
976 title += f'_{index}'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
977 tomo_file = re.sub(r"\s+", '_', f'{self.output_folder}/{title}.npy')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
978 t0 = time()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
979 logging.info(f'Saving {tomo_file} ...')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
980 np.save(tomo_file, stack)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
981 logging.info(f'... done in {time()-t0:.2f} seconds!')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
982
2718cbde04fe Uploaded
rv43
parents:
diff changeset
983 def _genTomo(self, tomo_stack_files, available_stacks):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
984 """Generate tomography fields.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
985 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
986 stacks = self.config['stack_info']['stacks']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
987 assert(len(self.tomo_stacks) == self.config['stack_info']['num'])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
988 assert(len(self.tomo_stacks) == len(stacks))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
989 if len(available_stacks) != len(stacks):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
990 logging.warning('Illegal dimension of available_stacks in _genTomo'+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
991 f'({len(available_stacks)}');
2718cbde04fe Uploaded
rv43
parents:
diff changeset
992 available_stacks = [False]*self.num_tomo_stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
993
2718cbde04fe Uploaded
rv43
parents:
diff changeset
994 preprocess = self.config.get('preprocess')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
995 if preprocess is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
996 img_x_bounds = [0, self.tbf.shape[0]]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
997 img_y_bounds = [0, self.tbf.shape[1]]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
998 zoom_perc = preprocess['zoom_perc']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
999 num_theta_skip = preprocess['num_theta_skip']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1000 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1001 img_x_bounds = preprocess.get('img_x_bounds', [0, self.tbf.shape[0]])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1002 img_y_bounds = preprocess.get('img_y_bounds', [0, self.tbf.shape[1]])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1003 zoom_perc = 100
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1004 num_theta_skip = 0
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1005
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1006 if self.tdf.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1007 tdf = self.tdf[img_x_bounds[0]:img_x_bounds[1],img_y_bounds[0]:img_y_bounds[1]]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1008 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1009 logging.warning('Dark field unavailable')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1010 if not self.tbf.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1011 raise ValueError('Bright field unavailable')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1012 tbf = self.tbf[img_x_bounds[0]:img_x_bounds[1],img_y_bounds[0]:img_y_bounds[1]]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1013
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1014 for i,stack in enumerate(stacks):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1015 # Check if stack is already loaded or available
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1016 if self.tomo_stacks[i].size or available_stacks[i]:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1017 continue
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1018
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1019 # Load a stack of tomography images
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1020 t0 = time()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1021 tomo_stack = msnc.loadImageStack(tomo_stack_files[i], self.config['data_filetype'],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1022 stack['img_offset'], self.config['theta_range']['num'], num_theta_skip,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1023 img_x_bounds, img_y_bounds)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1024 tomo_stack = tomo_stack.astype('float64')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1025 logging.debug(f'loading took {time()-t0:.2f} seconds!')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1026
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1027 # Subtract dark field
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1028 if self.tdf.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1029 t0 = time()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1030 with set_numexpr_threads(self.ncore):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1031 ne.evaluate('tomo_stack-tdf', out=tomo_stack)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1032 logging.debug(f'subtracting dark field took {time()-t0:.2f} seconds!')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1033
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1034 # Normalize
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1035 t0 = time()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1036 with set_numexpr_threads(self.ncore):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1037 ne.evaluate('tomo_stack/tbf', out=tomo_stack, truediv=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1038 logging.debug(f'normalizing took {time()-t0:.2f} seconds!')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1039
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1040 # Remove non-positive values and linearize data
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1041 t0 = time()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1042 cutoff = 1.e-6
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1043 with set_numexpr_threads(self.ncore):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1044 ne.evaluate('where(tomo_stack<cutoff, cutoff, tomo_stack)', out=tomo_stack)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1045 with set_numexpr_threads(self.ncore):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1046 ne.evaluate('-log(tomo_stack)', out=tomo_stack)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1047 logging.debug('removing non-positive values and linearizing data took '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1048 f'{time()-t0:.2f} seconds!')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1049
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1050 # Get rid of nans/infs that may be introduced by normalization
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1051 t0 = time()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1052 np.where(np.isfinite(tomo_stack), tomo_stack, 0.)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1053 logging.debug(f'remove nans/infs took {time()-t0:.2f} seconds!')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1054
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1055 # Downsize tomography stack to smaller size
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1056 tomo_stack = tomo_stack.astype('float32')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1057 if not self.galaxy_flag:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1058 index = stack['index']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1059 title = f'red stack fullres {index}'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1060 if not self.test_mode:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1061 msnc.quickImshow(tomo_stack[0,:,:], title=title, path=self.output_folder,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1062 save_fig=self.save_plots, save_only=self.save_plots_only)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1063 if zoom_perc != 100:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1064 t0 = time()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1065 logging.info(f'Zooming in ...')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1066 tomo_zoom_list = []
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1067 for j in range(tomo_stack.shape[0]):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1068 tomo_zoom = spi.zoom(tomo_stack[j,:,:], 0.01*zoom_perc)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1069 tomo_zoom_list.append(tomo_zoom)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1070 tomo_stack = np.stack([tomo_zoom for tomo_zoom in tomo_zoom_list])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1071 logging.info(f'... done in {time()-t0:.2f} seconds!')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1072 del tomo_zoom_list
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1073 if not self.galaxy_flag:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1074 title = f'red stack {zoom_perc}p {index}'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1075 if not self.test_mode:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1076 msnc.quickImshow(tomo_stack[0,:,:], title=title, path=self.output_folder,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1077 save_fig=self.save_plots, save_only=self.save_plots_only)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1078
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1079 # Convert tomography stack from theta,row,column to row,theta,column
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1080 tomo_stack = np.swapaxes(tomo_stack, 0, 1)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1081
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1082 # Save tomography stack to file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1083 if not self.galaxy_flag:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1084 if not self.test_mode:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1085 self._saveTomo('red stack', tomo_stack, index)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1086 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1087 np.savetxt(self.output_folder+f'red_stack_{index}.txt',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1088 tomo_stack[0,:,:], fmt='%.6e')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1089
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1090 # Combine stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1091 t0 = time()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1092 self.tomo_stacks[i] = tomo_stack
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1093 logging.debug(f'combining nstack took {time()-t0:.2f} seconds!')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1094
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1095 # Update config and save to file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1096 stack['preprocessed'] = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1097 self.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1098
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1099 if self.tdf.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1100 del tdf
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1101 del tbf
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1102
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1103 def _reconstructOnePlane(self, tomo_plane_T, center, thetas_deg, eff_pixel_size,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1104 cross_sectional_dim, plot_sinogram=True):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1105 """Invert the sinogram for a single tomography plane.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1106 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1107 # tomo_plane_T index order: column,theta
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1108 assert(0 <= center < tomo_plane_T.shape[0])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1109 center_offset = center-tomo_plane_T.shape[0]/2
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1110 two_offset = 2*int(np.round(center_offset))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1111 two_offset_abs = np.abs(two_offset)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1112 max_rad = int(0.5*(cross_sectional_dim/eff_pixel_size)*1.1) # 10% slack to avoid edge effects
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1113 dist_from_edge = max(1, int(np.floor((tomo_plane_T.shape[0]-two_offset_abs)/2.)-max_rad))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1114 if two_offset >= 0:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1115 logging.debug(f'sinogram range = [{two_offset+dist_from_edge}, {-dist_from_edge}]')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1116 sinogram = tomo_plane_T[two_offset+dist_from_edge:-dist_from_edge,:]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1117 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1118 logging.debug(f'sinogram range = [{dist_from_edge}, {two_offset-dist_from_edge}]')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1119 sinogram = tomo_plane_T[dist_from_edge:two_offset-dist_from_edge,:]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1120 if plot_sinogram:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1121 msnc.quickImshow(sinogram.T, f'sinogram center offset{center_offset:.2f}',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1122 path=self.output_folder, save_fig=self.save_plots,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1123 save_only=self.save_plots_only, aspect='auto')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1124
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1125 # Inverting sinogram
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1126 t0 = time()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1127 recon_sinogram = iradon(sinogram, theta=thetas_deg, circle=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1128 logging.debug(f'inverting sinogram took {time()-t0:.2f} seconds!')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1129 del sinogram
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1130
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1131 # Removing ring artifacts
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1132 # RV parameters for the denoise, gaussian, and ring removal will be different for different feature sizes
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1133 t0 = time()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1134 # recon_sinogram = filters.gaussian(recon_sinogram, 3.0)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1135 recon_sinogram = spi.gaussian_filter(recon_sinogram, 0.5)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1136 recon_clean = np.expand_dims(recon_sinogram, axis=0)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1137 del recon_sinogram
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1138 recon_clean = tomopy.misc.corr.remove_ring(recon_clean, rwidth=17)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1139 logging.debug(f'filtering and removing ring artifact took {time()-t0:.2f} seconds!')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1140 return recon_clean
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1141
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1142 def _plotEdgesOnePlane(self, recon_plane, base_name, weight=0.001):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1143 # RV parameters for the denoise, gaussian, and ring removal will be different for different feature sizes
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1144 edges = denoise_tv_chambolle(recon_plane, weight = weight)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1145 vmax = np.max(edges[0,:,:])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1146 vmin = -vmax
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1147 msnc.quickImshow(edges[0,:,:], f'{base_name} coolwarm', path=self.output_folder,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1148 save_fig=self.save_plots, save_only=self.save_plots_only, cmap='coolwarm')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1149 msnc.quickImshow(edges[0,:,:], f'{base_name} gray', path=self.output_folder,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1150 save_fig=self.save_plots, save_only=self.save_plots_only, cmap='gray',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1151 vmin=vmin, vmax=vmax)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1152 del edges
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1153
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1154 def _findCenterOnePlane(self, sinogram, row, thetas_deg, eff_pixel_size, cross_sectional_dim,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1155 tol=0.1):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1156 """Find center for a single tomography plane.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1157 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1158 # sinogram index order: theta,column
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1159 # need index order column,theta for iradon, so take transpose
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1160 sinogram_T = sinogram.T
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1161 center = sinogram.shape[1]/2
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1162
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1163 # try automatic center finding routines for initial value
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1164 tomo_center = tomopy.find_center_vo(sinogram)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1165 center_offset_vo = tomo_center-center
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1166 print(f'Center at row {row} using Nghia Vo’s method = {center_offset_vo:.2f}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1167 recon_plane = self._reconstructOnePlane(sinogram_T, tomo_center, thetas_deg,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1168 eff_pixel_size, cross_sectional_dim, False)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1169 base_name=f'edges row{row} center_offset_vo{center_offset_vo:.2f}'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1170 self._plotEdgesOnePlane(recon_plane, base_name)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1171 if pyip.inputYesNo('Try finding center using phase correlation (y/[n])? ',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1172 blank=True) == 'yes':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1173 tomo_center = tomopy.find_center_pc(sinogram, sinogram, tol=0.1,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1174 rotc_guess=tomo_center)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1175 error = 1.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1176 while error > tol:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1177 prev = tomo_center
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1178 tomo_center = tomopy.find_center_pc(sinogram, sinogram, tol=tol,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1179 rotc_guess=tomo_center)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1180 error = np.abs(tomo_center-prev)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1181 center_offset = tomo_center-center
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1182 print(f'Center at row {row} using phase correlation = {center_offset:.2f}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1183 recon_plane = self._reconstructOnePlane(sinogram_T, tomo_center, thetas_deg,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1184 eff_pixel_size, cross_sectional_dim, False)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1185 base_name=f'edges row{row} center_offset{center_offset:.2f}'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1186 self._plotEdgesOnePlane(recon_plane, base_name)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1187 if pyip.inputYesNo('Accept a center location ([y]) or continue search (n)? ',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1188 blank=True) != 'no':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1189 del sinogram_T
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1190 del recon_plane
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1191 center_offset = pyip.inputNum(
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1192 f' Enter chosen center offset [{-int(center)}, {int(center)}] '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1193 f'([{center_offset_vo}])): ', blank=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1194 if center_offset == '':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1195 center_offset = center_offset_vo
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1196 return float(center_offset)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1197
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1198 while True:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1199 center_offset_low = pyip.inputInt('\nEnter lower bound for center offset '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1200 f'[{-int(center)}, {int(center)}]: ', min=-int(center), max=int(center))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1201 center_offset_upp = pyip.inputInt('Enter upper bound for center offset '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1202 f'[{center_offset_low}, {int(center)}]: ',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1203 min=center_offset_low, max=int(center))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1204 if center_offset_upp == center_offset_low:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1205 center_offset_step = 1
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1206 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1207 center_offset_step = pyip.inputInt('Enter step size for center offset search '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1208 f'[1, {center_offset_upp-center_offset_low}]: ',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1209 min=1, max=center_offset_upp-center_offset_low)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1210 for center_offset in range(center_offset_low, center_offset_upp+center_offset_step,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1211 center_offset_step):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1212 logging.info(f'center_offset = {center_offset}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1213 recon_plane = self._reconstructOnePlane(sinogram_T, center_offset+center,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1214 thetas_deg, eff_pixel_size, cross_sectional_dim, False)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1215 base_name=f'edges row{row} center_offset{center_offset}'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1216 self._plotEdgesOnePlane(recon_plane, base_name)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1217 if pyip.inputInt('\nContinue (0) or end the search (1): ', min=0, max=1):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1218 break
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1219
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1220 del sinogram_T
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1221 del recon_plane
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1222 center_offset = pyip.inputNum(f' Enter chosen center offset '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1223 f'[{-int(center)}, {int(center)}]: ', min=-int(center), max=int(center))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1224 return float(center_offset)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1225
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1226 def _reconstructOneTomoStack(self, tomo_stack, thetas, row_bounds=None,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1227 center_offsets=[], sigma=0.1, rwidth=30, ncore=1, algorithm='gridrec',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1228 run_secondary_sirt=False, secondary_iter=100):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1229 """reconstruct a single tomography stack.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1230 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1231 # stack order: row,theta,column
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1232 # thetas must be in radians
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1233 # centers_offset: tomography axis shift in pixels relative to column center
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1234 # RV should we remove stripes?
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1235 # https://tomopy.readthedocs.io/en/latest/api/tomopy.prep.stripe.html
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1236 # RV should we remove rings?
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1237 # https://tomopy.readthedocs.io/en/latest/api/tomopy.misc.corr.html
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1238 # RV: Add an option to do (extra) secondary iterations later or to do some sort of convergence test?
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1239 if row_bounds is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1240 row_bounds = [0, tomo_stack.shape[0]]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1241 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1242 if not (0 <= row_bounds[0] <= row_bounds[1] <= tomo_stack.shape[0]):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1243 raise ValueError('Illegal row bounds in reconstructOneTomoStack')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1244 if thetas.size != tomo_stack.shape[1]:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1245 raise ValueError('theta dimension mismatch in reconstructOneTomoStack')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1246 if not len(center_offsets):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1247 centers = np.zeros((row_bounds[1]-row_bounds[0]))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1248 elif len(center_offsets) == 2:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1249 centers = np.linspace(center_offsets[0], center_offsets[1],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1250 row_bounds[1]-row_bounds[0])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1251 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1252 if center_offsets.size != row_bounds[1]-row_bounds[0]:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1253 raise ValueError('center_offsets dimension mismatch in reconstructOneTomoStack')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1254 centers = center_offsets
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1255 centers += tomo_stack.shape[2]/2
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1256 if True:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1257 tomo_stack = tomopy.prep.stripe.remove_stripe_fw(tomo_stack[row_bounds[0]:row_bounds[1]],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1258 sigma=sigma, ncore=ncore)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1259 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1260 tomo_stack = tomo_stack[row_bounds[0]:row_bounds[1]]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1261 tomo_recon_stack = tomopy.recon(tomo_stack, thetas, centers, sinogram_order=True,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1262 algorithm=algorithm, ncore=ncore)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1263 if run_secondary_sirt and secondary_iter > 0:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1264 #options = {'method':'SIRT_CUDA', 'proj_type':'cuda', 'num_iter':secondary_iter}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1265 #RV: doesn't work for me: "Error: CUDA error 803: system has unsupported display driver /
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1266 # cuda driver combination."
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1267 #options = {'method':'SIRT', 'proj_type':'linear', 'MinConstraint': 0, 'num_iter':secondary_iter}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1268 #SIRT did not finish while running overnight
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1269 #options = {'method':'SART', 'proj_type':'linear', 'num_iter':secondary_iter}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1270 options = {'method':'SART', 'proj_type':'linear', 'MinConstraint': 0, 'num_iter':secondary_iter}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1271 tomo_recon_stack = tomopy.recon(tomo_stack, thetas, centers, init_recon=tomo_recon_stack,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1272 options=options, sinogram_order=True, algorithm=tomopy.astra, ncore=ncore)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1273 if True:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1274 tomopy.misc.corr.remove_ring(tomo_recon_stack, rwidth=rwidth, out=tomo_recon_stack)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1275 return tomo_recon_stack
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1276
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1277 def genTomoStacks(self, tdf_files=None, tbf_files=None, tomo_stack_files=None,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1278 dark_field_pngname=None, bright_field_pngname=None, tomo_field_pngname=None,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1279 detectorbounds_pngname=None, output_name=None):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1280 """Preprocess tomography images.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1281 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1282 # Try loading any already preprocessed stacks (skip in Galaxy)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1283 # preprocessed stack order for each one in stack: row,theta,column
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1284 stack_info = self.config['stack_info']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1285 stacks = stack_info['stacks']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1286 num_tomo_stacks = stack_info['num']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1287 assert(num_tomo_stacks == len(self.tomo_stacks))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1288 available_stacks = [False]*num_tomo_stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1289 if self.galaxy_flag:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1290 assert(tdf_files is None or isinstance(tdf_files, list))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1291 assert(isinstance(tbf_files, list))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1292 assert(isinstance(tomo_stack_files, list))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1293 assert(num_tomo_stacks == len(tomo_stack_files))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1294 assert(isinstance(dark_field_pngname, str))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1295 assert(isinstance(bright_field_pngname, str))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1296 assert(isinstance(tomo_field_pngname, str))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1297 assert(isinstance(detectorbounds_pngname, str))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1298 assert(isinstance(output_name, str))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1299 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1300 if tdf_files:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1301 logging.warning('Ignoring tdf_files in genTomoStacks (only for Galaxy)')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1302 if tbf_files:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1303 logging.warning('Ignoring tbf_files in genTomoStacks (only for Galaxy)')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1304 if tomo_stack_files:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1305 logging.warning('Ignoring tomo_stack_files in genTomoStacks (only for Galaxy)')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1306 tdf_files, tbf_files, tomo_stack_files = self.findImageFiles()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1307 if not self.is_valid:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1308 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1309 for i,stack in enumerate(stacks):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1310 if not self.tomo_stacks[i].size and stack.get('preprocessed', False):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1311 self.tomo_stacks[i], available_stacks[i] = \
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1312 self._loadTomo('red stack', stack['index'])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1313 if dark_field_pngname:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1314 logging.warning('Ignoring dark_field_pngname in genTomoStacks (only for Galaxy)')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1315 if bright_field_pngname:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1316 logging.warning('Ignoring bright_field_pngname in genTomoStacks (only for Galaxy)')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1317 if tomo_field_pngname:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1318 logging.warning('Ignoring tomo_field_pngname in genTomoStacks (only for Galaxy)')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1319 if detectorbounds_pngname:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1320 logging.warning('Ignoring detectorbounds_pngname in genTomoStacks '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1321 '(only used in Galaxy)')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1322 if output_name:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1323 logging.warning('Ignoring output_name in genTomoStacks '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1324 '(only used in Galaxy)')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1325
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1326 # Preprocess any unloaded stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1327 if False in available_stacks:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1328 logging.debug('Preprocessing tomography images')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1329
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1330 # Check required image files (skip in Galaxy)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1331 if not self.galaxy_flag:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1332 self.selectImageRanges(available_stacks)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1333 if not self.is_valid:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1334 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1335
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1336 # Generate dark field
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1337 if tdf_files:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1338 self._genDark(tdf_files, dark_field_pngname)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1339
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1340 # Generate bright field
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1341 self._genBright(tbf_files, bright_field_pngname)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1342
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1343 # Set vertical detector bounds for image stack
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1344 self._setDetectorBounds(tomo_stack_files, tomo_field_pngname, detectorbounds_pngname)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1345
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1346 # Set zoom and/or theta skip to reduce memory the requirement
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1347 self._setZoomOrSkip()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1348
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1349 # Generate tomography fields
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1350 self._genTomo(tomo_stack_files, available_stacks)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1351
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1352 # Save tomography stack to file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1353 if self.galaxy_flag:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1354 t0 = time()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1355 logging.info(f'Saving preprocessed tomography stack to file ...')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1356 save_stacks = {f'set_{stack["index"]}':tomo_stack
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1357 for stack,tomo_stack in zip(stacks,self.tomo_stacks)}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1358 np.savez(output_name, **save_stacks)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1359 logging.info(f'... done in {time()-t0:.2f} seconds!')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1360
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1361 del available_stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1362
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1363 # Adjust sample reference height, update config and save to file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1364 preprocess = self.config.get('preprocess')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1365 if preprocess is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1366 img_x_bounds = [0, self.tbf.shape[0]]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1367 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1368 img_x_bounds = preprocess.get('img_x_bounds', [0, self.tbf.shape[0]])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1369 pixel_size = self.config['detector']['pixel_size']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1370 if pixel_size is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1371 raise ValueError('Detector pixel size unavailable')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1372 pixel_size *= img_x_bounds[0]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1373 for stack in stacks:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1374 stack['ref_height'] = stack['ref_height']+pixel_size
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1375 self.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1376
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1377 def findCenters(self):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1378 """Find rotation axis centers for the tomography stacks.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1379 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1380 logging.debug('Find centers for tomography stacks')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1381 stacks = self.config['stack_info']['stacks']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1382 available_stacks = [stack['index'] for stack in stacks if stack.get('preprocessed', False)]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1383 logging.debug('Avaliable stacks: {available_stacks}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1384
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1385 # Check for valid available center stack index
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1386 find_center = self.config.get('find_center')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1387 if find_center and 'center_stack_index' in find_center:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1388 center_stack_index = find_center['center_stack_index']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1389 if (not isinstance(center_stack_index, int) or
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1390 center_stack_index not in available_stacks):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1391 msnc.illegal_value('center_stack_index', center_stack_index, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1392 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1393 if self.test_mode:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1394 find_center['completed'] = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1395 self.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1396 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1397 print('\nFound calibration center offset info for stack '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1398 f'{center_stack_index}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1399 if pyip.inputYesNo('Do you want to use this again (y/n)? ') == 'yes':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1400 find_center['completed'] = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1401 self.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1402 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1403
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1404 # Load the required preprocessed stack
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1405 # preprocessed stack order: row,theta,column
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1406 num_tomo_stacks = self.config['stack_info']['num']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1407 assert(len(stacks) == num_tomo_stacks)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1408 assert(len(self.tomo_stacks) == num_tomo_stacks)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1409 if num_tomo_stacks == 1:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1410 center_stack_index = stacks[0]['index']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1411 if not self.tomo_stacks[0].size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1412 self.tomo_stacks[0], available = self._loadTomo('red stack', center_stack_index,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1413 required=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1414 center_stack = self.tomo_stacks[0]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1415 if not center_stack.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1416 logging.error('Unable to load the required preprocessed tomography stack')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1417 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1418 assert(stacks[0].get('preprocessed', False) == True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1419 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1420 while True:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1421 center_stack_index = pyip.inputInt('\nEnter tomography stack index to get '
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1422 f'rotation axis centers {available_stacks}: ')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1423 while center_stack_index not in available_stacks:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1424 center_stack_index = pyip.inputInt('\nEnter tomography stack index to get '
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1425 f'rotation axis centers {available_stacks}: ')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1426 tomo_stack_index = available_stacks.index(center_stack_index)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1427 if not self.tomo_stacks[tomo_stack_index].size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1428 self.tomo_stacks[tomo_stack_index], available = self._loadTomo(
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1429 'red stack', center_stack_index, required=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1430 center_stack = self.tomo_stacks[tomo_stack_index]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1431 if not center_stack.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1432 logging.error(f'Unable to load the {center_stack_index}th '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1433 'preprocessed tomography stack, pick another one')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1434 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1435 break
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1436 assert(stacks[tomo_stack_index].get('preprocessed', False) == True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1437 if find_center is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1438 self.config['find_center'] = {'center_stack_index' : center_stack_index}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1439 find_center = self.config['find_center']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1440
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1441 # Set thetas (in degrees)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1442 theta_range = self.config['theta_range']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1443 theta_start = theta_range['start']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1444 theta_end = theta_range['end']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1445 num_theta = theta_range['num']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1446 num_theta_skip = self.config['preprocess'].get('num_theta_skip', 0)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1447 thetas_deg = np.linspace(theta_start, theta_end, int(num_theta/(num_theta_skip+1)),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1448 endpoint=False)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1449
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1450 # Get non-overlapping sample row boundaries
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1451 zoom_perc = self.config['preprocess'].get('zoom_perc', 100)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1452 pixel_size = self.config['detector']['pixel_size']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1453 if pixel_size is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1454 raise ValueError('Detector pixel size unavailable')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1455 eff_pixel_size = 100.*pixel_size/zoom_perc
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1456 logging.debug(f'eff_pixel_size = {eff_pixel_size}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1457 tomo_ref_heights = [stack['ref_height'] for stack in stacks]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1458 if num_tomo_stacks == 1:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1459 n1 = 0
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1460 height = center_stack.shape[0]*eff_pixel_size
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1461 if pyip.inputYesNo('\nDo you want to reconstruct the full samply height '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1462 f'({height:.3f} mm) (y/n)? ') == 'no':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1463 height = pyip.inputNum('\nEnter the desired reconstructed sample height '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1464 f'in mm [0, {height:.3f}]: ', min=0, max=height)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1465 n1 = int(0.5*(center_stack.shape[0]-height/eff_pixel_size))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1466 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1467 n1 = int((1.+(tomo_ref_heights[0]+center_stack.shape[0]*eff_pixel_size-
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1468 tomo_ref_heights[1])/eff_pixel_size)/2)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1469 n2 = center_stack.shape[0]-n1
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1470 logging.info(f'n1 = {n1}, n2 = {n2} (n2-n1) = {(n2-n1)*eff_pixel_size:.3f} mm')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1471 if not center_stack.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1472 RuntimeError('Center stack not loaded')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1473 msnc.quickImshow(center_stack[:,0,:], title=f'center stack theta={theta_start}',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1474 path=self.output_folder, save_fig=self.save_plots, save_only=self.save_plots_only)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1475
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1476 # Get cross sectional diameter in mm
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1477 cross_sectional_dim = center_stack.shape[2]*eff_pixel_size
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1478 logging.debug(f'cross_sectional_dim = {cross_sectional_dim}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1479
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1480 # Determine center offset at sample row boundaries
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1481 logging.info('Determine center offset at sample row boundaries')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1482
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1483 # Lower row center
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1484 use_row = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1485 use_center = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1486 row = find_center.get('lower_row')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1487 if msnc.is_int(row, n1, n2-2):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1488 msnc.quickImshow(center_stack[:,0,:], title=f'theta={theta_start}', aspect='auto')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1489 use_row = pyip.inputYesNo('\nCurrent row index for lower center = '
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1490 f'{row}, use this value (y/n)? ')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1491 if self.save_plots_only:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1492 msnc.clearFig(f'theta={theta_start}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1493 if use_row:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1494 center_offset = find_center.get('lower_center_offset')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1495 if msnc.is_num(center_offset):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1496 use_center = pyip.inputYesNo('Current lower center offset = '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1497 f'{center_offset}, use this value (y/n)? ')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1498 if not use_center:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1499 if not use_row:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1500 msnc.quickImshow(center_stack[:,0,:], title=f'theta={theta_start}', aspect='auto')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1501 row = pyip.inputInt('\nEnter row index to find lower center '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1502 f'[[{n1}], {n2-2}]: ', min=n1, max=n2-2, blank=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1503 if row == '':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1504 row = n1
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1505 if self.save_plots_only:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1506 msnc.clearFig(f'theta={theta_start}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1507 # center_stack order: row,theta,column
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1508 center_offset = self._findCenterOnePlane(center_stack[row,:,:], row, thetas_deg,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1509 eff_pixel_size, cross_sectional_dim)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1510 logging.info(f'Lower center offset = {center_offset}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1511
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1512 # Update config and save to file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1513 find_center['row_bounds'] = [n1, n2]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1514 find_center['lower_row'] = row
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1515 find_center['lower_center_offset'] = center_offset
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1516 self.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1517 lower_row = row
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1518
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1519 # Upper row center
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1520 use_row = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1521 use_center = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1522 row = find_center.get('upper_row')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1523 if msnc.is_int(row, lower_row+1, n2-1):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1524 msnc.quickImshow(center_stack[:,0,:], title=f'theta={theta_start}', aspect='auto')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1525 use_row = pyip.inputYesNo('\nCurrent row index for upper center = '
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1526 f'{row}, use this value (y/n)? ')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1527 if self.save_plots_only:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1528 msnc.clearFig(f'theta={theta_start}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1529 if use_row:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1530 center_offset = find_center.get('upper_center_offset')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1531 if msnc.is_num(center_offset):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1532 use_center = pyip.inputYesNo('Current upper center offset = '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1533 f'{center_offset}, use this value (y/n)? ')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1534 if not use_center:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1535 if not use_row:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1536 msnc.quickImshow(center_stack[:,0,:], title=f'theta={theta_start}', aspect='auto')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1537 row = pyip.inputInt('\nEnter row index to find upper center '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1538 f'[{lower_row+1}, [{n2-1}]]: ', min=lower_row+1, max=n2-1, blank=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1539 if row == '':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1540 row = n2-1
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1541 if self.save_plots_only:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1542 msnc.clearFig(f'theta={theta_start}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1543 # center_stack order: row,theta,column
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1544 center_offset = self._findCenterOnePlane(center_stack[row,:,:], row, thetas_deg,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1545 eff_pixel_size, cross_sectional_dim)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1546 logging.info(f'upper_center_offset = {center_offset}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1547 del center_stack
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1548
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1549 # Update config and save to file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1550 find_center['upper_row'] = row
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1551 find_center['upper_center_offset'] = center_offset
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1552 find_center['completed'] = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1553 self.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1554
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1555 def checkCenters(self):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1556 """Check centers for the tomography stacks.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1557 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1558 #RV TODO load all stacks and check at all stack boundaries
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1559 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1560 logging.debug('Check centers for tomography stacks')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1561 center_stack_index = self.config.get('center_stack_index')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1562 if center_stack_index is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1563 raise ValueError('Unable to read center_stack_index from config')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1564 center_stack_index = self.tomo_stacks[self.tomo_data_indices.index(center_stack_index)]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1565 lower_row = self.config.get('lower_row')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1566 if lower_row is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1567 raise ValueError('Unable to read lower_row from config')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1568 lower_center_offset = self.config.get('lower_center_offset')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1569 if lower_center_offset is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1570 raise ValueError('Unable to read lower_center_offset from config')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1571 upper_row = self.config.get('upper_row')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1572 if upper_row is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1573 raise ValueError('Unable to read upper_row from config')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1574 upper_center_offset = self.config.get('upper_center_offset')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1575 if upper_center_offset is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1576 raise ValueError('Unable to read upper_center_offset from config')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1577 center_slope = (upper_center_offset-lower_center_offset)/(upper_row-lower_row)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1578 shift = upper_center_offset-lower_center_offset
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1579 if lower_row == 0:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1580 logging.warning(f'lower_row == 0: one row offset between both planes')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1581 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1582 lower_row -= 1
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1583 lower_center_offset -= center_slope
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1584
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1585 # stack order: stack,row,theta,column
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1586 if center_stack_index:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1587 stack1 = self.tomo_stacks[center_stack_index-1]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1588 stack2 = self.tomo_stacks[center_stack_index]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1589 if not stack1.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1590 logging.error(f'Unable to load required tomography stack {stack1}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1591 elif not stack2.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1592 logging.error(f'Unable to load required tomography stack {stack1}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1593 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1594 assert(0 <= lower_row < stack2.shape[0])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1595 assert(0 <= upper_row < stack1.shape[0])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1596 plane1 = stack1[upper_row,:]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1597 plane2 = stack2[lower_row,:]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1598 for i in range(-2, 3):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1599 shift_i = shift+2*i
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1600 plane1_shifted = spi.shift(plane2, [0, shift_i])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1601 msnc.quickPlot((plane1[0,:],), (plane1_shifted[0,:],),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1602 title=f'stacks {stack1} {stack2} shifted {2*i} theta={self.start_theta}',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1603 path=self.output_folder, save_fig=self.save_plots,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1604 save_only=self.save_plots_only)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1605 if center_stack_index < self.num_tomo_stacks-1:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1606 stack1 = self.tomo_stacks[center_stack_index]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1607 stack2 = self.tomo_stacks[center_stack_index+1]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1608 if not stack1.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1609 logging.error(f'Unable to load required tomography stack {stack1}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1610 elif not stack2.size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1611 logging.error(f'Unable to load required tomography stack {stack1}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1612 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1613 assert(0 <= lower_row < stack2.shape[0])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1614 assert(0 <= upper_row < stack1.shape[0])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1615 plane1 = stack1[upper_row,:]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1616 plane2 = stack2[lower_row,:]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1617 for i in range(-2, 3):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1618 shift_i = -shift+2*i
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1619 plane1_shifted = spi.shift(plane2, [0, shift_i])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1620 msnc.quickPlot((plane1[0,:],), (plane1_shifted[0,:],),
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1621 title=f'stacks {stack1} {stack2} shifted {2*i} theta={start_theta}',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1622 path=self.output_folder, save_fig=self.save_plots,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1623 save_only=self.save_plots_only)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1624 del plane1, plane2, plane1_shifted
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1625
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1626 # Update config file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1627 self.config = msnc.update('config.txt', 'check_centers', True, 'find_centers')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1628
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1629 def reconstructTomoStacks(self):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1630 """Reconstruct tomography stacks.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1631 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1632 logging.debug('Reconstruct tomography stacks')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1633
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1634 # Get rotation axis rows and centers
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1635 find_center = self.config['find_center']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1636 lower_row = find_center.get('lower_row')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1637 if lower_row is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1638 logging.error('Unable to read lower_row from config')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1639 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1640 lower_center_offset = find_center.get('lower_center_offset')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1641 if lower_center_offset is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1642 logging.error('Unable to read lower_center_offset from config')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1643 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1644 upper_row = find_center.get('upper_row')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1645 if upper_row is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1646 logging.error('Unable to read upper_row from config')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1647 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1648 upper_center_offset = find_center.get('upper_center_offset')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1649 if upper_center_offset is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1650 logging.error('Unable to read upper_center_offset from config')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1651 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1652 logging.debug(f'lower_row = {lower_row} upper_row = {upper_row}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1653 assert(lower_row < upper_row)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1654 center_slope = (upper_center_offset-lower_center_offset)/(upper_row-lower_row)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1655
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1656 # Set thetas (in radians)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1657 theta_range = self.config['theta_range']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1658 theta_start = theta_range['start']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1659 theta_end = theta_range['end']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1660 num_theta = theta_range['num']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1661 num_theta_skip = self.config['preprocess'].get('num_theta_skip', 0)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1662 thetas = np.radians(np.linspace(theta_start, theta_end,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1663 int(num_theta/(num_theta_skip+1)), endpoint=False))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1664
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1665 # Reconstruct tomo stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1666 zoom_perc = self.config['preprocess'].get('zoom_perc', 100)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1667 if zoom_perc == 100:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1668 basetitle = 'recon stack full'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1669 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1670 basetitle = f'recon stack {zoom_perc}p'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1671 load_error = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1672 stacks = self.config['stack_info']['stacks']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1673 for i,stack in enumerate(stacks):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1674 # Check if stack can be loaded
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1675 # reconstructed stack order for each one in stack : row/z,x,y
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1676 # preprocessed stack order for each one in stack: row,theta,column
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1677 index = stack['index']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1678 available = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1679 if stack.get('reconstructed', False):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1680 self.tomo_recon_stacks[i], available = self._loadTomo('recon stack', index)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1681 if self.tomo_recon_stacks[i].size or available:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1682 if self.tomo_stacks[i].size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1683 self.tomo_stacks[i] = np.array([])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1684 assert(stack.get('reconstructed', False) == True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1685 continue
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1686 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1687 stack['reconstructed'] = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1688 if not self.tomo_stacks[i].size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1689 self.tomo_stacks[i], available = self._loadTomo('red stack', index,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1690 required=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1691 if not self.tomo_stacks[i].size:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1692 logging.error(f'Unable to load tomography stack {index} for reconstruction')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1693 load_error = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1694 continue
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1695 assert(0 <= lower_row < upper_row < self.tomo_stacks[i].shape[0])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1696 center_offsets = [lower_center_offset-lower_row*center_slope,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1697 upper_center_offset+(self.tomo_stacks[i].shape[0]-1-upper_row)*center_slope]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1698 t0 = time()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1699 self.tomo_recon_stacks[i]= self._reconstructOneTomoStack(self.tomo_stacks[i],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1700 thetas, center_offsets=center_offsets, sigma=0.1, ncore=self.ncore,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1701 algorithm='gridrec', run_secondary_sirt=True, secondary_iter=25)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1702 logging.info(f'Reconstruction of stack {index} took {time()-t0:.2f} seconds!')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1703 if not self.test_mode:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1704 row_slice = int(self.tomo_stacks[i].shape[0]/2)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1705 title = f'{basetitle} {index} slice{row_slice}'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1706 msnc.quickImshow(self.tomo_recon_stacks[i][row_slice,:,:], title=title,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1707 path=self.output_folder, save_fig=self.save_plots,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1708 save_only=self.save_plots_only)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1709 msnc.quickPlot(self.tomo_recon_stacks[i]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1710 [row_slice,int(self.tomo_recon_stacks[i].shape[2]/2),:],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1711 title=f'{title} cut{int(self.tomo_recon_stacks[i].shape[2]/2)}',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1712 path=self.output_folder, save_fig=self.save_plots,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1713 save_only=self.save_plots_only)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1714 self._saveTomo('recon stack', self.tomo_recon_stacks[i], index)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1715 # else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1716 # np.savetxt(self.output_folder+f'recon_stack_{index}.txt',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1717 # self.tomo_recon_stacks[i][row_slice,:,:], fmt='%.6e')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1718 self.tomo_stacks[i] = np.array([])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1719
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1720 # Update config and save to file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1721 stack['reconstructed'] = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1722 self.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1723
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1724 def combineTomoStacks(self):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1725 """Combine the reconstructed tomography stacks.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1726 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1727 # stack order: stack,row(z),x,y
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1728 logging.debug('Combine reconstructed tomography stacks')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1729 # Load any unloaded reconstructed stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1730 stacks = self.config['stack_info']['stacks']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1731 for i,stack in enumerate(stacks):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1732 if not self.tomo_recon_stacks[i].size and stack.get('reconstructed', False):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1733 self.tomo_recon_stacks[i], available = self._loadTomo('recon stack',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1734 stack['index'], required=True)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1735 if not self.tomo_recon_stacks[i].size or not available:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1736 logging.error(f'Unable to load reconstructed stack {stack["index"]}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1737 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1738 if i:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1739 if (self.tomo_recon_stacks[i].shape[1] != self.tomo_recon_stacks[0].shape[1] or
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1740 self.tomo_recon_stacks[i].shape[1] != self.tomo_recon_stacks[0].shape[1]):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1741 logging.error('Incompatible reconstructed tomography stack dimensions')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1742 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1743
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1744 # Get center stack boundaries
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1745 row_bounds = self.config['find_center']['row_bounds']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1746 if not msnc.is_index_range(row_bounds, 0, self.tomo_recon_stacks[0].shape[0]):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1747 msnc.illegal_value('row_bounds', row_bounds, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1748 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1749
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1750 # Selecting xy bounds
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1751 tomosum = 0
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1752 #RV FIX :=
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1753 #[tomosum := tomosum+np.sum(tomo_recon_stack, axis=(0,2)) for tomo_recon_stack in
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1754 # self.tomo_recon_stacks]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1755 combine_stacks =self.config.get('combine_stacks')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1756 if combine_stacks and 'x_bounds' in combine_stacks:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1757 x_bounds = combine_stacks['x_bounds']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1758 if not msnc.is_index_range(x_bounds, 0, self.tomo_recon_stacks[0].shape[1]):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1759 msnc.illegal_value('x_bounds', x_bounds, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1760 elif not self.test_mode:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1761 msnc.quickPlot(tomosum, title='recon stack sum yz')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1762 if pyip.inputYesNo('\nCurrent image x-bounds: '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1763 f'[{x_bounds[0]}, {x_bounds[1]}], use these values ([y]/n)? ',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1764 blank=True) == 'no':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1765 if pyip.inputYesNo(
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1766 'Do you want to change the image x-bounds ([y]/n)? ',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1767 blank=True) == 'no':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1768 x_bounds = [0, self.tomo_recon_stacks[0].shape[1]]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1769 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1770 x_bounds = msnc.selectArrayBounds(tomosum, title='recon stack sum yz')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1771 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1772 msnc.quickPlot(tomosum, title='recon stack sum yz')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1773 if pyip.inputYesNo('\nDo you want to change the image x-bounds (y/n)? ') == 'no':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1774 x_bounds = [0, self.tomo_recon_stacks[0].shape[1]]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1775 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1776 x_bounds = msnc.selectArrayBounds(tomosum, title='recon stack sum yz')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1777 if False and self.test_mode:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1778 np.savetxt(self.output_folder+'recon_stack_sum_yz.txt',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1779 tomosum[x_bounds[0]:x_bounds[1]], fmt='%.6e')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1780 if self.save_plots_only:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1781 msnc.clearFig('recon stack sum yz')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1782 tomosum = 0
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1783 #RV FIX :=
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1784 #[tomosum := tomosum+np.sum(tomo_recon_stack, axis=(0,1)) for tomo_recon_stack in
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1785 # self.tomo_recon_stacks]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1786 if combine_stacks and 'y_bounds' in combine_stacks:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1787 y_bounds = combine_stacks['y_bounds']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1788 if not msnc.is_index_range(x_bounds, 0, self.tomo_recon_stacks[0].shape[1]):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1789 msnc.illegal_value('y_bounds', y_bounds, 'config file')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1790 elif not self.test_mode:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1791 msnc.quickPlot(tomosum, title='recon stack sum xz')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1792 if pyip.inputYesNo('\nCurrent image y-bounds: '+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1793 f'[{y_bounds[0]}, {y_bounds[1]}], use these values ([y]/n)? ',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1794 blank=True) == 'no':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1795 if pyip.inputYesNo(
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1796 'Do you want to change the image y-bounds ([y]/n)? ',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1797 blank=True) == 'no':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1798 y_bounds = [0, self.tomo_recon_stacks[0].shape[1]]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1799 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1800 y_bounds = msnc.selectArrayBounds(tomosum, title='recon stack sum yz')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1801 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1802 msnc.quickPlot(tomosum, title='recon stack sum xz')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1803 if pyip.inputYesNo('\nDo you want to change the image y-bounds (y/n)? ') == 'no':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1804 y_bounds = [0, self.tomo_recon_stacks[0].shape[1]]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1805 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1806 y_bounds = msnc.selectArrayBounds(tomosum, title='recon stack sum xz')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1807 if False and self.test_mode:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1808 np.savetxt(self.output_folder+'recon_stack_sum_xz.txt',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1809 tomosum[y_bounds[0]:y_bounds[1]], fmt='%.6e')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1810 if self.save_plots_only:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1811 msnc.clearFig('recon stack sum xz')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1812
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1813 # Combine reconstructed tomography stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1814 logging.info(f'Combining reconstructed stacks ...')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1815 t0 = time()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1816 num_tomo_stacks = self.config['stack_info']['num']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1817 if num_tomo_stacks == 1:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1818 low_bound = row_bounds[0]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1819 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1820 low_bound = 0
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1821 tomo_recon_combined = self.tomo_recon_stacks[0][low_bound:row_bounds[1]:,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1822 x_bounds[0]:x_bounds[1],y_bounds[0]:y_bounds[1]]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1823 if num_tomo_stacks > 2:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1824 tomo_recon_combined = np.concatenate([tomo_recon_combined]+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1825 [self.tomo_recon_stacks[i][row_bounds[0]:row_bounds[1],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1826 x_bounds[0]:x_bounds[1],y_bounds[0]:y_bounds[1]]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1827 for i in range(1, num_tomo_stacks-1)])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1828 if num_tomo_stacks > 1:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1829 tomo_recon_combined = np.concatenate([tomo_recon_combined]+
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1830 [self.tomo_recon_stacks[num_tomo_stacks-1][row_bounds[0]:,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1831 x_bounds[0]:x_bounds[1],y_bounds[0]:y_bounds[1]]])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1832 logging.info(f'... done in {time()-t0:.2f} seconds!')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1833 tomosum = np.sum(tomo_recon_combined, axis=(1,2))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1834 if self.test_mode:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1835 zoom_perc = self.config['preprocess'].get('zoom_perc', 100)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1836 filename = 'recon combined sum xy'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1837 if zoom_perc is None or zoom_perc == 100:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1838 filename += ' fullres.dat'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1839 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1840 filename += f' {zoom_perc}p.dat'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1841 msnc.quickPlot(tomosum, title='recon combined sum xy',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1842 path=self.output_folder, save_fig=self.save_plots,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1843 save_only=self.save_plots_only)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1844 if False:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1845 np.savetxt(self.output_folder+'recon_combined_sum_xy.txt',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1846 tomosum, fmt='%.6e')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1847 np.savetxt(self.output_folder+'recon_combined.txt',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1848 tomo_recon_combined[int(tomo_recon_combined.shape[0]/2),:,:], fmt='%.6e')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1849 combine_stacks =self.config.get('combine_stacks')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1850
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1851 # Update config and save to file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1852 if combine_stacks:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1853 combine_stacks['x_bounds'] = x_bounds
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1854 combine_stacks['y_bounds'] = y_bounds
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1855 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1856 self.config['combine_stacks'] = {'x_bounds' : x_bounds, 'y_bounds' : y_bounds}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1857 self.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1858 return
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1859 msnc.quickPlot(tomosum, title='recon combined sum xy')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1860 if pyip.inputYesNo(
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1861 '\nDo you want to change the image z-bounds (y/[n])? ',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1862 blank=True) != 'yes':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1863 z_bounds = [0, tomo_recon_combined.shape[0]]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1864 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1865 z_bounds = msnc.selectArrayBounds(tomosum, title='recon combined sum xy')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1866 if z_bounds[0] != 0 or z_bounds[1] != tomo_recon_combined.shape[0]:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1867 tomo_recon_combined = tomo_recon_combined[z_bounds[0]:z_bounds[1],:,:]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1868 logging.info(f'tomo_recon_combined.shape = {tomo_recon_combined.shape}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1869 if self.save_plots_only:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1870 msnc.clearFig('recon combined sum xy')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1871
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1872 # Plot center slices
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1873 msnc.quickImshow(tomo_recon_combined[int(tomo_recon_combined.shape[0]/2),:,:],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1874 title=f'recon combined xslice{int(tomo_recon_combined.shape[0]/2)}',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1875 path=self.output_folder, save_fig=self.save_plots,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1876 save_only=self.save_plots_only)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1877 msnc.quickImshow(tomo_recon_combined[:,int(tomo_recon_combined.shape[1]/2),:],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1878 title=f'recon combined yslice{int(tomo_recon_combined.shape[1]/2)}',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1879 path=self.output_folder, save_fig=self.save_plots,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1880 save_only=self.save_plots_only)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1881 msnc.quickImshow(tomo_recon_combined[:,:,int(tomo_recon_combined.shape[2]/2)],
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1882 title=f'recon combined zslice{int(tomo_recon_combined.shape[2]/2)}',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1883 path=self.output_folder, save_fig=self.save_plots,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1884 save_only=self.save_plots_only)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1885
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1886 # Save combined reconstructed tomo stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1887 base_name = 'recon combined'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1888 combined_stacks = []
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1889 for stack in stacks:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1890 base_name += f' {stack["index"]}'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1891 combined_stacks.append(stack['index'])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1892 self._saveTomo(base_name, tomo_recon_combined)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1893
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1894 # Update config and save to file
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1895 if combine_stacks:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1896 combine_stacks['x_bounds'] = x_bounds
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1897 combine_stacks['y_bounds'] = y_bounds
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1898 combine_stacks['stacks'] = combined_stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1899 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1900 self.config['combine_stacks'] = {'x_bounds' : x_bounds, 'y_bounds' : y_bounds,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1901 'stacks' : combined_stacks}
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1902 self.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1903
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1904 def runTomo(config_file=None, config_dict=None, output_folder='.', log_level='INFO',
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1905 test_mode=False):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1906 """Run a tomography analysis.
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1907 """
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1908 # Instantiate Tomo object
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1909 tomo = Tomo(config_file=config_file, output_folder=output_folder, log_level=log_level,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1910 test_mode=test_mode)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1911 if not tomo.is_valid:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1912 raise ValueError('Invalid config and/or detector file provided.')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1913
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1914 # Preprocess the image files
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1915 num_tomo_stacks = tomo.config['stack_info']['num']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1916 assert(num_tomo_stacks == len(tomo.tomo_stacks))
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1917 preprocess = tomo.config.get('preprocess', None)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1918 preprocessed_stacks = []
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1919 if preprocess:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1920 preprocessed_stacks = [stack['index'] for stack in tomo.config['stack_info']['stacks']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1921 if stack.get('preprocessed', False)]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1922 if len(preprocessed_stacks):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1923 tomo.genTomoStacks()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1924 if not tomo.is_valid:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1925 IOError('Unable to load all required image files.')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1926 find_center = tomo.config.get('find_center')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1927 if find_center and find_center.get('completed', False):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1928 center_stack_index = find_center['center_stack_index']
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1929 if not center_stack_index in preprocessed_stacks:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1930 find_center['completed'] = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1931 #RV FIX
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1932 # tomo.config.pop('check_center', 'check_center not found')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1933 # combined_stacks = tomo.config.get('combined_stacks')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1934 # if combined_stacks:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1935 # combined_stacks['completed'] = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1936 tomo.cf.saveFile(self.config_out)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1937
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1938 # Find centers
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1939 find_center = tomo.config.get('find_center')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1940 if find_center is None or not find_center.get('completed', False):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1941 tomo.findCenters()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1942
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1943 # Check centers
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1944 #if num_tomo_stacks > 1 and not tomo.config.get('check_centers', False):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1945 # tomo.checkCenters()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1946
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1947 # Reconstruct tomography stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1948 if len(tomo.config.get('reconstructed_stacks', [])) != tomo.config['stack_info']['num']:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1949 tomo.reconstructTomoStacks()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1950
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1951 # Combine reconstructed tomography stacks
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1952 combined_stacks = tomo.config.get('combined_stacks')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1953 if combined_stacks is None or not combined_stacks.get('completed', False):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1954 tomo.combineTomoStacks()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1955
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1956 #%%============================================================================
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1957 if __name__ == '__main__':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1958 # Parse command line arguments
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1959 arguments = sys.argv[1:]
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1960 config_file = None
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1961 output_folder = '.'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1962 log_level = 'INFO'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1963 test_mode = False
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1964 try:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1965 opts, args = getopt.getopt(arguments,"hc:o:l:t")
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1966 except getopt.GetoptError:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1967 print('usage: tomo.py -c <config_file> -o <output_folder> -l <log_level> -t')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1968 sys.exit(2)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1969 for opt, arg in opts:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1970 if opt == '-h':
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1971 print('usage: tomo.py -c <config_file> -o <output_folder> -l <log_level> -t')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1972 sys.exit()
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1973 elif opt in ("-c"):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1974 config_file = arg
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1975 elif opt in ("-o"):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1976 output_folder = arg
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1977 elif opt in ("-l"):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1978 log_level = arg
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1979 elif opt in ("-t"):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1980 test_mode = True
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1981 if config_file is None:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1982 if os.path.isfile('config.yaml'):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1983 config_file = 'config.yaml'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1984 else:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1985 config_file = 'config.txt'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1986
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1987 # Set basic log configuration
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1988 logging_format = '%(asctime)s : %(levelname)s - %(module)s : %(funcName)s - %(message)s'
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1989 if not test_mode:
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1990 level = getattr(logging, log_level.upper(), None)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1991 if not isinstance(level, int):
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1992 raise ValueError(f'Invalid log_level: {log_level}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1993 logging.basicConfig(format=logging_format, level=level, force=True,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1994 handlers=[logging.StreamHandler()])
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1995
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1996 logging.debug(f'config_file = {config_file}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1997 logging.debug(f'output_folder = {output_folder}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1998 logging.debug(f'log_level = {log_level}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
1999 logging.debug(f'test_mode = {test_mode}')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
2000
2718cbde04fe Uploaded
rv43
parents:
diff changeset
2001 # Run tomography analysis
2718cbde04fe Uploaded
rv43
parents:
diff changeset
2002 runTomo(config_file=config_file, output_folder=output_folder, log_level=log_level,
2718cbde04fe Uploaded
rv43
parents:
diff changeset
2003 test_mode=test_mode)
2718cbde04fe Uploaded
rv43
parents:
diff changeset
2004
2718cbde04fe Uploaded
rv43
parents:
diff changeset
2005 #%%============================================================================
2718cbde04fe Uploaded
rv43
parents:
diff changeset
2006 input('Press any key to continue')
2718cbde04fe Uploaded
rv43
parents:
diff changeset
2007 #%%============================================================================