9#ifndef __ADIobservation_hpp__
10#define __ADIobservation_hpp__
11#include "../ioutils/fileUtils.hpp"
14#include "../ioutils/fits/fitsHeader.hpp"
93template <
typename _realT,
class _derotFunctObj>
97 typedef _derotFunctObj derotFunctObj;
98 typedef Eigen::Array<realT, Eigen::Dynamic, Eigen::Dynamic> eigenImageT;
100 derotFunctObj m_derotF;
102 derotFunctObj m_RDIderotF;
104 bool m_doDerotate{
true };
106 bool m_postMedSub{
false };
111 const std::string &prefix,
112 const std::string &ext
118 const std::string &dir,
119 const std::string &prefix,
120 const std::string &ext,
121 const std::string &RDIdir,
124 const std::string &RDIext =
""
129 const std::string &RDIfileListFile
168 int readPSFSub(
const std::string &dir,
const std::string &prefix,
const std::string &ext,
size_t nReductions );
194 std::vector<std::string> &fileList,
196 derotFunctObj &derotF,
227 double t_fake_begin{ 0 };
228 double t_fake_end{ 0 };
230 double t_derotate_begin{ 0 };
231 double t_derotate_end{ 0 };
234template <
typename _realT,
class _derotFunctObj>
235ADIobservation<_realT, _derotFunctObj>::ADIobservation()
239template <
typename _realT,
class _derotFunctObj>
241 const std::string &prefix,
242 const std::string &ext )
247template <
typename _realT,
class _derotFunctObj>
253template <
typename _realT,
class _derotFunctObj>
255 const std::string &prefix,
256 const std::string &ext,
257 const std::string &RDIdir,
258 const std::string &RDIprefix,
259 const std::string &RDIext )
260 :
HCIobservation<realT>( dir, prefix, ext, RDIdir, RDIprefix, RDIext )
264template <
typename _realT,
class _derotFunctObj>
266 const std::string &RDIfileListFile )
271template <
typename _realT,
class _derotFunctObj>
274 this->m_keywords.clear();
276 if( !m_derotF.isSetup() )
278 mxError(
"ADIobservation::readFiles", MXE_PARAMNOTSET,
"Derotator is not configured." );
284 for(
size_t i = 0; i < m_derotF.m_keywords.size(); ++i )
286 this->m_keywords.push_back( m_derotF.m_keywords[i] );
297template <
typename _realT,
class _derotFunctObj>
300 std::optional<std::vector<size_t>> bad = m_derotF.extractKeywords( this->m_heads );
304 for(
size_t n = 0; n < bad->size(); ++n )
306 std::cerr << this->m_fileList[( *bad )[n]] <<
" conversion failed for " << m_derotF.m_angleKeyword <<
"\n";
310 "ADIobservation::postReadFiles",
311 "bad derotation angles in FITS header" );
314 if( m_fakeFileName !=
"" && !this->m_skipPreProcess )
316 std::cerr <<
"Injecting fakes in target images...\n";
317 if( injectFake( this->m_tgtIms, this->m_fileList, m_derotF, 1, 1 ) < 0 )
326template <
typename _realT,
class _derotFunctObj>
329 m_derotF.extractKeywords( this->m_heads );
333template <
typename _realT,
class _derotFunctObj>
336 this->m_RDIkeywords.clear();
338 if( !m_RDIderotF.isSetup() )
340 mxError(
"ADIobservation::readRDIFiles", MXE_PARAMNOTSET,
"Derotator is not configured." );
346 for(
size_t i = 0; i < m_RDIderotF.m_keywords.size(); ++i )
348 this->m_RDIkeywords.push_back( m_RDIderotF.m_keywords[i] );
357template <
typename _realT,
class _derotFunctObj>
360 m_RDIderotF.extractKeywords( this->m_RDIheads );
370template <
typename _realT,
class _derotFunctObj>
373 m_RDIderotF.extractKeywords( this->m_RDIheads );
377template <
typename _realT,
class _derotFunctObj>
379 const std::string &prefix,
380 const std::string &ext,
390 ff.
read( im, fh, flist[0] );
392 if( !m_derotF.isSetup() )
394 mxError(
"ADIobservation::readFiles", MXE_PARAMNOTSET,
"Derotator is not configured." );
398 if( fh.
count(
"POSTMEDS" ) != 0 )
400 m_postMedSub = fh[
"POSTMEDS"].Int();
401 std::cerr <<
"postMedSub: " << m_postMedSub <<
"\n";
404 if( fh.
count(
"FAKEFILE" ) != 0 )
406 m_fakeFileName = fh[
"FAKEFILE"].String();
407 std::cerr <<
"fakeFileName: " << m_fakeFileName <<
"\n";
410 if( fh.
count(
"FAKESCFL" ) != 0 )
412 m_fakeScaleFileName = fh[
"FAKESCFL"].String();
413 std::cerr <<
"fakeScaleFileName: " << m_fakeScaleFileName <<
"\n";
416 if( fh.
count(
"FAKESEP" ) != 0 )
420 if( m_fakeSep.size() == 0 )
422 mxError(
"KLIPReduction", MXE_PARSEERR,
"FAKESEP vector did not parse correctly." );
425 std::cerr <<
"fakeSep: " << fh[
"FAKESEP"].String() <<
"\n";
428 if( fh.
count(
"FAKEPA" ) != 0 )
432 if( m_fakePA.size() == 0 )
434 mxError(
"KLIPReduction", MXE_PARSEERR,
"FAKEPA vector did not parse correctly." );
437 std::cerr <<
"fakePA: " << fh[
"FAKEPA"].String() <<
"\n";
440 if( fh.
count(
"FAKECONT" ) != 0 )
444 if( m_fakeContrast.size() == 0 )
446 mxError(
"KLIPReduction", MXE_PARSEERR,
"FAKECONT vector did not parse correctly." );
449 std::cerr <<
"fakeContrast: " << fh[
"FAKECONT"].String() <<
"\n";
454 for(
size_t i = 0; i < m_derotF.m_keywords.size(); ++i )
456 this->m_keywords.push_back( m_derotF.m_keywords[i] );
462 m_derotF.extractKeywords( this->m_heads );
467template <
typename _realT,
class _derotFunctObj>
469 std::vector<std::string> &fileList,
470 derotFunctObj &derotF,
478 std::vector<std::string> fakeFiles;
481 std::ifstream scaleFin;
484 std::vector<realT> fakeScale( ims.planes(), 1.0 );
485 if( m_fakeScaleFileName !=
"" )
487 std::vector<std::string> sfileNames;
488 std::vector<realT> imS;
494 if( sfileNames.size() != imS.size() )
496 std::cerr <<
"fake scale file must be two columns of: fileName scale\n";
499 std::map<std::string, realT> scales;
500 for(
size_t i = 0; i < sfileNames.size(); ++i )
503 for(
size_t i = 0; i < fileList.size(); ++i )
511 std::cerr <<
"File name not found in fakeScaleFile:\n";
520 if( ff.
read( fakePSF, m_fakeFileName ) < 0 )
530 for(
int i = 0; i < ims.planes(); ++i )
534 ff.
read( fakePSF, fakeFiles[i] );
537 for(
size_t j = 0; j < m_fakeSep.size(); ++j )
539 if( injectFake( fakePSF,
542 derotF.derotAngle( i ),
558template <
typename _realT,
class _derotFunctObj>
571 if( ( fakePSF.rows() < ims.rows() && fakePSF.cols() >= ims.cols() ) ||
572 ( fakePSF.rows() >= ims.rows() && fakePSF.cols() < ims.cols() ) )
575 "ADIobservation::injectFake",
576 "fake PSF has different dimensions and can't be sized properly" );
580 if( fakePSF.rows() < ims.rows() && fakePSF.cols() < ims.cols() )
582 eigenImageT pfake( ims.rows(), ims.cols() );
583 padImage( pfake, fakePSF, 0.5 * ( ims.rows() - fakePSF.rows() ), 0 );
588 if( fakePSF.rows() > ims.rows() && fakePSF.cols() > ims.cols() )
590 eigenImageT cfake( ims.rows(), ims.cols() );
591 cutPaddedImage( cfake, fakePSF, 0.5 * ( fakePSF.rows() - ims.rows() ) );
595 if( fakePSF.rows() != ims.rows() || fakePSF.cols() != ims.cols() )
599 "ADIobservation::injectFake",
600 "fake PSF has different dimensions and can't be sized properly (is it even in rows and cols?)" );
605 eigenImageT shiftFake( fakePSF.rows(), fakePSF.cols() );
611 dx = sep * RDISepScale * sin( ang );
612 dy = sep * RDISepScale * cos( ang );
616 ims.
image( image_i ) = ims.
image( image_i ) + shiftFake * scale * RDIFluxScale * contrast;
621template <
typename realT,
class derotFunctObj>
624 if( this->m_mask.rows() != this->m_Nrows || this->m_mask.cols() != this->m_Ncols )
627 std::string message =
"Mask is not the same size as images.\n";
628 message +=
" Mask: rows=" + std::to_string(this->m_mask.rows()) +
"\n";
629 message +=
" cols=" + std::to_string(this->m_mask.cols()) +
"\n";
630 message +=
" Images: rows=" + std::to_string(this->m_Nrows) +
"\n";
631 message +=
" cols=" + std::to_string(this->m_Ncols) +
"\n";
634 mxThrowException(
err::invalidconfig,
"ADIobservation<realT, derotFunctObj>::makeMaskCube", message );
637 this->m_maskCube.resize( this->m_Nrows, this->m_Ncols, this->m_Nims );
644 for(
int i = 0; i < this->m_Nims; ++i )
646 rotateMask( rm, this->m_mask, m_derotF.derotAngle( i ) );
647 this->m_maskCube.image( i ) = rm;
652 std::ofstream fout( this->m_auxDataDir +
"angles.dat" );
653 for(
int i = 0; i < this->m_Nims; ++i )
655 fout << m_derotF.derotAngle( i ) <<
"\n";
660 ff.
write( this->m_auxDataDir +
"maskCube.fits", this->m_maskCube );
663template <
typename _realT,
class _derotFunctObj>
668 for(
size_t n = 0; n < this->m_psfsub.size(); ++n )
676 for(
int i = 0; i < this->m_psfsub[n].planes(); ++i )
678 derot = m_derotF.derotAngle( i );
682 this->m_psfsub[n].image( i ) = rotim;
693template <
typename _realT,
class _derotFunctObj>
699 head->
append(
"", fits::fitsCommentType(),
"----------------------------------------" );
700 head->
append(
"", fits::fitsCommentType(),
"mx::ADIobservation parameters:" );
701 head->
append(
"", fits::fitsCommentType(),
"----------------------------------------" );
703 head->
append(
"POSTMEDS", m_postMedSub,
"median subtraction after processing" );
705 if( m_fakeFileName !=
"" )
707 head->
append(
"FAKEFILE", m_fakeFileName,
"name of fake planet PSF file" );
709 if( m_fakeScaleFileName !=
"" )
711 head->
append(
"FAKESCFL", m_fakeScaleFileName,
"name of fake planet scale file name" );
714 std::stringstream str;
716 if( m_fakeSep.size() > 0 )
718 for(
size_t nm = 0; nm < m_fakeSep.size() - 1; ++nm )
720 str << m_fakeSep[nm] <<
",";
722 str << m_fakeSep[m_fakeSep.size() - 1];
723 head->
append<
char *>(
"FAKESEP", (
char *)str.str().c_str(),
"separation of fake planets" );
726 if( m_fakePA.size() > 0 )
729 for(
size_t nm = 0; nm < m_fakePA.size() - 1; ++nm )
731 str << m_fakePA[nm] <<
",";
733 str << m_fakePA[m_fakePA.size() - 1];
734 head->
append<
char *>(
"FAKEPA", (
char *)str.str().c_str(),
"PA of fake planets" );
737 if( m_fakeContrast.size() > 0 )
740 for(
size_t nm = 0; nm < m_fakeContrast.size() - 1; ++nm )
742 str << m_fakeContrast[nm] <<
",";
744 str << m_fakeContrast[m_fakeContrast.size() - 1];
745 head->
append<
char *>(
"FAKECONT", (
char *)str.str().c_str(),
"Contrast of fake planets" );
750template <
typename realT>
753extern template class ADIobservation<float, ADIDerotator<float>>;
754extern template class ADIobservation<double, ADIDerotator<double>>;
Defines the basic high contrast imaging data type.
mxException for invalid arguments
mxException for invalid config settings
mxException for a size error
Class to manage interactions with a FITS file.
int write(const dataT *im, int d1, int d2, int d3, fitsHeader *head)
Write the contents of a raw array to the FITS file.
int read(dataT *data)
Read the contents of the FITS file into an array.
An image cube with an Eigen-like API.
Eigen::Map< Eigen::Array< dataT, Eigen::Dynamic, Eigen::Dynamic > > image(Index n)
Returns a 2D Eigen::Eigen::Map pointed at the specified image.
int readColumns(const std::string &fname, arrTs &...arrays)
Read in columns from a text file.
Eigen::Array< scalarT, -1, -1 > eigenImage
Definition of the eigenImage type, which is an alias for Eigen::Array.
std::vector< std::string > getFileNames(const std::string &directory, const std::string &prefix, const std::string &substr, const std::string &extension)
Get a list of file names from the specified directory, specifying a prefix, a substring to match,...
int createDirectories(const std::string &path)
Create a directory or directories.
std::string pathFilename(const std::string &fname)
Get the base filename.
realT dtor(realT q)
Convert from degrees to radians.
fakeMethods
Fake injection PSF file specification methods.
@ single
A single PSF is used.
@ list
A list of PSF files, one per input image, is used.
int padImage(imOutT &imOut, imInT &imIn, unsigned int padSz, typename imOutT::Scalar value)
Pad an image with a constant value.
int cutPaddedImage(imOutT &imOut, const imInT &imIn, unsigned int padSz)
Cut down a padded image.
void parseStringVector(std::vector< typeT > &v, const std::string &s, char delim=',')
Parses a string into a vector of tokens delimited by a character.
typeT get_curr_time()
Get the current system time in seconds.
void rotateMask(imageT &rotMask, imageT &mask, typename imageT::Scalar angle)
Rotate a binary mask.
int fakeMethodFmStr(const std::string &method)
Get the fake injection method from its string name.
std::string fakeMethodsStr(int method)
Get the string name of a fake injection method.
Process an angular differential imaging (ADI) observation.
int readRDIFiles()
Read in the RDI files.
virtual int postRDIReadFiles()
Post reference read actions, including fake injection.
int injectFake(eigenCube< realT > &ims, std::vector< std::string > &fileList, derotFunctObj &derotF, realT RDIfluxScale, realT RDISepScale)
Inkect the fake plants.
std::vector< realT > m_fakeSep
Separation(s) of the fake planet(s)
virtual int postReadFiles()
Actions to take after the files are first read in by HCIobservation.
int readFiles()
Read in the target files.
std::vector< realT > m_fakePA
Position angles(s) of the fake planet(s)
std::vector< realT > m_fakeContrast
Contrast(s) of the fake planet(s)
virtual int postRDICoadd()
Post reference coadd actions.
void derotate()
De-rotate the PSF subtracted images.
std::string m_fakeScaleFileName
One-column text file containing a scale factor for each point in time.
std::string m_fakeFileName
FITS file containing the fake planet PSF to inject or a list of fake images.
int m_fakeMethod
Method for reading fake files, either HCI::single or HCI::list.
virtual int postCoadd()
Post target coadd actions.
virtual void makeMaskCube()
Populate the mask cube which is used for post-processing.
int readPSFSub(const std::string &dir, const std::string &prefix, const std::string &ext, size_t nReductions)
Read in already PSF-subtracted files.
The basic high contrast imaging data type.