-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathppspline.py
executable file
·383 lines (346 loc) · 18.6 KB
/
ppspline.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
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
#!/usr/bin/env python
############
# ppspline #
############
#ppspline is a command-line program to make a frequency-parameterized model of
# wideband profile evolution. The parameterization is good within the range
# of data frequencies, provided there are not huge gaps in frequency. For
# an input nchan x nbin average portrait of aligned profiles, the model is a
# B-spline representation of the curve traced out by the nchan profile
# amplitude vectors in an nbin vector space. Since the profile shapes are
# highly correlated, the B-spline representation can be reduced to << nbin
# dimensions, and is limited to ten dimensions. Therefore, the profile
# variability is decomposed using PCA and a small number of eigenprofiles
# that encompass most of the profile evolution are selected. The input data
# should be high S/N, averaged, and "aligned" (e.g. output from
# ppalign.py). Pre-normalization of the input is encouraged (specifically
# using normalize_portrait('prof')), as is using smooth=True in the
# make_model function.
#Written by Timothy T. Pennucci (TTP; [email protected]).
from pplib import *
class DataPortrait(DataPortrait):
"""
DataPortrait is a class that contains the data to which a model is fit.
This class adds methods and attributes to the parent class specific to
modeling profile evolution with a B-spline curve.
"""
def make_spline_model(self, max_ncomp=10, smooth=True, snr_cutoff=150.0,
rchi2_tol=0.1, k=3, sfac=1.0, max_nbreak=None, model_name=None,
quiet=False, **kwargs):
"""
Make a model based on PCA and B-spline interpolation.
max_ncomp is the maximum number of PCA components to use in the
B-spline parameterization; max_ncomp <= 10.
smooth=True will smooth the eigenvectors and mean profile using
a reduced chi-squared figure-of-merit.
snr_cutoff is the S/N ratio value above or equal to which an
eigenvector is deemed "significant". Setting it equal to np.inf
would ensure only a mean profile model is returned.
rchi2_tol is the tolerance parameter that will allow greater deviations
in the smoothed profile from the input profiles' shapes.
k is the polynomial degree of the spline; cubic splines (k=3)
recommended; 1 <= k <= 5. NB: polynomial order = degree + 1.
sfac is a multiplicative smoothing factor passed to si.splprep; greater
values result in more smoothing. sfac=0 will make an interpolating
model anchored on the input data profiles.
max_nbreak is the maximum number of breakpoints (unique knots) to
allow. If provided, this may override sfac and enforce smoothing
based on max_nbreak breakpoints. That is, if the fit returns n >
max_nbreak breakpoints, it will refit using maximum max_nbreak
breakpoints, irrespective of the other smoothing condition. The
corresponding maximum number of B-splines will be max_nspline =
max_nbreak + k - 1. max_nbreak should be >= 2.
model_name is the name of the model; defaults to self.datafile +
'.spl'
quiet=True suppresses output.
**kwargs get passed to find_significant_eigvec(...).
"""
#Definitions
port = self.portx
pca_weights = self.SNRsxs / np.sum(self.SNRsxs)
mean_prof = (port.T * pca_weights).T.sum(axis=0) / pca_weights.sum()
freqs = self.freqsxs[0]
nu_lo = freqs.min()
nu_hi = freqs.max()
#Check nbin
nbin = port.shape[1]
if nbin % 2 != 0:
print "nbin = %d is odd; cannot wavelet_smooth.\n"%nbin
smooth = False
elif np.modf(np.log2(nbin))[0] != 0.0:
print "nbin = %d is not a power of two; can only try wavelet_smooth to one level; recommend resampling to a power-of-two number of phase bins.\n"%nbin
#Do principal component analysis
eigval, eigvec = pca(port, mean_prof, pca_weights, quiet=quiet)
#Get "significant" eigenvectors
if max_ncomp is None:
return_max = 10
else:
return_max = min(max_ncomp, 10)
if smooth:
if 'pywt' not in sys.modules:
raise ImportError("You failed to import pywt and need PyWavelets to use smooth=True!")
ieig, smooth_eigvec = find_significant_eigvec(eigvec, check_max=10,
return_max=return_max, snr_cutoff=snr_cutoff,
return_smooth=True, rchi2_tol=rchi2_tol, **kwargs)
else:
ieig = find_significant_eigvec(eigvec, check_max=10,
return_max=return_max, snr_cutoff=snr_cutoff,
return_smooth=False, rchi2_tol=rchi2_tol, **kwargs)
ncomp = len(ieig)
if smooth:
smooth_mean_prof = smart_smooth(mean_prof,
rchi2_tol=rchi2_tol)
if ncomp == 0: #Will make model with constant average port
proj_port = port[:,:ncomp]
if smooth:
modelx = reconst_port = np.tile(smooth_mean_prof,
len(freqs)).reshape(len(freqs), port.shape[1])
model = np.tile(smooth_mean_prof,
len(self.freqs[0])).reshape(len(self.freqs[0]),
port.shape[1])
else:
modelx = reconst_port = np.tile(mean_prof,
len(freqs)).reshape(len(freqs), port.shape[1])
model = np.tile(mean_prof,
len(self.freqs[0])).reshape(len(self.freqs[0]),
port.shape[1])
else:
delta_port = port - mean_prof
if smooth:
reconst_port = reconstruct_portrait(port, mean_prof,
smooth_eigvec[:,ieig])
#Find the projections of the profiles onto the basis components
proj_port = np.dot(delta_port, smooth_eigvec[:,ieig])
else:
reconst_port = reconstruct_portrait(port, mean_prof,
eigvec[:,ieig])
#Find the projections of the profiles onto the basis components
proj_port = np.dot(delta_port, eigvec[:,ieig])
if ncomp == 0:
(tck,u) = [np.array([]), np.array([]), 0], np.array([])
fp, ier, msg = None, None, None
else:
spl_weights = pca_weights
s = sfac * len(proj_port) * \
np.sum((self.SNRsxs * self.noise_stdsxs)**2) / \
sum(self.SNRsxs)**2
if self.bw < 0: flip = -1 #u in si.splprep has to be increasing...
else: flip = 1
#Find the B-spline curve traced by the projected vectors,
#parameterized by frequency
(tck,u), fp, ier, msg = si.splprep(proj_port[::flip].T,
w=spl_weights[::flip], u=freqs[::flip], ub=nu_lo, ue=nu_hi,
k=k, task=0, s=s, t=None, full_output=1, nest=None, per=0,
quiet=int(quiet))
if max_nbreak is not None and len(np.unique(tck[0])) > max_nbreak:
if max_nbreak < 2:
print "max_nbreak not >= 2; setting max_nbreak = 2..."
max_nbreak = 2
if max_nbreak == 2: s = np.inf
(tck,u), fp, ier, msg = si.splprep(proj_port[::flip].T,
w=spl_weights[::flip], u=freqs[::flip], ub=nu_lo,
ue=nu_hi, k=k, task=0, s=s, t=None, full_output=1,
nest=max_nbreak+(k*2), per=0, quiet=int(quiet))
if ier > 1: #Will also catch when ier == "unknown"
print "Something went wrong in si.splprep for %s:\n%s"%(
self.source, msg)
#Build model
if ncomp != 0:
if smooth:
modelx = gen_spline_portrait(smooth_mean_prof, freqs,
smooth_eigvec[:,ieig], tck)
model = gen_spline_portrait(smooth_mean_prof, self.freqs[0],
smooth_eigvec[:,ieig], tck)
else:
modelx = gen_spline_portrait(mean_prof, freqs, eigvec[:,ieig],
tck)
model = gen_spline_portrait(mean_prof, self.freqs[0],
eigvec[:,ieig], tck)
#Assign new attributes
self.ieig = ieig
self.ncomp = ncomp
self.eigvec = eigvec
self.eigval = eigval
self.mean_prof = mean_prof
if smooth:
self.smooth_mean_prof = smooth_mean_prof
self.smooth_eigvec = smooth_eigvec
self.proj_port = proj_port
self.reconst_port = reconst_port
#tck contains the knot locations t, B-spline coefficients c, and
#polynomial degree k -- end knots will have multiplicity k+1, interior
#breakpoints will have multiplicity 1 for maximum continuity. The
#number of B-splines will be n = l + k, where l is the number of
#intervals. l = number of breakpoints - 1 = number of unique knots - 1
# = len(tck[0]) - 2*tck[2] - 1.
self.tck, self.u, self.fp, self.ier, self.msg = tck, u, fp, ier, msg
if model_name is None: self.model_name = self.datafile + '.spl'
else: self.model_name = model_name
self.model = model
self.modelx = modelx
self.model_masked = self.model * self.masks[0,0]
if not quiet:
if proj_port.sum():
print "B-spline interpolation model %s uses %d basis profile components and %d breakpoints (%d B-splines with k=%d)."%(self.model_name, ncomp,
len(np.unique(self.tck[0])),
len(self.tck[0])-self.tck[2]-1, self.tck[2])
else:
print "B-spline interpolation model %s uses 0 basis profile components; it returns the average profile."%(self.model_name)
def write_model(self, outfile, quiet=False):
"""
Write the output (pickle file) model to outfile.
"""
of = open(outfile, "wb")
if hasattr(self, "smooth_eigvec"):
if len(self.ieig):
pickle.dump([self.model_name, self.source, self.datafile,
self.smooth_mean_prof, self.smooth_eigvec[:,self.ieig],
self.tck], of, protocol=2)
else:
pickle.dump([self.model_name, self.source, self.datafile,
self.smooth_mean_prof, self.smooth_eigvec[:,[]],
self.tck], of, protocol=2)
else:
if len(self.ieig):
pickle.dump([self.model_name, self.source, self.datafile,
self.mean_prof, self.eigvec[:,self.ieig], self.tck], of,
protocol=2)
else:
pickle.dump([self.model_name, self.source, self.datafile,
self.mean_prof, self.eigvec[:,[]], self.tck], of,
protocol=2)
of.close()
if not quiet:
print "Wrote modelfile %s."%outfile
def show_eigenprofiles(self, ncomp=None, title=None, **kwargs):
"""
Calls show_eigenprofiles(...) to make plots of mean/eigen profiles.
see show_eigenprofiles(...) for details.
ncomp=None plots self.ncomp PCA components, otherwise plots the number
of components specified.
**kwargs get passed to show_eigenprofiles(...).
"""
if ncomp is None: ncomp = self.ncomp
if hasattr(self, "smooth_eigvec"):
if ncomp:
eigvec = self.eigvec[:,self.ieig[:ncomp]].T
seigvec = self.smooth_eigvec[:,self.ieig[:ncomp]].T
else:
eigvec = None
seigvec = None
show_eigenprofiles(eigvec, seigvec, self.mean_prof,
self.smooth_mean_prof, title=title, **kwargs)
else:
if ncomp:
eigvec = self.eigvec[:,self.ieig[:ncomp]].T
else:
eigvec = None
show_eigenprofiles(eigvec, None, self.mean_prof, None, title=title,
**kwargs)
def show_spline_curve_projections(self, ncomp=None, title=None, **kwargs):
"""
Calls show_spline_curve_projections(...) to make plots of the model.
see show_spline_curve_projections(...) for details.
ncomp=None plots self.ncomp PCA components, otherwise plots the number
of components specified.
**kwargs get passed to show_spline_curve_projections(...).
"""
if ncomp is None: ncomp = self.ncomp
if ncomp:
show_spline_curve_projections(self.proj_port, self.tck,
self.freqsxs[0], self.SNRsxs / np.sum(self.SNRsxs),
ncoord=ncomp, title=title, **kwargs)
if __name__ == "__main__":
from optparse import OptionParser
usage = "Usage: %prog -d <datafile> [options]"
parser = OptionParser(usage)
#parser.add_option("-h", "--help",
# action="store_true", dest="help", default=False,
# help="Show this help message and exit.")
parser.add_option("-d", "--datafile",
action="store", metavar="archive", dest="datafile",
help="PSRCHIVE archive from which to make model, or a metafile listing multiple archives (i.e., from different bands). If providing a metafile, the achives must already be aligned.")
parser.add_option("-o", "--modelfile",
action="store", metavar="modelfile", dest="modelfile",
help="Name for output model (pickle) file. [default=datafile.spl].")
parser.add_option("-l", "--model_name",
action="store", metavar="model_name", dest="model_name",
default=None,
help="Optional name for model [default=datafile.spl].")
parser.add_option("-a", "--archive",
action="store", metavar="archive", dest="archive",
default=None,
help="Name for optional output PSRCHIVE archive. Will work only if the input is a single archive.")
parser.add_option("-N", "--norm",
action="store", metavar="normalization", dest="norm",
default="prof",
help="Normalize the input data by channel ('None', 'mean', 'max' (not recommended), 'rms' (off-pulse noise), 'prof' (mean profile flux) [default], or 'abs' (sqrt{vector modulus})).")
parser.add_option("-s", "--smooth",
action="store_true", metavar="smooth", dest="smooth",
default=False,
help="Smooth the eigenvectors and mean profile [recommended] using default wavelet_smooth options and smart_smooth.")
parser.add_option("-n", "--max_ncomp",
action="store", metavar="max_ncomp", dest="max_ncomp",
default=10,
help="Maximum number of principal components to use in PCA reconstruction of the data. max_ncomp is limited to a maximum of 10 by the B-spline representation in scipy.interpolate.")
parser.add_option("-S", "--snr",
action="store", metavar="snr_cutoff", dest="snr_cutoff",
default=150.0,
help="S/N ratio cutoff for determining 'significant' eigenprofiles. A value somewhere over 100.0 should be good. [default=150.0].")
parser.add_option("-T", "--rchi2_tol",
action="store", metavar="tolerance", dest="rchi2_tol",
default=0.1,
help="Tweak this between 0.0 and 0.1 [default] if the returned eigenprofiles are not smooth enough.")
parser.add_option("-k", "--degree",
action="store", metavar="degree", dest="k", default=3,
help="Degree of the spline. Cubic splines (k=3) are recommended [default]. 1 <= k <=5.")
parser.add_option("-f", "--sfac",
action="store", metavar="smooth_factor", dest="sfac",
default=1.0,
help="To change the smoothness of the B-spline model, tweak this between 0.0 (interpolating spline that passes through all data points) and a large number (guarantees maximum two breakpoints = maximum smoothness). Alternatively, use -t.")
parser.add_option("-t", "--knots",
action="store", metavar="max_knots", dest="max_nbreak",
default=None,
help="The maximum number of unique knots. This functions esentially as an ignorant smoothing condition in case the default settings return a fit with more than max_knots number of unique knots in the spline model. e.g., 10 unique knots are more than usually necessary.")
parser.add_option("--plots",
action="store_true", dest="make_plots", default=False,
help="Save some plots related to the model with basename model_name (-l).")
parser.add_option("--quiet",
action="store_true", dest="quiet", default=False,
help="Suppresses output.")
(options, args) = parser.parse_args()
if (options.datafile is None):
print "\nppspline.py - make a pulse portrait model using PCA & B-spline interpolation\n"
parser.print_help()
print ""
parser.exit()
datafile = options.datafile
modelfile = options.modelfile
model_name = options.model_name
archive = options.archive
norm = options.norm
smooth = options.smooth
max_ncomp = int(options.max_ncomp)
snr_cutoff = float(options.snr_cutoff)
rchi2_tol = float(options.rchi2_tol)
k = int(options.k)
sfac = float(options.sfac)
if options.max_nbreak is not None: max_nbreak = int(options.max_nbreak)
else: max_nbreak = None
make_plots = options.make_plots
quiet = options.quiet
dp = DataPortrait(datafile, quiet=quiet)
if norm in ("mean", "max", "prof", "rms", "abs"):
dp.normalize_portrait(norm)
dp.make_spline_model(max_ncomp=max_ncomp, smooth=smooth,
snr_cutoff=snr_cutoff, rchi2_tol=rchi2_tol, k=k, sfac=sfac,
max_nbreak=max_nbreak, model_name=model_name, quiet=quiet)
if modelfile is None: modelfile = datafile + ".spl"
dp.write_model(modelfile, quiet=quiet)
if archive is not None and len(dp.datafiles) == 1:
dp.write_model_archive(archive, quiet=quiet)
if make_plots:
dp.show_eigenprofiles(title=dp.model_name, savefig=dp.model_name)
dp.show_spline_curve_projections(title=dp.model_name,
savefig=dp.model_name)
dp.show_model_fit(savefig=dp.model_name+'.resids.png')