Line data Source code
1 : //# CalSummary.cc: Helper class for listing a CalTable
2 : //# Copyright (C) 2018
3 : //# Associated Universities, Inc. Washington DC, USA.
4 : //#
5 : //# This library is free software; you can redistribute it and/or modify it
6 : //# under the terms of the GNU Library General Public License as published by
7 : //# the Free Software Foundation; either version 2 of the License, or (at your
8 : //# option) any later version.
9 : //#
10 : //# This library is distributed in the hope that it will be useful, but WITHOUT
11 : //# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
12 : //# FITNESS FOR A PARTICULAR PURPOSE. See the GNU Library General Public
13 : //# License for more details.
14 : //#
15 : //# You should have received a copy of the GNU Library General Public License
16 : //# along with this library; if not, write to the Free Software Foundation,
17 : //# Inc., 675 Massachusetts Ave, Cambridge, MA 02139, USA.
18 : //#
19 : //# Correspondence concerning AIPS++ should be addressed as follows:
20 : //# Internet email: aips2-request@nrao.edu.
21 : //# Postal address: AIPS++ Project Office
22 : //# National Radio Astronomy Observatory
23 : //# 520 Edgemont Road
24 : //# Charlottesville, VA 22903-2475 USA
25 : //#
26 : //#
27 :
28 : #include <synthesis/CalTables/CalSummary.h>
29 : #include <synthesis/CalTables/CalMainColumns.h>
30 : #include <synthesis/CalTables/CalDescColumns.h>
31 : #include <synthesis/CalTables/CalIter.h>
32 : #include <synthesis/CalTables/CalBuffer.h>
33 : #include <synthesis/CalTables/CalSetMetaInfo.h>
34 : #include <synthesis/MeasurementComponents/MSMetaInfoForCal.h>
35 :
36 : #include <casacore/casa/Quanta/QVector.h>
37 : #include <casacore/casa/iomanip.h>
38 : #include <casacore/casa/iostream.h>
39 : #include <casacore/casa/OS/Path.h>
40 : #include <casacore/measures/Measures/MeasTable.h>
41 : #include <casacore/ms/MSOper/MSMetaData.h>
42 : #include <casacore/tables/Tables/TableRecord.h>
43 : #include <casacore/casa/Utilities/GenSort.h>
44 :
45 : using namespace casacore;
46 : namespace casa { //# NAMESPACE CASA - BEGIN
47 :
48 : //
49 : // Constructor assigns pointer. If CT goes out of scope you
50 : // will get rubbish. Also sets string to separate subtable output.
51 : //
52 0 : CalSummary::CalSummary (CalTable& ct)
53 : : pCT(&ct),
54 : msname_p(""),
55 : haveMS_p(false),
56 : dashlin1(replicate("-",80)),
57 0 : dashlin2(replicate("=",80))
58 0 : { setMSname(); }
59 :
60 0 : CalSummary::CalSummary (CalTable* ct)
61 : : pCT(ct),
62 : msname_p(""),
63 : haveMS_p(false),
64 : dashlin1(replicate("-",80)),
65 0 : dashlin2(replicate("=",80))
66 0 : { setMSname(); }
67 :
68 0 : CalSummary::~CalSummary () { }
69 :
70 :
71 : //
72 : // Retrieve number of rows
73 : //
74 0 : Int CalSummary::nrow () const
75 : {
76 0 : return pCT->nRowMain();
77 : }
78 :
79 :
80 : //
81 : // Get cal table name
82 : //
83 0 : const String CalSummary::name () const
84 : {
85 0 : return pCT->tableName();
86 : }
87 :
88 :
89 : //
90 : // Reassign pointer.
91 : //
92 0 : Bool CalSummary::setCT (CalTable& ct)
93 : {
94 : CalTable* pTemp;
95 0 : pTemp = &ct;
96 0 : if (pTemp == 0) {
97 0 : return False;
98 : } else {
99 0 : pCT = pTemp;
100 0 : setMSname();
101 0 : return True;
102 : }
103 : }
104 :
105 0 : void CalSummary::setMSname() {
106 0 : ROCalDescColumns calDescCol(*pCT);
107 0 : Path filepath(name());
108 0 : String msname(calDescCol.msName()(0)), path(filepath.dirName());
109 0 : if (!path.empty())
110 0 : path += "/";
111 0 : String fullmsname = path + msname;
112 0 : if (!msname.empty() && Table::isReadable(fullmsname)) {
113 0 : msname_p = fullmsname;
114 0 : haveMS_p = true;
115 : } else {
116 0 : haveMS_p = false;
117 : }
118 0 : }
119 :
120 : //
121 : // List information about a ct to the logger
122 : //
123 0 : void CalSummary::list (LogIO& os, Bool verbose) const
124 : {
125 : // List a title for the Summary
126 0 : listTitle (os);
127 : // List the main table and subtables in a useful order and format
128 0 : listWhere(os, verbose);
129 0 : os << dashlin1 << endl;
130 0 : listWhat(os, verbose);
131 0 : os << dashlin1 << endl;
132 0 : listHow(os, verbose);
133 0 : os << dashlin1 << endl;
134 : // Post it
135 0 : os.post();
136 0 : }
137 :
138 : //
139 : // List a title for the Summary
140 : //
141 0 : void CalSummary::listTitle (LogIO& os) const
142 : {
143 0 : os << LogIO::NORMAL;
144 : // Type of cal table
145 0 : casacore::String type("unknown");
146 0 : os << dashlin2 << endl << "Cal Table Name: " << this->name();
147 0 : type = pCT->type();
148 0 : os << " Cal Type: " << type;
149 0 : os << endl << dashlin2 << endl;
150 0 : }
151 :
152 :
153 : //
154 : // Convenient table groupings
155 : //
156 0 : void CalSummary::listWhere (LogIO& os, Bool verbose) const
157 : {
158 0 : listObservation (os,verbose);
159 0 : }
160 :
161 0 : void CalSummary::listWhat (LogIO& os, Bool verbose) const
162 : {
163 0 : listMain(os, verbose);
164 0 : os << dashlin1 << endl;
165 0 : listField(os, verbose);
166 0 : }
167 :
168 0 : void CalSummary::listHow (LogIO& os, Bool verbose) const
169 : {
170 0 : listSpectralWindow(os, verbose);
171 0 : os << dashlin1 << endl;
172 0 : listAntenna(os, verbose);
173 0 : }
174 :
175 : //
176 : // SUBTABLES
177 : //
178 0 : void CalSummary::listMain (LogIO& os, Bool verbose) const
179 : {
180 0 : if (nrow() <= 0) {
181 0 : os << "The MAIN table is empty: there are no data!!!" << endl << LogIO::POST;
182 0 : return;
183 : }
184 : // to get columns from main and subtables
185 0 : ROCalMainColumns ctmain(*pCT);
186 0 : ROCalDescColumns ctdesc(*pCT);
187 :
188 : // Time:
189 0 : casacore::Vector<casacore::Double> times = ctmain.time().getColumn();
190 0 : casacore::Double startTime = times(0);
191 0 : casacore::Double stopTime = times(times.size()-1);
192 0 : casacore::Double exposTime = stopTime - startTime;
193 0 : casacore::String timeref = ctmain.time().keywordSet().subRecord("MEASINFO").asString("Ref");
194 : // Output info
195 : os << " Data records: " << nrow() << " Total elapsed time = "
196 : << exposTime << " seconds" << endl
197 : << " Observed from "
198 0 : << MVTime(startTime/C::day).string(MVTime::DMY,7)
199 : << " to "
200 0 : << MVTime(stopTime/C::day).string(MVTime::DMY,7)
201 : << " (" << timeref << ")"
202 0 : << endl << endl;
203 0 : os << LogIO::POST;
204 :
205 0 : if (verbose) {
206 : // Field names from MS:
207 0 : casacore::Vector<casacore::String> fieldnames;
208 0 : if (haveMS_p) {
209 0 : MSMetaInfoForCal msmeta(msname_p);
210 0 : msmeta.fieldNames(fieldnames);
211 : }
212 : // Spw Ids from CAL_DESC table
213 0 : casacore::Array<casacore::Int> spwids(ctdesc.spwId().getColumn());
214 :
215 : // Field widths for printing:
216 0 : casacore::Int width_lead(2),
217 0 : width_scan(4),
218 0 : width_time(22),
219 0 : width_fieldId(5),
220 0 : width_fieldname(20),
221 0 : width_nrow(10);
222 :
223 : // Sort order
224 0 : casacore::Vector<casacore::Int> icols(3);
225 0 : icols[0] = MSC::OBSERVATION_ID;
226 0 : icols[1] = MSC::SCAN_NUMBER;
227 0 : icols[2] = MSC::FIELD_ID;
228 0 : CalIter* iter = new CalIter(*pCT, icols);
229 0 : CalBuffer* buff = new CalBuffer(*iter);
230 : // values from main table
231 0 : casacore::Vector<casacore::Int> obs = buff->obsId();
232 0 : casacore::Vector<casacore::Double> times = buff->time();
233 0 : casacore::Vector<casacore::Int> scans = buff->scanNo();
234 0 : casacore::Vector<casacore::Int> fieldIds = buff->fieldId();
235 0 : casacore::Vector<casacore::Double> intervals = buff->interval();
236 0 : casacore::Vector<casacore::Int> calDescIds = buff->calDescId();
237 0 : casacore::Int lastobs(-2); // print obsid when changes
238 0 : casacore::Double lastday(0.0), lastTime(0.0); // print date when changes
239 0 : casacore::Int lastScan(-2), lastField(-2), lastCaldesc(-1), lastInt(-1);
240 0 : casacore::Int rowTotal(1);
241 0 : delete iter;
242 0 : delete buff;
243 0 : for (uInt i=0; i<obs.size(); ++i) {
244 0 : Int thisobs = obs(i);
245 : // print heading for each obs id
246 0 : if (thisobs != lastobs) {
247 0 : lastobs = thisobs;
248 0 : lastday = 0.0; // force printing date
249 0 : os << endl << " ObservationID = " << thisobs << endl;
250 0 : os << LogIO::POST;
251 : // print header for output columns
252 0 : casacore::String datetime = " Date Time ( ";
253 : // variable-len timeref string
254 0 : datetime.replace(20,timeref.length(),timeref);
255 0 : datetime.replace(20+timeref.length(),1,")");
256 0 : os << datetime;
257 0 : os << " Scan FldId FieldName nRows ";
258 0 : os << "SpwIds Interval" << endl;
259 0 : os << LogIO::POST;
260 : }
261 :
262 : // get values for this iteration
263 0 : Double time = times(i);
264 0 : casacore::Double day = floor(MVTime(time/C::day).day());
265 0 : casacore::Int scan = scans(i);
266 0 : casacore::Int fieldid = fieldIds(i);
267 0 : casacore::Int caldescid = calDescIds(i);
268 0 : casacore::Int interval = intervals(i);
269 :
270 0 : bool lastrow = (i==obs.size()-1);
271 : // print line when something changes and at end
272 0 : if ((time!=lastTime) || (scan!=lastScan) || (fieldid!=lastField) ||
273 0 : (caldescid!=lastCaldesc) || (interval!=lastInt) || lastrow) {
274 0 : if (rowTotal>1) { // but not first row
275 0 : os.output().precision(3);
276 0 : os.output().setf(ios::right, ios::adjustfield);
277 0 : os.output().width(width_lead); os << " ";
278 : // Timerange beginning time
279 0 : os.output().width(width_time);
280 0 : if (day!=lastday) { // print date
281 0 : os << MVTime(lastTime/C::day).string(MVTime::DMY,7);
282 : } else { // omit date
283 0 : os << MVTime(lastTime/C::day).string(MVTime::TIME,7);
284 : }
285 : // Scan
286 0 : os.output().width(width_lead); os << " ";
287 0 : os.output().setf(ios::right, ios::adjustfield);
288 0 : os.output().width(width_scan);
289 0 : os << lastScan;
290 : // FldId
291 0 : os.output().width(width_lead); os << " ";
292 0 : os.output().setf(ios::right, ios::adjustfield);
293 0 : os.output().width(width_fieldId);
294 0 : casacore::String fieldname("");
295 0 : os << lastField << " ";
296 0 : if (lastField >= 0 && !fieldnames.empty())
297 0 : fieldname = fieldnames(lastField);
298 : // FieldName
299 0 : os.output().setf(ios::left, ios::adjustfield);
300 0 : if (fieldname.length()>20)
301 0 : fieldname.replace(19,1,'*');
302 0 : os.output().width(width_fieldname);
303 0 : os << fieldname.at(0,20);
304 : // nRows
305 0 : os.output().width(width_nrow);
306 0 : os.output().setf(ios::left, ios::adjustfield);
307 0 : if (!lastrow) --rowTotal;
308 0 : os << rowTotal;
309 : // SpwIds
310 0 : os.output().width(width_lead); os << " ";
311 0 : Slicer spwSlicer = Slicer(Slice(), Slice(lastCaldesc));
312 0 : casacore::Vector<casacore::Int> spws = spwids(spwSlicer);
313 0 : os << spws;
314 : // Interval
315 0 : os.output().width(width_lead); os << " ";
316 0 : os << " " << interval;
317 0 : os << endl;
318 0 : os << LogIO::POST;
319 :
320 0 : lastday = day; // next lastday is this day
321 0 : rowTotal = 1; // restart counter
322 : }
323 0 : lastTime = time;
324 0 : lastScan = scan;
325 0 : lastField = fieldid;
326 0 : lastCaldesc = caldescid;
327 0 : lastInt = interval;
328 : }
329 0 : ++rowTotal;
330 : }
331 0 : os << LogIO::POST;
332 : }
333 0 : clearFormatFlags(os);
334 : }
335 :
336 0 : void CalSummary::listAntenna (LogIO& os, Bool verbose) const
337 : {
338 : // Get unique antenna ids present in the main table
339 0 : ROCalMainColumns ctmain(*pCT);
340 0 : std::vector<casacore::Int> ant1col = ctmain.antenna1().getColumn().tovector();
341 0 : std::set<casacore::Int> antIds = std::set<casacore::Int>(ant1col.begin(), ant1col.end());
342 0 : uInt nAnt = antIds.size();
343 0 : os << "Antennas: " << nAnt << endl;
344 :
345 : // Antenna info from MS
346 0 : casacore::Vector<casacore::String> antnames, stations;
347 0 : if (!haveMS_p) {
348 0 : os << "MS is not available for detailed antenna listing" << LogIO::POST;
349 : } else {
350 0 : MSMetaInfoForCal msmeta(msname_p);
351 0 : msmeta.antennaNames(antnames);
352 0 : stations = casacore::Vector<casacore::String>(msmeta.msmd().getAntennaStations());
353 :
354 0 : if (verbose) {
355 : // Detailed antenna list
356 0 : casacore::String indent(" ");
357 0 : casacore::uInt indwidth(5),
358 0 : namewidth(6),
359 0 : statwidth(10),
360 0 : diamwidth(5),
361 0 : latwidth(13),
362 0 : longwidth(14),
363 0 : offsetwidth(14),
364 0 : positionwidth(17);
365 0 : casacore::Int diamprec(1);
366 :
367 : // Antenna info from MS
368 0 : MSMetaInfoForCal msmeta(msname_p);
369 0 : QVector<Double> diameters = msmeta.msmd().getAntennaDiameters();
370 0 : vector<MPosition> positions = msmeta.msmd().getAntennaPositions();
371 0 : vector<QVector<Double> > offsets = msmeta.msmd().getAntennaOffsets();
372 0 : Bool posIsITRF = (positions[0].getRef().getType() == MPosition::ITRF);
373 :
374 : // Get observatory position info for antenna offsets
375 : // *not read from antenna table OFFSET column!*
376 0 : casacore::MPosition obsPos;
377 0 : casacore::String telname;
378 0 : bool doOffset = getObservatoryPosition(obsPos, telname);
379 0 : casacore::Double rObs = 0.0, longObs = 0.0, latObs = 0.0;
380 0 : if (!doOffset) {
381 0 : os << "Warning: Telescope name '" << telname << "' is not recognized by CASA. Cannot compute offsets." << endl << endl;
382 : } else {
383 0 : casacore::Vector<casacore::Double> obsXYZ = obsPos.get("m").getValue();
384 0 : casacore::Double xo(obsXYZ[0]), yo(obsXYZ[1]), zo(obsXYZ[2]);
385 0 : rObs = sqrt(xo*xo + yo*yo + zo*zo);
386 0 : casacore::Vector<casacore::Double> obsLongLat = obsPos.getAngle("rad").getValue();
387 0 : longObs = obsLongLat[0];
388 0 : latObs = obsLongLat[1];
389 : }
390 :
391 0 : os.output().setf(ios::fixed, ios::floatfield);
392 0 : os.output().setf(ios::left, ios::adjustfield);
393 : // Write the column headings:
394 0 : os << indent;
395 0 : os.output().width(indwidth); os << "ID";
396 0 : os.output().width(namewidth); os << "Name";
397 0 : os.output().width(statwidth); os << "Station";
398 0 : os.output().width(diamwidth+4); os << "Diam.";
399 0 : os.output().width(longwidth); os << "Long.";
400 0 : os.output().width(latwidth); os << "Lat.";
401 0 : if (doOffset) {
402 0 : os.output().width(3 * offsetwidth);
403 0 : os << " Offset from array center (m)";
404 : }
405 0 : os.output().width(3 * positionwidth);
406 0 : os << " ITRF Geocentric coordinates (m)";
407 0 : os << endl << indent;
408 0 : os.output().width(indwidth + namewidth + statwidth + diamwidth + 4 + longwidth + latwidth);
409 0 : os << " ";
410 0 : os.output().setf(ios::right, ios::adjustfield);
411 0 : if (doOffset) {
412 0 : os.output().width(offsetwidth); os << "East";
413 0 : os.output().width(offsetwidth); os << "North";
414 0 : os.output().width(offsetwidth); os << "Elevation";
415 : }
416 0 : os.output().width(positionwidth); os << "x";
417 0 : os.output().width(positionwidth); os << "y";
418 0 : os.output().width(positionwidth); os << "z";
419 0 : os << endl;
420 : // Iterator through antennas and log info for each
421 0 : const static Unit diamUnit("m");
422 0 : std::set<Int>::const_iterator iter = antIds.begin();
423 0 : std::set<Int>::const_iterator end = antIds.end();
424 0 : for (; iter!=end; ++iter) {
425 0 : Int antId = *iter;
426 : // diameter
427 0 : const Quantity& diam = diameters[antId];
428 : // xyz
429 0 : casacore::Vector<casacore::Double> xyz = positions[antId].get("m").getValue();
430 0 : casacore::Double x = xyz(0);
431 0 : casacore::Double y = xyz(1);
432 0 : casacore::Double z = xyz(2);
433 0 : casacore::Double rAnt = sqrt(x*x + y*y + z*z);
434 : // offsets
435 0 : const MPosition& mLongLat = positions[antId];
436 0 : casacore::Vector<casacore::Double> antOffset;
437 0 : if (doOffset) {
438 0 : casacore::Vector<casacore::Double> longLatRad = mLongLat.getAngle("rad").getValue();
439 0 : casacore::Double longAnt = longLatRad(0);
440 0 : casacore::Double latAnt = longLatRad(1);
441 0 : casacore::Vector<casacore::Double> offset(3);
442 0 : offset[0] = (longAnt - longObs) * rObs * cos(latObs);
443 0 : offset[1] = (latAnt - latObs) * rObs;
444 0 : offset[2] = rAnt - rObs;
445 0 : casacore::QVD qoffset(offset, "m");
446 0 : antOffset = qoffset.getValue("m");
447 : }
448 : // lat/long
449 0 : MVAngle mvLong = mLongLat.getAngle().getValue()(0);
450 0 : MVAngle mvLat = mLongLat.getAngle().getValue()(1);
451 0 : if (!posIsITRF) {
452 0 : MeasConvert<MPosition> toItrf(positions[antId], MPosition::ITRF);
453 0 : positions[antId] = toItrf(positions[antId]);
454 : }
455 : // write the row
456 0 : os.output().setf(ios::left, ios::adjustfield);
457 0 : os << indent;
458 : // ID, Name, Station
459 0 : os.output().width(indwidth); os << antId;
460 0 : os.output().width(namewidth); os << antnames(antId);
461 0 : os.output().width(statwidth); os << stations(antId);
462 : // Diam.
463 0 : os.output().precision(diamprec);
464 0 : os.output().width(diamwidth);
465 0 : os << diam.getValue(diamUnit) << "m ";
466 : // Long. and Lat.
467 0 : os.output().width(longwidth);
468 0 : os << mvLong.string(MVAngle::ANGLE,7);
469 0 : os.output().width(latwidth);
470 0 : os << mvLat.string(MVAngle::DIG2,7);
471 0 : if (doOffset) {
472 : // Offset from array center (m) (East, North, Elevation)
473 0 : os.output().setf(ios::right, ios::adjustfield);
474 0 : os.output().precision(4);
475 0 : os.output().width(offsetwidth); os << antOffset(0);
476 0 : os.output().width(offsetwidth); os << antOffset(1);
477 0 : os.output().width(offsetwidth); os << antOffset(2);
478 : }
479 : // ITRF Geocentric coordinates (m) (x, y, z)
480 0 : os.output().precision(6);
481 0 : os.output().width(positionwidth); os << x;
482 0 : os.output().width(positionwidth); os << y;
483 0 : os.output().width(positionwidth); os << z;
484 0 : os << endl;
485 : }
486 : } else {
487 : // Horizontal list of the stations names:
488 0 : os << " ('name'@'station') " << endl;
489 0 : casacore::String line, leader;
490 : // track last id of previous line for leader
491 0 : casacore::Int lastIdInLine = *antIds.begin() - 1;
492 0 : std::set<casacore::Int>::const_iterator iter = antIds.begin();
493 0 : std::set<casacore::Int>::const_iterator end = antIds.end();
494 0 : casacore::Int maxAnt = *std::max_element(antIds.begin(), antIds.end());
495 0 : for (; iter!=end; ++iter) {
496 0 : Int antId = *iter;
497 : // Build the line
498 0 : if (!antnames.empty())
499 0 : line = line + "'" + antnames(antId) + "'" + "@";
500 0 : if (!stations.empty())
501 0 : line = line + "'" + stations(antId) + "'";
502 : // Add comma if not at the end
503 0 : if (antId != maxAnt) line = line + ", ";
504 0 : if (line.length() > 55 || antId == maxAnt) {
505 : // This line is finished, dump it after the line leader
506 0 : leader = String::toString(lastIdInLine+1) + "-" + String::toString(antId) +": ";
507 0 : os << " ID=";
508 0 : os.output().setf(ios::right, ios::adjustfield);
509 0 : os.output().width(8); os << leader;
510 0 : os << line << endl;
511 0 : line = "";
512 0 : lastIdInLine = antId;
513 : }
514 : }
515 : }
516 : }
517 0 : os << LogIO::POST;
518 0 : clearFormatFlags(os);
519 0 : }
520 :
521 0 : bool CalSummary::getObservatoryPosition(casacore::MPosition& obspos,
522 : casacore::String& name) const {
523 : // Returns observatory position in ITRF
524 0 : bool obsposOk(false);
525 0 : MSMetaInfoForCal msmeta(msname_p);
526 : // get telescope name
527 0 : name = msmeta.msmd().getObservatoryNames()[0];
528 0 : obsposOk = MeasTable::Observatory(obspos, name);
529 0 : if (obsposOk && (obspos.getRef().getType() != MPosition::ITRF)) {
530 0 : MeasConvert<MPosition> toItrf(obspos, MPosition::ITRF);
531 0 : obspos = toItrf(obspos);
532 : }
533 0 : return obsposOk;
534 : }
535 :
536 0 : void CalSummary::listField (LogIO& os, Bool /*verbose*/) const
537 : {
538 : // Get unique field ids present in the main table
539 0 : ROCalMainColumns ctmain(*pCT);
540 0 : std::vector<casacore::Int> fieldcol = ctmain.fieldId().getColumn().tovector();
541 0 : std::set<casacore::Int> fieldIds = std::set<casacore::Int>(fieldcol.begin(), fieldcol.end());
542 0 : uInt nfield = fieldIds.size();
543 :
544 0 : os << "Fields: " << nfield << endl;
545 0 : if (!haveMS_p)
546 0 : os << "MS is not available for detailed field listing" << LogIO::POST;
547 : else {
548 0 : casacore::Int widthLead(2),
549 0 : widthField(6),
550 0 : widthCode(5),
551 0 : widthName(20),
552 0 : widthRA(16),
553 0 : widthDec(16),
554 0 : widthType(8),
555 0 : widthSrc(6);
556 :
557 : // Print heading
558 0 : os.output().setf(ios::left, ios::adjustfield);
559 0 : os.output().width(widthLead); os << " ";
560 0 : os.output().width(widthField); os << "FldId";
561 0 : if (haveMS_p) {
562 0 : os.output().width(widthCode); os << "Code";
563 0 : os.output().width(widthName); os << "Name";
564 0 : os.output().width(widthRA); os << "RA";
565 0 : os.output().width(widthDec); os << " Decl";
566 0 : os.output().width(widthType); os << "Epoch";
567 0 : os.output().width(widthSrc); os << "SrcId";
568 : }
569 0 : os << endl;
570 :
571 : // get values from MS
572 0 : casacore::Vector<casacore::String> fieldNames, codes;
573 0 : vector<MDirection> phaseDirs;
574 0 : vector<Int> sourceIds;
575 0 : if (haveMS_p) {
576 0 : MSMetaInfoForCal msmeta(msname_p);
577 0 : msmeta.fieldNames(fieldNames);
578 0 : codes = casacore::Vector<casacore::String>(msmeta.msmd().getFieldCodes());
579 0 : phaseDirs = msmeta.msmd().getPhaseDirs();
580 0 : sourceIds = msmeta.msmd().getFieldTableSourceIDs();
581 : }
582 :
583 : // loop through fields
584 0 : std::set<Int>::const_iterator fiter = fieldIds.begin();
585 0 : std::set<Int>::const_iterator fend = fieldIds.end();
586 0 : static const MEpoch ezero(Quantity(0, "s"));
587 0 : for (; fiter!=fend; ++fiter) {
588 0 : Int fieldId = *fiter;
589 0 : if (fieldId >=0) {
590 0 : os.output().setf(ios::left, ios::adjustfield);
591 0 : os.output().width(widthLead); os << " ";
592 0 : os.output().width(widthField); os << fieldId;
593 0 : if (haveMS_p) {
594 0 : casacore::MDirection mRaDec = phaseDirs[fieldId];
595 0 : MVAngle mvRa = mRaDec.getAngle().getValue()(0);
596 0 : MVAngle mvDec = mRaDec.getAngle().getValue()(1);
597 0 : String name = fieldNames(fieldId);
598 0 : if (name.length()>20)
599 0 : name.replace(19,1,"*");
600 0 : os.output().width(widthCode); os << codes(fieldId);
601 0 : os.output().width(widthName); os << name.at(0,20);
602 0 : os.output().width(widthRA); os << mvRa(0.0).string(MVAngle::TIME,12);
603 0 : os.output().width(widthDec); os << mvDec.string(MVAngle::DIG2,11);
604 0 : os.output().width(widthType);
605 0 : os << MDirection::showType(mRaDec.getRefPtr()->getType());
606 0 : os.output().width(widthSrc); os << sourceIds[fieldId];
607 : }
608 0 : os << endl;
609 : } else {
610 0 : os.output().setf(ios::left, ios::adjustfield);
611 0 : os.output().width(widthLead); os << " ";
612 0 : os.output().width(widthField); os << fieldId;
613 0 : os << endl;
614 : }
615 : }
616 0 : os << LogIO::POST;
617 : }
618 0 : clearFormatFlags(os);
619 0 : }
620 :
621 0 : void CalSummary::listObservation (LogIO& os, Bool /*verbose*/) const
622 : {
623 : // Get unique obs ids present in the main table
624 0 : ROCalMainColumns ctmain(*pCT);
625 0 : std::vector<casacore::Int> obscol = ctmain.obsId().getColumn().tovector();
626 0 : std::set<casacore::Int> obsIds = std::set<casacore::Int>(obscol.begin(), obscol.end());
627 0 : os << "Observation IDs in main table: " << obsIds << endl;
628 :
629 0 : Int nrow = pCT->nRowObservation();
630 0 : if (nrow <= 0) {
631 0 : os << "The OBSERVATION table is empty";
632 : } else {
633 0 : os << "Observation table entries: " << nrow << LogIO::POST;
634 0 : for (Int irow=0 ; irow < nrow; irow++) {
635 0 : Record obsRow = pCT->getRowObservation(irow);
636 0 : String observer = obsRow.asString("OBSERVER");
637 0 : if (!observer.empty())
638 0 : os << " Observer: " << observer;
639 0 : String project = obsRow.asString("PROJECT");
640 0 : if (!project.empty())
641 0 : os << " Project: " << project;
642 0 : String telescope = obsRow.asString("TELESCOPE_NAME");
643 0 : if (!telescope.empty())
644 0 : os << " Telescope: " << telescope;
645 : // TIME_RANGE values don't match main table or ms!
646 : }
647 : }
648 0 : os << LogIO::POST;
649 0 : clearFormatFlags(os);
650 0 : }
651 :
652 0 : void CalSummary::listHistory (LogIO& os) const
653 : {
654 0 : Int nrow = pCT->nRowHistory();
655 0 : if (nrow <= 0) {
656 0 : os << "The HISTORY table is empty" << endl;
657 : } else {
658 0 : os << "History table entries: " << nrow << endl << LogIO::POST;
659 0 : for (Int irow=0 ; irow < nrow; irow++) {
660 0 : Record histRow = pCT->getRowHistory(irow);
661 0 : os << "ROW " << irow << " ";
662 0 : os << "PARMS: " << histRow.asString("CAL_PARMS") << endl;
663 0 : os << "TABLES: " << histRow.asString("CAL_TABLES") << endl;
664 0 : os << "SELECT: " << histRow.asString("CAL_SELECT") << endl;
665 0 : os << "NOTES: " << histRow.asString("CAL_NOTES") << endl;
666 : }
667 : }
668 0 : os << LogIO::POST;
669 0 : }
670 :
671 0 : void CalSummary::listSpectralWindow (LogIO& os, Bool /*verbose*/) const
672 : {
673 0 : ROCalDescColumns ctdesc(*pCT);
674 0 : casacore::Array<casacore::Int> spwIdArray = ctdesc.spwId().getColumn();
675 0 : IPosition vshape(1, spwIdArray.nelements());
676 0 : casacore::Vector<casacore::Int> spwIds(spwIdArray.reform(vshape));
677 0 : Int nSpws = GenSort<Int>::sort(spwIds, Sort::Ascending, Sort::NoDuplicates);
678 0 : spwIds.resize(nSpws, true);
679 0 : os << "Spectral Windows: " << nSpws;
680 0 : Bool noSpws = (nSpws==1 && spwIds(0)==-1);
681 0 : if (noSpws)
682 0 : os << " " << spwIds;
683 0 : os << endl;
684 :
685 0 : if (!noSpws) {
686 0 : if (haveMS_p) {
687 0 : MSMetaInfoForCal msmeta(msname_p);
688 0 : vector<MFrequency> refFreqs = msmeta.msmd().getRefFreqs();
689 0 : vector<uInt> nChans = msmeta.msmd().nChans();
690 0 : vector<Double> totalBWs = msmeta.msmd().getBandWidths();
691 0 : vector<QVD> resolution = msmeta.msmd().getChanResolutions(False);
692 :
693 : // Header: SpwId Ref.Freq #Chans Resolution TotalBW
694 0 : casacore::Int widthLead(2), widthId(7), widthFreq(13), widthFrqNum(7), widthNumChan(8);
695 0 : os.output().setf(ios::left, ios::adjustfield);
696 0 : os.output().width(widthLead); os << " ";
697 0 : os.output().width(widthId); os << "SpwId";
698 0 : os.output().width(widthFreq); os << "Ref.Freq";
699 0 : os.output().width(widthNumChan); os << "#Chans";
700 0 : os.output().width(widthFreq); os << "Resolution";
701 0 : os.output().width(widthFreq); os << "TotalBW";
702 0 : os << endl;
703 :
704 0 : for (Int ispw=0; ispw<nSpws; ispw++) {
705 0 : Int spwId = spwIds(ispw);
706 0 : os.output().setf(ios::left, ios::adjustfield);
707 0 : os.output().width(widthLead); os << " ";
708 0 : os.output().width(widthId); os << spwId;
709 0 : if (spwId >= 0) {
710 0 : os.output().width(widthFrqNum);
711 0 : os << refFreqs[spwId].get("MHz"); // MFreqency
712 0 : os.output().width(widthLead); os << " ";
713 0 : os.output().width(widthNumChan);
714 0 : os << nChans[spwId];
715 0 : os.output().width(widthFrqNum);
716 0 : os << resolution[spwId][0].get("kHz"); // QVD
717 0 : os.output().width(widthLead); os << " ";
718 0 : os.output().width(widthLead); os << " ";
719 0 : os.output().width(widthFrqNum);
720 0 : os << totalBWs[spwId] / 1000.0; // convert Hz to kHz
721 0 : os << " kHz";
722 0 : os << endl;
723 : }
724 : }
725 : } else {
726 0 : os << "MS is not available for detailed spw listing" << endl;
727 : }
728 : }
729 0 : os << LogIO::POST;
730 0 : clearFormatFlags(os);
731 0 : }
732 :
733 0 : void CalSummary::listTables (LogIO& os, Bool verbose) const
734 : {
735 : // Get nrows for each table (=-1 if table absent)
736 0 : uInt nTables = 4;
737 0 : casacore::Vector<casacore::Int> tableRows(nTables);
738 0 : tableRows(0) = pCT->nRowMain();
739 0 : tableRows(1) = pCT->nRowDesc();
740 0 : tableRows(2) = pCT->nRowHistory();
741 0 : tableRows(3) = pCT->nRowObservation();
742 :
743 0 : casacore::Vector<casacore::String> tableStrings(nTables);
744 0 : tableStrings(0) = "MAIN";
745 0 : tableStrings(1) = "CAL_DESC";
746 0 : tableStrings(2) = "CAL_HISTORY";
747 0 : tableStrings(3) = "OBSERVATION";
748 :
749 : // Just to make things read better
750 0 : casacore::Vector<casacore::String> rowStrings(6);
751 0 : rowStrings = " rows";
752 0 : for (uInt i=0; i<nTables; i++) {
753 0 : if (tableRows(i)==1) rowStrings(i) = " row";
754 : // if table exists, but empty:
755 0 : if (tableRows(i)==0) rowStrings(i) = " <empty>";
756 : // if table absent:
757 0 : if (tableRows(i)==-1) rowStrings(i) = "<absent>";
758 : }
759 :
760 0 : if (verbose) {
761 0 : os << " Tables" << endl;
762 0 : for (uInt i=0; i<nTables; i++) {
763 0 : os.output().setf(ios::left, ios::adjustfield);
764 0 : os.output().width(3); os << " ";
765 0 : os.output().width(20); os << tableStrings(i);
766 0 : os.output().setf(ios::right, ios::adjustfield);
767 0 : os.output().width(8); os << tableRows(i);
768 0 : os.output().setf(ios::left, ios::adjustfield);
769 0 : os.output().width(10); os << rowStrings(i);
770 0 : os << endl;
771 : }
772 : } else {
773 0 : os << " Tables (rows): (-1 = table absent)" << endl;
774 0 : for (uInt i=0; i<nTables; i++) {
775 0 : casacore::String tableinfo = tableStrings(i) + " (" +
776 0 : casacore::String::toString(tableRows(i)) + ")";
777 0 : os.output().setf(ios::left, ios::adjustfield);
778 0 : casacore::Int infowidth = 15;
779 0 : if (i==1 || i==4) infowidth = 25;
780 0 : os.output().width(infowidth);
781 0 : os << tableinfo;
782 0 : if ((i+1) % 3==0) os << endl;
783 : }
784 : }
785 0 : os << LogIO::POST;
786 0 : clearFormatFlags(os);
787 0 : }
788 :
789 : //
790 : // Clear all the formatting flags
791 : //
792 0 : void CalSummary::clearFormatFlags(LogIO& os) const
793 : {
794 0 : os.output().unsetf(ios::left);
795 0 : os.output().unsetf(ios::right);
796 0 : os.output().unsetf(ios::internal);
797 :
798 0 : os.output().unsetf(ios::dec);
799 0 : os.output().unsetf(ios::oct);
800 0 : os.output().unsetf(ios::hex);
801 :
802 0 : os.output().unsetf(ios::showbase | ios::showpos | ios::uppercase | ios::showpoint);
803 :
804 0 : os.output().unsetf(ios::scientific);
805 0 : os.output().unsetf(ios::fixed);
806 0 : }
807 :
808 : } //# NAMESPACE CASA - END
809 :
|