9 #ifndef app_MagAOXApp_hpp 10 #define app_MagAOXApp_hpp 20 #include <unordered_map> 22 #include <boost/filesystem.hpp> 24 #include <mx/mxlib.hpp> 25 #include <mx/app/application.hpp> 26 #include <mx/environment.hpp> 30 #include "../common/environment.hpp" 31 #include "../common/paths.hpp" 32 #include "../common/defaults.hpp" 33 #include "../common/config.hpp" 35 #include "../logger/logFileRaw.hpp" 36 #include "../logger/logManager.hpp" 67 template<
bool _useINDI = true >
113 const bool git_modified
126 virtual void setDefaults(
int argc,
138 virtual void setupBasicConfig();
148 virtual void loadBasicConfig();
154 virtual int execute();
168 virtual int appStartup() = 0;
179 virtual int appLogic() = 0;
184 virtual int appShutdown() = 0;
201 template<
typename logT,
int retval=0>
202 static int log(
const typename logT::messageT & msg,
213 template<
typename logT,
int retval=0>
226 int setSigTermHandler();
229 static void _handlerSigTerm(
int signum,
235 void handlerSigTerm(
int signum,
278 int m_RTPriority {0};
291 int RTPriority(
int prio );
333 int m_stateLogged {0} ;
380 constexpr
static bool m_useINDI = _useINDI;
391 pcf::IndiProperty *
property {0};
392 int (*callBack)(
void *,
const pcf::IndiProperty &) {0};
393 bool m_defReceived {
false};
407 bool m_allDefsReceived {
false};
439 int registerIndiPropertyNew( pcf::IndiProperty & prop,
440 const std::string & propName,
441 const pcf::IndiProperty::Type &
propType,
442 const pcf::IndiProperty::PropertyPermType & propPerm,
443 const pcf::IndiProperty::PropertyStateType & propState,
444 int (*)(
void *,
const pcf::IndiProperty &)
457 int registerIndiPropertySet( pcf::IndiProperty & prop,
458 const std::string & devName,
459 const std::string & propName,
460 int (*)(
void *,
const pcf::IndiProperty &)
467 int createINDIFIFOS();
478 void sendGetPropertySetList(
bool all=
false);
484 void handleDefProperty(
const pcf::IndiProperty &ipRecv );
490 void handleGetProperties(
const pcf::IndiProperty &ipRecv );
500 void handleNewProperty(
const pcf::IndiProperty &ipRecv );
509 void handleSetProperty(
const pcf::IndiProperty &ipRecv );
519 void updateIfChanged( pcf::IndiProperty & p,
520 const std::string & el,
531 int sendNewProperty(
const pcf::IndiProperty & ipSend,
532 const std::string & el,
553 bool m_powerMgtEnabled {
false};
557 std::string m_powerElement {
"state"};
559 int m_powerState {-1};
594 std::string configName();
600 std::string driverInName();
606 std::string driverOutName();
612 std::string driverCtrlName();
623 template<
bool _useINDI>
625 const bool git_modified
628 if( m_self !=
nullptr )
630 std::cerr <<
"Attempt to instantiate 2nd MagAOXApp. Exiting immediately.\n";
643 log<git_state>(
git_state::messageT(
"mxlib", MXLIB_UNCOMP_CURRENT_SHA1, MXLIB_UNCOMP_REPO_MODIFIED), gl);
646 getresuid(&m_euidReal, &m_euidCalled, &m_suid);
651 template<
bool _useINDI>
654 if(m_indiDriver)
delete m_indiDriver;
656 MagAOXApp<_useINDI>::m_self =
nullptr;
659 template<
bool _useINDI>
665 std::string configDir;
683 configDir = MagAOXPath +
"/" + tmpstr;
684 configPathGlobal = configDir +
"/magaox.conf";
689 m_log.logPath(tmpstr);
697 secretsPath = tmpstr;
700 #ifdef MAGAOX_configBase 702 configPathUser = configDir +
"/" + MAGAOX_configBase;
706 config.add(
"name",
"n",
"name",mx::argType::Required,
"",
"name",
false,
"string",
"The name of the application, specifies config.");
708 config.parseCommandLine(argc, argv,
"name");
709 config(m_configName,
"name");
711 if(m_configName ==
"")
713 boost::filesystem::path p(invokedName);
714 m_configName = p.stem().string();
715 log<text_log>(
"Application name (-n --name) not set. Using argv[0].");
719 configPathLocal = configDir +
"/" + m_configName +
".conf";
723 m_indiP_state.add (pcf::IndiElement(
"current"));
730 template<
bool _useINDI>
734 config.add(
"loopPause",
"p",
"loopPause", mx::argType::Required,
"",
"loopPause",
false,
"unsigned long",
"The main loop pause time in ns");
735 config.add(
"RTPriority",
"P",
"RTPriority", mx::argType::Required,
"",
"RTPriority",
false,
"unsigned",
"The real-time priority (0-99)");
741 config.add(
"power.device",
"",
"power.device", mx::argType::Required,
"power",
"device",
false,
"string",
"Device controlling power for this app's device (INDI name).");
742 config.add(
"power.outlet",
"",
"power.outlet", mx::argType::Required,
"power",
"outlet",
false,
"string",
"Outlet (or channel) on device for this app's device (INDI name).");
743 config.add(
"power.element",
"",
"power.element", mx::argType::Required,
"power",
"element",
false,
"string",
"INDI element name. Default is \"state\", only need to specify if different.");
746 template<
bool _useINDI>
750 m_log.logName(m_configName);
754 config(loopPause,
"loopPause");
757 int prio = m_RTPriority;
758 config(prio,
"RTPriority");
759 if(prio != m_RTPriority)
765 config(m_powerDevice,
"power.device");
766 config(m_powerOutlet,
"power.outlet");
767 config(m_powerElement,
"power.element");
769 if(m_powerDevice !=
"" && m_powerOutlet !=
"")
771 log<text_log>(
"enabling power management: " + m_powerDevice +
"." + m_powerOutlet +
"." + m_powerElement);
773 m_powerMgtEnabled =
true;
780 template<
bool _useINDI>
785 state(stateCodes::FAILURE);
800 std::this_thread::sleep_for( std::chrono::duration<unsigned long, std::nano>(500000));
809 std::cerr <<
"\nCRITICAL: log thread not running. Exiting.\n\n";
817 if( m_shutdown == 0 )
819 state(stateCodes::INITIALIZED);
820 if(appStartup() < 0) m_shutdown = 1;
824 if(m_useINDI && m_shutdown == 0)
828 state(stateCodes::FAILURE);
834 if(m_powerMgtEnabled)
836 while(m_powerState < 0)
841 if(!stateLogged()) log<text_log>(
"waiting for power state");
844 if(m_powerState > 0) state(stateCodes::POWERON);
845 else state(stateCodes::POWEROFF);
857 while( m_shutdown == 0)
859 if(m_powerMgtEnabled)
861 if(state() == stateCodes::POWEROFF)
863 if(m_powerState == 1)
865 state(stateCodes::POWERON);
870 if(m_powerState == 0)
872 state(stateCodes::POWEROFF);
884 if( !m_powerMgtEnabled || m_powerState > 0 )
892 else if(m_powerState == 0)
894 if( whilePowerOff() < 0)
910 sendGetPropertySetList(
false);
920 std::this_thread::sleep_for( std::chrono::duration<unsigned long, std::nano>(loopPause));
926 state(stateCodes::SHUTDOWN);
929 if(m_indiDriver !=
nullptr)
931 m_indiDriver->quitProcess();
932 m_indiDriver->deactivate();
933 log<indidriver_stop>();
941 template<
bool _useINDI>
942 template<
typename logT,
int retval>
947 m_log.
log<logT>(msg, level);
951 template<
bool _useINDI>
952 template<
typename logT,
int retval>
955 m_log.
log<logT>(level);
959 template<
bool _useINDI>
962 struct sigaction act;
966 act.sa_flags = SA_SIGINFO;
971 if( sigaction(SIGTERM, &act, 0) < 0 )
973 std::string logss =
"Setting handler for SIGTERM failed. Errno says: ";
974 logss += strerror(errno);
976 log<software_error>({__FILE__, __LINE__, errno, 0, logss});
982 if( sigaction(SIGQUIT, &act, 0) < 0 )
984 std::string logss =
"Setting handler for SIGQUIT failed. Errno says: ";
985 logss += strerror(errno);
987 log<software_error>({__FILE__, __LINE__, errno, 0,logss});
993 if( sigaction(SIGINT, &act, 0) < 0 )
995 std::string logss =
"Setting handler for SIGINT failed. Errno says: ";
996 logss += strerror(errno);
998 log<software_error>({__FILE__, __LINE__, errno, 0, logss});
1003 log<text_log>(
"Installed SIGTERM/SIGQUIT/SIGINT signal handler.",
logPrio::LOG_DEBUG);
1008 template<
bool _useINDI>
1017 template<
bool _useINDI>
1019 siginfo_t *siginf __attribute__((unused)),
1020 void *ucont __attribute__((unused))
1025 std::string signame;
1029 signame =
"SIGTERM";
1035 signame =
"SIGQUIT";
1041 std::string logss =
"Caught signal ";
1043 logss +=
". Shutting down.";
1045 std::cerr <<
"\n" << logss << std::endl;
1046 log<text_log>(logss);
1049 template<
bool _useINDI>
1053 if(seteuid(m_euidCalled) < 0)
1055 std::string logss =
"Setting effective user id to euidCalled (";
1056 logss += mx::ioutils::convertToString<int>(m_euidCalled);
1057 logss +=
") failed. Errno says: ";
1058 logss += strerror(errno);
1060 log<software_error>({__FILE__, __LINE__, errno, 0, logss});
1068 template<
bool _useINDI>
1072 if(seteuid(m_euidReal) < 0)
1074 std::string logss =
"Setting effective user id to euidReal (";
1075 logss += mx::ioutils::convertToString<int>(m_euidReal);
1076 logss +=
") failed. Errno says: ";
1077 logss += strerror(errno);
1079 log<software_error>({__FILE__, __LINE__, errno, 0, logss});
1088 template<
bool _useINDI>
1091 struct sched_param schedpar;
1093 if(prio < 0) prio = 0;
1094 if(prio > 99) prio = 99;
1095 schedpar.sched_priority = prio;
1098 if( euidCalled() < 0 )
1100 log<software_error>({__FILE__, __LINE__, 0, 0,
"Seeting euid to called failed."});
1110 else rv = sched_setscheduler(0, SCHED_OTHER, &schedpar);
1114 std::stringstream logss;
1115 logss <<
"Setting scheduler priority to " << prio <<
" failed. Errno says: " << strerror(errno) <<
". ";
1116 log<software_error>({__FILE__, __LINE__, errno, 0, logss.str()});
1120 m_RTPriority = prio;
1122 std::stringstream logss;
1123 logss <<
"Scheduler priority (RT_priority) set to " << m_RTPriority <<
".";
1124 log<text_log>(logss.str());
1128 if( euidReal() < 0 )
1130 log<software_error>({__FILE__, __LINE__, 0, 0,
"Setting euid to real failed."});
1137 template<
bool _useINDI>
1142 std::string statusDir = sysPath;
1145 if( euidCalled() < 0 )
1147 log<software_error>({__FILE__, __LINE__, 0, 0,
"Seeting euid to called failed."});
1153 if( mkdir(statusDir.c_str(), S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH) < 0 )
1155 if( errno != EEXIST)
1157 std::stringstream logss;
1158 logss <<
"Failed to create root of statusDir (" << statusDir <<
"). Errno says: " << strerror(errno);
1159 log<software_critical>({__FILE__, __LINE__, errno, 0, logss.str()});
1170 statusDir += m_configName;
1172 pidFileName = statusDir +
"/pid";
1176 if( mkdir(statusDir.c_str(), S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH) < 0 )
1178 if( errno != EEXIST)
1180 std::stringstream logss;
1181 logss <<
"Failed to create statusDir (" << statusDir <<
"). Errno says: " << strerror(errno);
1182 log<software_critical>({__FILE__, __LINE__, errno, 0, logss.str()});
1192 std::ifstream pidIn;
1193 pidIn.open( pidFileName );
1203 std::stringstream procN;
1204 procN <<
"/proc/" << testPid <<
"/cmdline";
1206 std::ifstream procIn;
1207 std::string pidCmdLine;
1211 procIn.open(procN.str());
1212 if(procIn.good()) procIn >> pidCmdLine;
1217 log<software_critical>({__FILE__, __LINE__, 0, 0,
"exception caught testing /proc/pid"});
1226 size_t invokedPos = pidCmdLine.find( invokedName );
1229 size_t configPos = std::string::npos;
1230 if(invokedPos != std::string::npos) configPos = pidCmdLine.find( m_configName );
1233 if( invokedPos != std::string::npos && configPos != std::string::npos)
1236 std::stringstream logss;
1237 logss <<
"PID already locked (" << testPid <<
"). Time to die.";
1238 std::cerr << logss.str() << std::endl;
1256 std::ofstream pidOut;
1257 pidOut.open(pidFileName);
1261 log<software_critical>({__FILE__, __LINE__, errno, 0,
"could not open pid file for writing."});
1270 std::stringstream logss;
1271 logss <<
"PID (" << m_pid <<
") locked.";
1272 log<text_log>(logss.str());
1275 if( euidReal() < 0 )
1277 log<software_error>({__FILE__, __LINE__, 0, 0,
"Seeting euid to real failed."});
1284 template<
bool _useINDI>
1287 if( ::
remove(pidFileName.c_str()) < 0)
1289 log<software_error>({__FILE__, __LINE__, errno, 0, std::string(
"Failed to remove PID file: ") + strerror(errno)});
1293 std::stringstream logss;
1294 logss <<
"PID (" << m_pid <<
") unlocked.";
1295 log<text_log>(logss.str());
1300 template<
bool _useINDI>
1306 template<
bool _useINDI>
1316 log<state_change>( {m_state, s}, lvl );
1323 std::unique_lock<std::mutex> lock(m_indiMutex, std::try_to_lock);
1325 if(lock.owns_lock())
1327 updateIfChanged(m_indiP_state,
"current", m_state);
1331 template<
bool _useINDI>
1334 if(m_stateLogged > 0)
1337 return m_stateLogged - 1;
1350 template<
bool _useINDI>
1352 const std::string & propName,
1353 const pcf::IndiProperty::Type & propType,
1354 const pcf::IndiProperty::PropertyPermType & propPerm,
1355 const pcf::IndiProperty::PropertyStateType & propState,
1356 int (*callBack)(
void *,
const pcf::IndiProperty &ipRecv)
1359 if(!m_useINDI)
return 0;
1361 prop = pcf::IndiProperty (propType);
1362 prop.setDevice(m_configName);
1363 prop.setName(propName);
1364 prop.setPerm(propPerm);
1365 prop.setState( propState);
1368 callBackInsertResult result = m_indiNewCallBacks.insert(callBackValueType( propName, {&prop, callBack}));
1378 template<
bool _useINDI>
1380 const std::string & devName,
1381 const std::string & propName,
1382 int (*callBack)(
void *,
const pcf::IndiProperty &ipRecv)
1385 if(!m_useINDI)
return 0;
1387 prop = pcf::IndiProperty();
1388 prop.setDevice(devName);
1389 prop.setName(propName);
1391 callBackInsertResult result = m_indiSetCallBacks.insert(callBackValueType( devName +
"." + propName, {&prop, callBack}));
1401 template<
bool _useINDI>
1404 if(!m_useINDI)
return 0;
1408 driverFIFOPath +=
"/";
1411 m_driverInName = driverFIFOPath +
"/" + configName() +
".in";
1412 m_driverOutName = driverFIFOPath +
"/" + configName() +
".out";
1413 m_driverCtrlName = driverFIFOPath +
"/" + configName() +
".ctrl";
1419 mode_t prev = umask(0);
1422 if(mkfifo(m_driverInName.c_str(), S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP) !=0)
1428 log<software_critical>({__FILE__, __LINE__, errno, 0,
"mkfifo failed"});
1435 if(mkfifo(m_driverOutName.c_str(), S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP) !=0 )
1441 log<software_critical>({__FILE__, __LINE__, errno, 0,
"mkfifo failed"});
1448 if(mkfifo(m_driverCtrlName.c_str(), S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP) !=0 )
1454 log<software_critical>({__FILE__, __LINE__, errno, 0,
"mkfifo failed"});
1465 template<
bool _useINDI>
1468 if(!m_useINDI)
return 0;
1472 if(createINDIFIFOS() < 0)
1484 log<software_critical>({__FILE__, __LINE__, 0, 0,
"INDI Driver construction exception."});
1489 if(m_indiDriver ==
nullptr)
1491 log<software_critical>({__FILE__, __LINE__, 0, 0,
"INDI Driver construction failed."});
1496 if(m_indiDriver->good() ==
false)
1498 log<software_critical>({__FILE__, __LINE__, 0, 0,
"INDI Driver failed to open FIFOs."});
1499 delete m_indiDriver;
1500 m_indiDriver =
nullptr;
1505 m_indiDriver->activate();
1506 log<indidriver_start>();
1508 sendGetPropertySetList();
1513 template<
bool _useINDI>
1517 if(!all && m_allDefsReceived)
return;
1519 callBackIterator it = m_indiSetCallBacks.begin();
1522 while(it != m_indiSetCallBacks.end() )
1524 if(all || it->second.m_defReceived ==
false)
1526 if( it->second.property )
1528 m_indiDriver->sendGetProperties( *(it->second.property) );
1531 it->second.m_defReceived =
false;
1537 if(nowFalse != 0) m_allDefsReceived =
false;
1538 if(nowFalse == 0) m_allDefsReceived =
true;
1541 template<
bool _useINDI>
1544 handleSetProperty(ipRecv);
1547 template<
bool _useINDI>
1550 if(!m_useINDI)
return;
1551 if(m_indiDriver ==
nullptr)
return;
1554 if (ipRecv.hasValidDevice() && ipRecv.getDevice() != m_indiDriver->getName())
1560 if( !ipRecv.hasValidName() )
1562 callBackIterator it = m_indiNewCallBacks.begin();
1564 while(it != m_indiNewCallBacks.end() )
1566 if( it->second.property )
1568 m_indiDriver->sendDefProperty( *(it->second.property) );
1574 sendGetPropertySetList(
true);
1580 if( m_indiNewCallBacks.count(ipRecv.getName()) == 0)
1586 if(m_indiNewCallBacks[ ipRecv.getName() ].property)
1588 m_indiDriver->sendDefProperty( *(m_indiNewCallBacks[ ipRecv.getName() ].property) );
1593 template<
bool _useINDI>
1596 if(!m_useINDI)
return;
1597 if(m_indiDriver ==
nullptr)
return;
1600 if( m_indiNewCallBacks.count(ipRecv.getName()) == 0 )
1606 int (*callBack)(
void *,
const pcf::IndiProperty &) = m_indiNewCallBacks[ ipRecv.getName() ].callBack;
1608 if(callBack) callBack(
this, ipRecv);
1615 template<
bool _useINDI>
1618 if(!m_useINDI)
return;
1619 if(m_indiDriver ==
nullptr)
return;
1621 std::string key = ipRecv.getDevice() +
"." + ipRecv.getName();
1624 if( m_indiSetCallBacks.count(key) > 0 )
1626 m_indiSetCallBacks[ key ].m_defReceived =
true;
1629 int (*callBack)(
void *,
const pcf::IndiProperty &) = m_indiSetCallBacks[ key ].callBack;
1630 if(callBack) callBack(
this, ipRecv);
1642 template<
bool _useINDI>
1643 template<
typename T>
1645 const std::string & el,
1649 if(!_useINDI)
return;
1651 if(!m_indiDriver)
return;
1653 T oldVal = p[el].get<T>();
1655 if(oldVal != newVal)
1658 p.setState (pcf::IndiProperty::Ok);
1659 m_indiDriver->sendSetProperty (p);
1665 template<
typename T>
1668 return pcf::IndiProperty::Unknown;
1674 pcf::IndiProperty::Type propType<char *>()
1676 return pcf::IndiProperty::Text;
1681 pcf::IndiProperty::Type propType<std::string>()
1683 return pcf::IndiProperty::Text;
1690 return pcf::IndiProperty::Number;
1697 return pcf::IndiProperty::Number;
1700 template<
bool _useINDI>
1701 template<
typename T>
1703 const std::string & el,
1707 if(!_useINDI)
return 0;
1711 log<software_error>({__FILE__, __LINE__,
"INDI communications not initialized."});
1714 pcf::IndiProperty ipToSend = ipSend;
1718 ipToSend[el].setValue(newVal);
1722 log<software_error>({__FILE__, __LINE__,
"Exception caught setting " + ipSend.getDevice() +
"." + ipSend.getName() +
"." + el});
1729 log<software_error>({__FILE__, __LINE__});
1736 template<
bool _useINDI>
1745 ps = ipRecv[m_powerElement].get<std::string>();
1749 log<software_error>({__FILE__, __LINE__,
"Exception caught."});
1757 else if (ps ==
"Off")
1769 template<
bool _useINDI>
1772 return m_configName;
1775 template<
bool _useINDI>
1778 return m_driverInName;
1781 template<
bool _useINDI>
1784 return m_driverOutName;
1787 template<
bool _useINDI>
1790 return m_driverCtrlName;
1796 #endif //app_MagAOXApp_hpp std::string m_driverOutName
Full path name of the INDI driver output FIFO.
pcf::IndiProperty::Type propType()
int logThreadStart()
Start the logger thread.
pcf::IndiProperty::Type propType< int >()
virtual int whilePowerOff()
This method is called while the power is off, once per FSM loop.
static constexpr logPrioT LOG_INFO
Informational. The info log level is the lowest level recorded during normal operations.
The standard MagAOX log manager, used for both process logs and telemetry streams.
#define MAGAOX_env_config
Environment variable setting the relative config path.
#define MAGAOX_logRelPath
The relative path to the log directory.
static constexpr logPrioT LOG_WARNING
A condition has occurred which may become an error, but the process continues.
#define REG_INDI_SETPROP(prop, devName, propName)
Register a SET INDI property with the class, using the standard callback name.
std::unordered_map< std::string, indiCallBack > m_indiNewCallBacks
Map to hold the NewProperty indiCallBacks for this App, with fast lookup by property name...
void log(const typename logT::messageT &msg, logPrioT level=logPrio::LOG_DEFAULT)
Make a log entry, including a message.
std::string pidFileName
The name of the PID file.
#define REG_INDI_NEWPROP_NOCB(prop, propName, type)
Register a NEW INDI property with the class, with no callback.
std::string m_configName
The name of the configuration file (minus .conf).
#define INDI_SETCALLBACK_DEFN(class, prop)
Define the callback for a set property request.
pcf::IndiProperty::Type propType< double >()
std::string m_powerDevice
std::string m_powerOutlet
int sendNewProperty(const pcf::IndiProperty &ipSend, const std::string &el, const T &newVal)
Send a newProperty command to another device (using the INDI Client interface)
static constexpr logPrioT LOG_ERROR
An error has occured which the software will attempt to correct.
static MagAOXApp * m_self
Static pointer to this (set in constructor). Used to test whether a a MagAOXApp is already instatiate...
std::string m_driverCtrlName
Full path name of the INDI driver control FIFO.
#define MAGAOX_default_loopPause
The default application loopPause.
The base-class for MagAO-X applications.
std::unordered_map< std::string, indiCallBack >::iterator callBackIterator
Iterator type of the indiCallBack map.
#define MAGAOX_secretsRelPath
The relative path to the secrets directory. Used for storing passwords, etc.
std::string sysPath
The path to the system directory, for PID file, etc.
std::string m_driverInName
Full path name of the INDI driver input FIFO.
pcf::IndiProperty m_indiP_powerOutlet
#define MAGAOX_RT_SCHED_POLICY
The real-time scheduling policy.
virtual int onPowerOff()
This method is called when the change to poweroff is detected.
#define MAGAOX_driverFIFORelPath
The relative path to the INDI driver FIFOs.
uid_t m_euidCalled
The user id of the process as called (i.e. the higher privileged id of the owner, root if setuid)...
std::unordered_map< std::string, indiCallBack > m_indiSetCallBacks
Map to hold the SetProperty indiCallBacks for this App, with fast lookup by property name...
MagAO-X INDI Driver Wrapper.
int8_t logPrioT
The type of the log priority code.
std::pair< callBackIterator, bool > callBackInsertResult
Return type of insert on the indiCallBack map.
MagAO-X Application States.
uid_t m_euidReal
The real user id of the proces (i.e. the lower privileged id of the user)
#define MAGAOX_sysRelPath
The relative path to the system directory.
int loadConfig(mx::appConfigurator &config)
Load the logger section from an application configurator.
std::string MagAOXPath
The base path of the MagAO-X system.
#define MAGAOX_configRelPath
The relative path to the configuration files.
int16_t stateCodeT
The type of the state code.
uid_t m_suid
The save-set user id of the process.
#define MAGAOX_path
The path to the MagAO-X system files.
The type of the input message.
#define INDI_SETCALLBACK_DECL(class, prop)
Declare the callback for a set property request, and declare and define the static wrapper...
bool logThreadRunning()
Get status of the log thread running flag.
static constexpr logPrioT LOG_CRITICAL
The process can not continue and will shut down (fatal)
std::mutex m_indiMutex
Mutex for locking INDI communications.
logger::logManager< logFileRaw > logManagerT
The log manager type.
void handlerSigTerm(int signum, siginfo_t *siginf, void *ucont)
Handles SIGTERM, SIGQUIT, and SIGINT. Sets m_shutdown to 1 and logs the signal.
Structure to hold the call-back details for handling INDI communications.
static constexpr logPrioT LOG_DEFAULT
Used to denote "use the default level for this log type".
#define MAGAOX_env_path
Environment variable setting the MagAO-X path.
std::pair< std::string, indiCallBack > callBackValueType
Value type of the indiCallBack map.
static constexpr logPrioT LOG_DEBUG
Used for debugging.
std::string secretsPath
Path to the secrets directory, where passwords, etc, are stored.
int setupConfig(mx::appConfigurator &config)
Setup an application configurator for the logger section.
pcf::IndiProperty m_indiP_state
indi Property to report the application state.