-
Notifications
You must be signed in to change notification settings - Fork 3
/
casapy2bbs_one_patch_per_cc.py
executable file
·337 lines (274 loc) · 13.4 KB
/
casapy2bbs_one_patch_per_cc.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
#!/usr/bin/env python
import sys
import numpy
import scipy.ndimage
import pyrap.images
from optparse import OptionParser
# Return the fractional part of the floating point number x.
def fraction(x):
return numpy.modf(x)[0]
# Convert an angle from degrees to radians.
def deg2rad(angle):
return (angle * numpy.pi) / 180.0
# Convert an angle from radians to degrees.
def rad2deg(angle):
return (angle * 180.0) / numpy.pi
# Compute hours, min, sec from an angle in radians.
def rad2ra(angle):
deg = numpy.fmod(rad2deg(angle), 360.0)
if deg < 0.0:
deg += 360.0
# Ensure positive output (deg could equal -0.0).
deg = abs(deg)
assert(deg < 360.0)
return (int(deg / 15.0), int(fraction(deg / 15.0) * 60.0), fraction(deg * 4.0) * 60.0)
# Compute degrees, arcmin, arcsec from an angle in radians, in the range [-90.0, +90.0].
def rad2dec(angle):
deg = numpy.fmod(rad2deg(angle), 360.0)
if deg > 180.0:
deg -= 360.0
elif deg < -180.0:
deg += 360.0
sign = (deg < 0.0)
deg = abs(deg)
assert(deg <= 90.0)
return (sign, int(deg), int(fraction(deg) * 60.0), fraction(deg * 60.0) * 60.0)
# Return string representation of the input right ascension (as returned by rad2ra).
def ra2str(ra):
return "%02d:%02d:%07.4f" % ra
# Return string representation of the input declination (as returned by rad2dec).
def dec2str(dec):
return "%s%02d.%02d.%07.4f" % ("-" if dec[0] else "+", dec[1], dec[2], dec[3])
# Recursively flatten a list.
def flatten_list(l):
def flatten_rec(l, out):
if type(l) == list:
for el in l:
flatten_rec(el, out)
else:
out.append(l)
out = []
flatten_rec(l, out)
return out
# Compute an index that maps each element in requested to the index of the same
# element in available. If such an element cannot be found and mandatory is
# set to True an exception is raised. If mandatory is set to False, None is
# stored in the index instead.
def build_index(available, requested, mandatory=True):
index = [None for i in range(len(requested))]
for i in range(len(requested)):
try:
index[i] = available.index(requested[i])
except ValueError:
if mandatory:
raise
return index
# Return a sequence that contains the index of each element in the input
# sequence if the input sequence would be sorted (ascendingly).
#
# For example: [4,2,9,11] => [1,0,2,3]
#
def renumber(sequence):
index = sorted(range(len(sequence)), key=lambda x: sequence[x])
return sorted(range(len(index)), key=lambda x: index[x])
# Load pixel data from the input CASA image and create a slice that contains
# only the requested axes in order. Zero is used as the coordinate for each
# additional axis that the image contains when creating the slice. Both the
# slice and the index of each requested axis in the original image is returned.
def load_data_and_reformat(im, axes):
# Get image coordinate system and find the index of the requested axes.
csys = im.coordinates()
try:
index = build_index(flatten_list(csys.get_axes()), axes)
except:
raise Exception("error: incompatible image format: one or more required axes undefined.")
# Load pixel data.
data = im.getdata()
# Create slice.
slicer = [0 for i in range(len(data.shape))]
for i in range(len(axes)):
slicer[index[i]] = slice(data.shape[index[i]])
return (numpy.transpose(data[slicer], renumber(index)), index)
def main(options, args):
assert(len(args) >= 1)
# Load CLEAN component map and reformat.
component_map_im = pyrap.images.image(args[0])
component_map, axis_index = load_data_and_reformat(component_map_im, ["Stokes", "Declination", "Right Ascension"])
# Ugly kludge to find out mapping from Stokes coordinates to indices. No
# better alternative seems to be available using pyrap at this time.
stokes = component_map_im.coordinates().get_coordinate("stokes")._coord["stokes"]
stokes_index = build_index(stokes, ["I", "Q", "U", "V"], False)
# Make sure Stokes I is available.
if stokes_index[0] is None:
print "error: incompatible CLEAN component image format: Stokes I unavailable."
sys.exit(1)
# Find locations of all CLEAN components (pixels with non-zero flux).
component_flux = component_map[stokes_index[0], :, :]
components = numpy.nonzero(component_flux)
# Compute statistics.
total_component_count = len(components[0])
print "info: total number of CLEAN components: %d" % total_component_count
if total_component_count == 0:
print "error: no CLEAN components found in CASA image:", args[0]
sys.exit(1)
total_component_flux = numpy.sum(component_flux[components])
print "info: total flux in CLEAN components: %.2f Jy" % total_component_flux
have_mask = not options.mask is None
if have_mask:
# Load CASA mask image and reformat.
mask_im = pyrap.images.image(options.mask)
mask, mask_index = load_data_and_reformat(mask_im, ["Declination", "Right Ascension"])
# Sanity check mask image dimensions.
if mask.shape != component_flux.shape:
print "error: CASA mask image should have the same dimensions as the CLEAN component image."
sys.exit(1)
# Find islands in the mask image.
(island_label, island_count) = scipy.ndimage.measurements.label(mask)
if island_count == 0:
print "error: no islands found in CASA mask image:", options.mask
sys.exit(1)
# Assign CLEAN components to the corresponding (co-located) islands and
# keep track of the total flux in each island. Also compute the total
# flux present in all the islands.
total_flux = 0.0
islands = [[0.0, []] for i in range(island_count)]
for component in zip(*components):
# Get the label of the island at the position of the CLEAN
# component.
label = island_label[component]
# Label 0 is used for the background of the mask image and as
# such does not define an island, so skip it.
if label == 0:
continue
# Get the absolute flux for the current CLEAN component.
flux = component_flux[component]
# Append CLEAN component to the right patch and update the patch
# flux.
islands[label - 1][0] += flux
islands[label - 1][1].append(component)
# Update total flux.
total_flux += flux
# Prune islands that do not contain any CLEAN components.
islands = filter(lambda x: len(x[1]) > 0, islands)
# Print total number of non-empty island.
print "info: total number of non-empty islands: %d" % len(islands)
print "info: total flux in non-empty islands: %.2f Jy" % total_flux
# Determine clip level.
clip_flux = options.clip_level / 100.0 * total_flux
print "info: clipping at: %.2f Jy" % clip_flux
# Sort islands by absolute flux (descending).
islands = sorted(islands, key = lambda x: abs(x[0]), reverse = True)
# Add islands to the patch list until the absolute flux clip level is
# reached.
patches = []
component_count = 0
sum_flux = 0.0
for island in islands:
if options.clip_level < 100.0 and sum_flux + island[0] > clip_flux:
break
sum_flux += island[0]
component_count += len(island[1])
patches.append(island[1])
print "info: number of non-empty islands selected: %d" % len(patches),
if len(islands) > 0:
print "(%.2f%%)" % ((100.0 * len(patches)) / len(islands))
else:
print "(100.00%)"
if len(patches) == 0:
print "warning: no non-empty islands selected; you may need to raise the clip level (option -c)."
print "info: number of CLEAN components in selected non-empty islands: %d (%.2f%%)" % (component_count, (100.0 * component_count) / total_component_count)
print "info: flux in selected non-empty islands: %.2f Jy (%.2f%%)" % (sum_flux, (100.0 * sum_flux) / total_flux)
else:
# Without a mask we are selecting CLEAN components, so the total
# absolute flux is equal to the sum of the absolute flux of all CLEAN
# components.
total_flux = total_component_flux
# Determine clip level.
clip_flux = options.clip_level / 100.0 * total_flux
print "info: clipping at: %.2f Jy" % clip_flux
# Sort CLEAN components on absolute flux (descending).
components = sorted(zip(*components),
key = lambda x: abs(component_flux[x]), reverse = True)
# Create a single patch and add CLEAN components to it until the
# absolute flux clip level is reached.
patch = []
sum_flux = 0.0
for component in components:
flux = component_flux[component]
if options.clip_level < 100.0 and sum_flux + flux > clip_flux:
break
sum_flux += flux
patch.append(component)
patches = []
if len(patch) > 0:
patches.append(patch)
print "info: number of CLEAN components selected: %d (%.2f%%)" % (len(patch), (100.0 * len(patch)) / total_component_count)
if len(patch) == 0:
print "warning: no CLEAN components selected; you may need to raise the clip level (option -c)."
print "info: flux in selected CLEAN components: %.2f Jy (%.2f%%)" % (sum_flux, (100.0 * sum_flux) / total_flux)
# Open output file.
out = sys.stdout
if len(args) == 1:
out = file("%s.catalog" % args[0], 'w')
elif args[1] != "-":
out = file(args[1], 'w')
# Write the catalog header.
if options.use_patches:
print >>out, "# (Name, Type, Patch, Ra, Dec, I, Q, U, V) = format"
else:
print >>out, "# (Name, Type, Ra, Dec, I, Q, U, V) = format"
print >>out
print >>out, "# CLEAN component list converted from:", args[0]
if not options.mask is None:
print >>out, "# Mask:", options.mask
print >>out, "# Total flux in CLEAN components: %.2f Jy" % total_flux
print >>out, "# Percentage of total flux kept: %.2f%%" % options.clip_level
print >>out
# Output all the patches in BBS catalog file format.
patch_count = 0
component_count = 0
stokes_desc = ["" for i in range(len(stokes_index))]
pixel_coord = [0 for i in range(len(component_map_im.shape()))]
for patch in patches:
# Output patch definition.
#if options.use_patches:
# print >>out, ", , patch-%d, 00:00:00, +90.00.00" % patch_count
# When using patches, reset the component count at the start of each
# patch. Thus, components within a patch are counted from zero.
# Components are labelled both with a patch count and a component count
# to avoid name clashes. If not using patches, just continue counting.
if options.use_patches:
component_count = 0
# Output all the CLEAN components in the patch.
basename = args[0]
basename = basename.split('.')[0]
for component in patch:
if options.use_patches:
print >>out, ", , %spatch-%d, 00:00:00, +90.00.00" % (basename,component_count)
print >>out, "%spatch-%d-%d, POINT, %spatch-%d," % (basename,patch_count, component_count, basename,component_count),
else:
print >>out, "component-%d, POINT," % (component_count),
pixel_coord[axis_index[1]] = component[0]
pixel_coord[axis_index[2]] = component[1]
world_coord = component_map_im.toworld(pixel_coord)
print >>out, "%s," % ra2str(rad2ra(world_coord[axis_index[2]])),
print >>out, "%s," % dec2str(rad2dec(world_coord[axis_index[1]])),
for i in range(len(stokes_index)):
if stokes_index[i] is None:
stokes_desc[i] = ""
else:
stokes_desc[i] = "%f" % component_map[(stokes_index[i], component[0], component[1])]
print >>out, ", ".join(stokes_desc)
component_count += 1
print >>out
patch_count += 1
parser = OptionParser(usage="%prog [options] <CLEAN component image> [output catalog file]")
parser.add_option("-n", "--no-patches", action="store_false", dest="use_patches", default=True, help="Do not group components into patches. All CLEAN components will be written as separate components.")
parser.add_option("-m", "--mask", dest="mask", help="CASA mask image; If provided, the islands in the mask image define separate patches; CLEAN components that do not fall on an island are discarded, as are islands that do not contain any CLEAN components.")
parser.add_option("-c", "--clip-level", type="float", dest="clip_level", default=100.0, help="Percentage [0.0, 100.0] of the total flux that should be kept (default: %default). If there are CLEAN components with negative flux in the CLEAN component image, you may not recover all CLEAN components even if you specify 100% here. This is due to the way the cumulative flux builds up in the presence of negative components.")
(options, args) = parser.parse_args()
if options.clip_level < 0.0 or options.clip_level > 100.0:
parser.error("option -c: invalid percentage: %.2f" % options.clip_level)
if len(args) < 1 or len(args) > 2:
parser.error("incorrect number of arguments")
main(options, args)