mxlib
c++ tools for analyzing astronomical data and other tasks by Jared R. Males. [git repo]
Loading...
Searching...
No Matches
fitsHeader.hpp
Go to the documentation of this file.
1/** \file fitsHeader.hpp
2 * \brief Declares and defines a class to work with a FITS header
3 * \ingroup fits_processing_files
4 * \author Jared R. Males (jaredmales@gmail.com)
5 *
6 */
7
8//***********************************************************************//
9// Copyright 2015-2022 Jared R. Males (jaredmales@gmail.com)
10//
11// This file is part of mxlib.
12//
13// mxlib is free software: you can redistribute it and/or modify
14// it under the terms of the GNU General Public License as published by
15// the Free Software Foundation, either version 3 of the License, or
16// (at your option) any later version.
17//
18// mxlib is distributed in the hope that it will be useful,
19// but WITHOUT ANY WARRANTY; without even the implied warranty of
20// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
21// GNU General Public License for more details.
22//
23// You should have received a copy of the GNU General Public License
24// along with mxlib. If not, see <http://www.gnu.org/licenses/>.
25//***********************************************************************//
26
27#ifndef ioutils_fits__fitsHeader_hpp
28#define ioutils_fits__fitsHeader_hpp
29
30#include <list>
31#include <unordered_map>
32#include <iostream>
33#include <vector>
34#include <optional>
35
36#include "fitsHeaderCard.hpp"
37
38namespace mx
39{
40namespace fits
41{
42
43/// Class to manage a complete fits header
44/** Manages tasks such as insertion (avoiding duplicates), and keyword lookup.
45 * The \ref fitsHeaderCard "cards" are stored in a std::list to preserve order, but
46 * a std::unordered_multimap is used to provide fast keyword lookup.
47 *
48 * \ingroup fits_processing
49 */
51{
52
53 public:
54 /// The iterator type for the cards list
55 typedef std::list<fitsHeaderCard>::iterator headerIterator;
56
57 /// The iterator type for the card map
58 typedef std::unordered_multimap<std::string, headerIterator>::iterator mapIterator;
59
60 protected:
61 /// The storage for the FITS header cards
62 /** We use a list, rather than forward_list, so that append (insert at end) is constant time.
63 *
64 */
65 std::list<fitsHeaderCard> cards;
66
67 /// This multimap allows for fast lookup by keyword.
68 /** Use unordered_multimap to handle HISTORY and COMMENT properly, and be as efficient as possible.
69 */
70 std::unordered_multimap<std::string, headerIterator> cardMap;
71
72 public:
73 /// Default c'tor
74 fitsHeader();
75
76 /// Copy constructor
77 /** Must be explicitly defined to handle creation of new iterators in the cardMap
78 */
79 fitsHeader( const fitsHeader &head /**< The fitsHeader to copy */ );
80
81 /// Destructor
83
84 /// Assignment
85 /** Must be explicitly defined to handle creation of new iterators in the cardMap
86 */
87 fitsHeader &operator=( const fitsHeader &head /**< The fitsHeader to copy */ );
88
89 /// Get iterator to the beginning of the cards list
91
92 /// Get iterator to the end of the cards list
94
95 /// Get iterator pointing to a specific element
96 headerIterator iterator( const std::string &keyword /**< The keyword to look up*/ );
97
98 /// Test whether the header is empty.
99 bool empty();
100
101 /// Get number of cards currently stored in the header.
102 size_t size();
103
104 /// Clear all cards from the header
105 void clear();
106
107 /// Get number of cards with a given keyword
108 /** Reeturns the result of the count() method of the header map.
109 *
110 * \retval the number of cards with keyword.
111 */
112 size_t count( const std::string &keyword /**< [in] the keyword to loop up*/ );
113
114 /// Erase card by keyword
115 /** This can not be used to erase COMMENT or HISTORY cards.
116 *
117 * \param keyword the keyword of the card to delete
118 *
119 */
120 void erase( const std::string &keyword );
121
122 /// Erase card by iterator
123 /** This handles COMMENT and HISTORY cards, deleting only the one pointed to by it
124 *
125 * \param it is a headerIterator pointing to the card to delete.
126 *
127 */
128 void erase( headerIterator it );
129
130 /// Erase the standard entries at the top of the header
131 /** Erases each entry down to BSCALE. This is useful for appending
132 * a header to a newly created file.
133 */
134 void eraseStandardTop();
135
136 /// Append a fitsHeaderCard to the end of the header
137 /**
138 * \param card is a fitsHeaderCard already populated
139 */
140 void append( const fitsHeaderCard & card );
141
142 /// Append a card to the end of the header, from the three components of a card.
143 /**
144 * \tparam typeT is the data type of the value
145 *
146 * \param k is the keyword string
147 * \param v is the value of typeT
148 * \param c is the comment string
149 */
150 template <typename typeT>
151 void append( const std::string &k, const typeT &v, const std::string &c );
152
153 /// Append a card to the end of the header, from the components of a card with no comment.
154 /**
155 * \tparam typeT is the data type of the value
156 *
157 * \param k is the keyword string
158 * \param v is the value of typeT
159 */
160 template <typename typeT>
161 void append( const std::string &k, const typeT &v );
162
163 /// Append a card to the end of the header, with just a keyword.
164 /** Appens a headerCard with unknownType
165 *
166 * \param k is the keyword string
167 */
168 void append( const std::string &k );
169
170 /// Append a fitsHeader to the end of the header
171 /**
172 * \param head is a populated fitsHeader
173 */
174 void append( fitsHeader &head );
175
176 /// Insert a card before another card.
177 /**
178 * \param it points to the element before which to insert
179 * \param card contains the card to insert
180 */
182
183 /// Insert a card before another card, specifying the card by its components.
184 /**
185 * \tparam typeT is the type of the value, which is converted to string for insertion
186 *
187 * \param it points to the element before which to insert
188 * \param k is the keyword
189 * \param v is the value
190 * \param c is the comment
191 *
192 */
193 template <typename typeT>
194 void insert_before( headerIterator it, const std::string &k, typeT v, const std::string &c );
195
196 /// Insert a card before another card, specifying the card by its components.
197 /**
198 * \tparam typeT is the type of the value, which is converted to string for insertion
199 *
200 * \param it points to the element before which to insert
201 * \param k is the keyword
202 * \param v is the value
203 *
204 */
205 template <typename typeT>
206 void insert_before( headerIterator it, const std::string &k, typeT v );
207
208 /// Insert a card after another card.
209 /**
210 * \param it points to the element after which to insert
211 * \param card contains the card to insert
212 */
214
215 /// Insert a card after another card, specifying the card by its components.
216 /**
217 * \tparam typeT is the type of the value, which is converted to string for insertion
218 *
219 * \param it points to the element after which to insert
220 * \param k is the keyword
221 * \param v is the value
222 * \param c is the comment
223 *
224
225 */
226 template <typename typeT>
227 void insert_after( headerIterator it, const std::string &k, typeT v, const std::string &c );
228
229 /// Insert a card after another card, specifying the card by its components.
230 /**
231 * \tparam typeT is the type of the value, which is converted to string for insertion
232 *
233 * \param it points to the element after which to insert
234 * \param k is the keyword
235 * \param v is the value
236 *
237 */
238 template <typename typeT>
239 void insert_after( headerIterator it, const std::string &k, typeT v );
240
241 /// Card access by keyword operator
242 /** Looks up the card by its keyword, and returns a reference to it.
243 *
244 * \param keyword is the header keyword to look up
245 *
246 * \retval fitsHeaderCard& reference to the \ref fitsHeaderCard
247 */
248 fitsHeaderCard &operator[]( const std::string &keyword );
249
250 /// Card access by keyword operator (const version)
251 /** Looks up the card by its keyword, and returns a reference to it.
252 *
253 * \param keyword is the header keyword to look up
254 *
255 * \retval fitsHeaderCard& const reference to the \ref fitsHeaderCard
256 */
257 const fitsHeaderCard &operator[]( const std::string &keyword ) const;
258
259}; // fitsHeader
260
261template <typename typeT>
262void fitsHeader::append( const std::string &k, const typeT &v, const std::string &c )
263{
264 append( fitsHeaderCard( k, v, c ) );
265}
266
267template <typename typeT>
268void fitsHeader::append( const std::string &k, const typeT &v )
269{
270 append( fitsHeaderCard( k, v ) );
271}
272
273template <typename typeT>
274void fitsHeader::insert_before( headerIterator it, const std::string &k, typeT v, const std::string &c )
275{
276 insert_before( it, fitsHeaderCard( k, v, c ) );
277}
278
279template <typename typeT>
280void fitsHeader::insert_before( headerIterator it, const std::string &k, typeT v )
281{
282 insert_before( it, fitsHeaderCard( k, v ) );
283}
284
285template <typename typeT>
286void fitsHeader::insert_after( headerIterator it, const std::string &k, typeT v, const std::string &c )
287{
288 insert_after( it, fitsHeaderCard( k, v, c ) );
289}
290
291template <typename typeT>
292void fitsHeader::insert_after( headerIterator it, const std::string &k, typeT v )
293{
294 insert_after( it, fitsHeaderCard( k, v ) );
295}
296
297/** \addtogroup fits_utils
298 * @{
299 */
300
301/// Convert the values in a std::vector of \ref fitsHeader "fits headers" into a std::vector of values.
302/** Resizes the vector of the appropriate type.
303 *
304 * \tparam dataT is the type of the header value
305 *
306 * \param[out] v will contain the converted values
307 * \param[in] heads contains the headers
308 * \param[in] keyw contains the keyword designating which value to convert
309 *
310 * \returns an optional which, if true, contains a vector of the indices of \p heads for which
311 * the extraction of a value for \p keyw failed
312 *
313 */
314template <typename dataT>
315std::optional<std::vector<size_t>> headersToValues( std::vector<dataT> &v, std::vector<fitsHeader> &heads, const std::string &keyw )
316{
317 v.resize( heads.size() );
318 std::vector<size_t> bad;
319 for( size_t i = 0; i < heads.size(); ++i )
320 {
321 try
322 {
323 v[i] = heads[i][keyw].value<dataT>(); // convertFromString<dataT>(heads[i][keyw].value);
324 }
325 catch(...)
326 {
327 bad.push_back(i);
328 v[i] = std::numeric_limits<dataT>::max();
329 }
330 }
331
332 if(bad.size() > 0)
333 {
334 return bad;
335 }
336 else
337 {
338 return {};
339 }
340
341}
342
343/// Write the status of a Git repository to HISTORY in a FITS header.
344/**
345 * \param [in,out] head the HISTORY cards will be appended to this header
346 * \param [in] repoName the name of the repository
347 * \param [in] sha1 is the SHA-1 hash string of the repository
348 * \param [in] modified whether or not the repository has been modified after the
349 * commit referred to by sha1
350 */
351void fitsHeaderGitStatus( fitsHeader &head, const std::string &repoName, const char *sha1, int modified );
352
353///@}
354
355} // namespace fits
356} // namespace mx
357
358#endif // ioutils_fits__fitsHeader_hpp
Class to manage the three components of a FITS header card.
Class to manage a complete fits header.
size_t count(const std::string &keyword)
Get number of cards with a given keyword.
~fitsHeader()
Destructor.
std::list< fitsHeaderCard >::iterator headerIterator
The iterator type for the cards list.
fitsHeader()
Default c'tor.
fitsHeaderCard & operator[](const std::string &keyword)
Card access by keyword operator.
headerIterator begin()
Get iterator to the beginning of the cards list.
void clear()
Clear all cards from the header.
std::list< fitsHeaderCard > cards
The storage for the FITS header cards.
void eraseStandardTop()
Erase the standard entries at the top of the header.
headerIterator iterator(const std::string &keyword)
Get iterator pointing to a specific element.
void erase(const std::string &keyword)
Erase card by keyword.
void insert_before(headerIterator it, fitsHeaderCard card)
Insert a card before another card.
size_t size()
Get number of cards currently stored in the header.
std::unordered_multimap< std::string, headerIterator > cardMap
This multimap allows for fast lookup by keyword.
std::unordered_multimap< std::string, headerIterator >::iterator mapIterator
The iterator type for the card map.
fitsHeader & operator=(const fitsHeader &head)
Assignment.
bool empty()
Test whether the header is empty.
void append(const fitsHeaderCard &card)
Append a fitsHeaderCard to the end of the header.
void insert_after(headerIterator it, fitsHeaderCard card)
Insert a card after another card.
headerIterator end()
Get iterator to the end of the cards list.
A class to work with a FITS header card.
void fitsHeaderGitStatus(fitsHeader &head, const std::string &repoName, const char *sha1, int modified)
Write the status of a Git repository to HISTORY in a FITS header.
std::optional< std::vector< size_t > > headersToValues(std::vector< dataT > &v, std::vector< fitsHeader > &heads, const std::string &keyw)
Convert the values in a std::vector of fits headers into a std::vector of values.
The mxlib c++ namespace.
Definition mxError.hpp:106