Isis 3 Programmer Reference
FitsToJson.cpp
1
6/* SPDX-License-Identifier: CC0-1.0 */
7#include "FitsToJson.h"
8
9#include <nlohmann/json.hpp>
10#include <QString>
11
12#include "IException.h"
13#include "IString.h"
14#include "PvlObject.h"
15#include "PvlGroup.h"
16#include "PvlToJSON.h"
17
18using json = nlohmann::json;
19
20namespace Isis {
21
30 json fitsToJson(std::ifstream &fileStream) {
31 PvlObject fitsPvl;
32 QList<PvlGroup *> *fitsImageLabels = new QList< PvlGroup * >;
33 QList<PvlGroup *> *extraFitsLabels = new QList< PvlGroup * >;
34 QList<int> *headerSizes = new QList < int >;
35 QList<int> *dataStarts = new QList < int >;
36
37 // Process each FITS label area, storing each in its own PvlGroup.
38 char readBuf[81];
39 IString line = "";
40 unsigned int place;
41
42 // Start at the beginning of the file for the main FITS label
43 fileStream.seekg(0, std::ios_base::beg);
44
45 // Reads each label line (80 chars)
46 while (fileStream.read(readBuf, 80) && fileStream.gcount() == 80) {
47
48 PvlGroup *fitsLabel = new PvlGroup("FitsLabels");
49
50 readBuf[80] = '\0';
51 line = readBuf;
52 place = 80;
53
54 // Process each fits label record (80 bytes) and place keyword, value pairs into PvlKeywords
55 // with any associated comments.
56 while (line.substr(0, 3) != "END") {
57
58 // Check for blank lines
59 if (line.substr(0, 1) != " " && line.substr(0, 1) != "/") {
60 // Name of keyword
61 PvlKeyword label(line.Token(" =").ToQt()); // Stop on spaces OR equal sign
62 if (QString::compare(label.name(), "OBJECT", Qt::CaseInsensitive) == 0) {
63 label.setName("TARGET");
64 label.addComment("NOTE: This keyword name was changed from 'OBJECT' in the original "
65 "fit header file.");
66 }
67 // Remove up to beginning of data
68 line.TrimHead(" =");
69 line.TrimTail(" ");
70 if (label.name() == "COMMENT" || label.name() == "HISTORY") {
71 label += line.ToQt();
72 }
73 else {
74 // Check for a quoted value
75 if (line.substr(0,1) == "'") {
76 line.TrimHead("'");
77 label += line.Token("'").TrimHead(" ").TrimTail(" ").ToQt();
78 line.TrimHead(" '");
79 }
80 else {
81 // Access any remaining data without the trailing comment if there is one
82 IString value = line.Token("/");
83 value.TrimTail(" ");
84 label += value.ToQt();
85 line.TrimHead(" ");
86 }
87 // If the line still has anything in it, treat it is as a comment.
88 if (line.size() > 2) {
89 line.TrimHead(" /");
90 label.addComment(line.ToQt());
91 if (line != line.Token("[")) {
92 label.setUnits(line.Token("[").Token("]").ToQt());
93 }
94 }
95 }
96 fitsLabel->addKeyword(label);
97 }
98
99 fileStream.read(readBuf, 80);
100 readBuf[80] = '\0';
101 line = readBuf;
102 place += 80;
103 }
104
105 // Save off the PvlGroup and the number of records read from this label
106 fitsPvl.addGroup(*fitsLabel);
107 headerSizes->append((int)ceil(place / 2880.0));
108
109 // Move the file pointer past the padding after the "END" (i.e., points to start of data)
110 std::streamoff jump = 0;
111 jump = headerSizes->last() * 2880 - place;
112 fileStream.seekg(jump, std::ios_base::cur);
113
114 dataStarts->append(fileStream.tellg());
115
116 if (fitsLabel->hasKeyword("BITPIX") && fitsLabel->hasKeyword("NAXIS")) {
117
118 if((int)fitsLabel->findKeyword("NAXIS") > 0) {
119 int bytesPerPixel = 0;
120 bytesPerPixel = (int)((*fitsLabel)["BITPIX"]);
121 bytesPerPixel = std::abs(bytesPerPixel);
122 bytesPerPixel /= 8;
123
124 unsigned int axis1 = 1;
125 axis1 = toInt((*fitsLabel)["NAXIS1"]);
126
127 unsigned int axis2 = 1;
128 if (fitsLabel->hasKeyword("NAXIS2")) {
129 axis2 = toInt((*fitsLabel)["NAXIS2"]);
130 }
131
132 unsigned int axis3 = 1;
133 if (fitsLabel->hasKeyword("NAXIS3")) {
134 axis3 = toInt((*fitsLabel)["NAXIS3"]);
135 }
136
137 jump = (int)(ceil(bytesPerPixel * axis1 * axis2 * axis3 / 2880.0) * 2880.0);
138 fileStream.seekg(jump, std::ios_base::cur);
139 }
140 else {
141 // Note: this will allow us to read extra label sections that have 0 axes,
142 // but has image-related info (so BITPIX and NAXIS keywords exist). This
143 // includes informational labels, as seen at the beginning of hayabusa2
144 // images in this case, there is NO DATA, so no jump should be needed to
145 // get to the next section.
146 PvlGroup *extraLabelGroup = fitsImageLabels->last();
147 extraLabelGroup->setName("FitsExtras");
148 extraFitsLabels->append(extraLabelGroup);
149
150 fitsImageLabels->removeLast();
151 headerSizes->removeLast();
152 dataStarts->removeLast();
153
154 fitsPvl.addGroup(*extraLabelGroup);
155 }
156 }
157 else if (fitsImageLabels->size() > 1) {
158 fitsImageLabels->removeLast();
159 headerSizes->removeLast();
160 dataStarts->removeLast();
161 break;
162 }
163
164 else {
165 QString msg = QObject::tr("The FITS file does not contain a section header that appears "
166 "to describe an image.");
167 throw IException(IException::User, msg, _FILEINFO_);
168 }
169 }
170 return pvlObjectToJSON(fitsPvl);
171 }
172
173
181 json fitsToJson(FileName fitsFile) {
182 std::ifstream fileStream;
183 try {
184 fileStream.open(fitsFile.expanded().toLocal8Bit().constData(), std::ios::in | std::ios::binary);
185 }
186 catch (IException &e) {
187 QString msg = QString("Unable to open FITS formatted file [%1].")
188 .arg(fitsFile.toString());
189 throw IException(e, IException::User, msg, _FILEINFO_);
190 }
191
192 return fitsToJson(fileStream);
193 }
194
195}
File name manipulation and expansion.
Definition FileName.h:100
Isis exception class.
Definition IException.h:91
@ User
A type of error that could only have occurred due to a mistake on the user's part (e....
Definition IException.h:126
Adds specific functionality to C++ strings.
Definition IString.h:165
IString TrimTail(const std::string &chars)
Trims the input characters from the end of the object IString.
Definition IString.cpp:587
IString Token(const IString &separator)
Returns the first token in the IString.
Definition IString.cpp:897
QString ToQt() const
Retuns the object string as a QString.
Definition IString.cpp:869
IString TrimHead(const std::string &chars)
Trims The input characters from the beginning of the object IString.
Definition IString.cpp:558
void setName(const QString &name)
Set the name of the container.
Contains multiple PvlContainers.
Definition PvlGroup.h:41
A single keyword-value pair.
Definition PvlKeyword.h:87
void setName(QString name)
Sets the keyword name.
QString name() const
Returns the keyword name.
Definition PvlKeyword.h:103
void setUnits(QString units)
Sets the unit of measure for all current values if any exist.
void addComment(QString comment)
Add a comment to the PvlKeyword.
Contains Pvl Groups and Pvl Objects.
Definition PvlObject.h:61
This is free and unencumbered software released into the public domain.
Definition Apollo.h:16
int toInt(const QString &string)
Global function to convert from a string to an integer.
Definition IString.cpp:93
json fitsToJson(std::ifstream &fileStream)
Convert the contents of a stream to a JSON object.
json pvlObjectToJSON(PvlObject &object)
Convert the contents of a PvlObject to a JSON object.