-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmmdeconv.py
executable file
·111 lines (88 loc) · 3.8 KB
/
mmdeconv.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
#!/usr/bin/env python
import argparse
import numpy as np
from pathlib import Path
from mmtools import stack, deconvolve, log, gpuimage
# defaults
psf_folder = Path(__file__).parent.joinpath('psf')
input_filename = None
output_filename = None
output_suffix = '_deconv.tif'
psf_filename = None
psf_iso = 'dispim_iso_bw.tif'
psf_noniso = 'dispim_0.5um_bw.tif'
psf_2d = 'dispim_2d_bw.tif'
iterations = 10
parser = argparse.ArgumentParser(description = 'Deconvolve images using the Richardson-Lucy algorhythm', \
formatter_class=argparse.ArgumentDefaultsHelpFormatter)
parser.add_argument('-o', '--output-file', default = output_filename, \
help='Output image file name ([basename]{0} if not specified)'.format(output_suffix))
gpuimage.add_gpu_argument(parser)
parser.add_argument('-p', '--psf-image', default = psf_filename, \
help='Name of psf image in the current or system folders. None: {0}'.format([psf_iso, psf_noniso, psf_2d]))
parser.add_argument('-i', '--iterations', type = int, default = iterations, \
help='Number of iterations')
parser.add_argument('-s', '--scale-isometric', action = 'store_true', \
help='Scale the image to make it isometric.')
parser.add_argument('-r', '--restore-scale', action = 'store_true', \
help='Restore the z scale of image after deconvolution')
log.add_argument(parser)
parser.add_argument('input_file', default = input_filename, \
help='Image file to deconvolve')
args = parser.parse_args()
# logging
logger = log.get_logger(__file__, level = args.log_level)
# turn on gpu
gpu_id = gpuimage.parse_gpu_argument(args)
# defaults
iterations = args.iterations
psf_filename = args.psf_image
scale_isometric = args.scale_isometric
restore_scale = args.restore_scale
input_filename = args.input_file
if args.output_file is None:
output_filename = stack.with_suffix(input_filename, output_suffix)
else:
output_filename = args.output_file
# load input image
logger.info("Loading image: {0}.".format(input_filename))
input_stack = stack.Stack(input_filename)
# load psf image
if psf_filename is None:
if input_stack.z_count > 1:
if scale_isometric:
psf_path = Path(psf_folder).joinpath(psf_iso)
else:
psf_path = Path(psf_folder).joinpath(psf_noniso)
else:
psf_path = Path(psf_folder).joinpath(psf_2d)
else:
if Path(psf_filename).exists():
psf_path = Path(psf_filename)
else:
psf_path = Path(psf_folder).joinpath(psf_filename)
logger.info("PSF image: {0}.".format(psf_path))
psf_stack = stack.Stack(psf_path)
psf_image = psf_stack.image_array[0, 0]
# setting image scale
pixel_orig = input_stack.voxel_um
if scale_isometric:
logger.info("Scaling image to be isometric. Pixel-size: {0}.".format(min(pixel_orig)))
input_stack.scale_isometric(gpu_id = gpu_id)
if np.allclose(input_stack.voxel_um, psf_stack.voxel_um, atol = 1e-2) == False:
logger.warning("Pixel sizes are different. Image: {0}. PSF: {1}.".format(input_stack.voxel_um, psf_stack.voxel_um))
# deconvolution
if input_stack.z_count > 1:
def deconvolve_image (image, t_index, c_index):
return deconvolve.deconvolve(image, psf_image, iterations = iterations, gpu_id = gpu_id)
else:
def deconvolve_image (image, t_index, c_index):
return deconvolve.deconvolve(image[0], psf_image[0], iterations = iterations, gpu_id = gpu_id)[np.newaxis]
logger.info("Deconvolution started.")
input_stack.apply_all(deconvolve_image, progress = True)
if restore_scale:
logger.info("Restoring scale: {0}.".format(pixel_orig))
input_stack.scale_by_pixelsize(pixel_orig, gpu_id = gpu_id)
# output image
logger.info("Saving image: {0}.".format(output_filename))
input_stack.save_ome_tiff(output_filename, dtype = np.float32)