MagAO-X
Operations Applications Utilities Source
MagAOXApp.hpp
Go to the documentation of this file.
1 /** \file magAOXApp.hpp
2  * \brief The basic MagAO-X Application
3  * \author Jared R. Males (jaredmales@gmail.com)
4  *
5  * History:
6  * - 2017-12-24 created by JRM
7  */
8 
9 #ifndef app_MagAOXApp_hpp
10 #define app_MagAOXApp_hpp
11 
12 
13 #include <signal.h>
14 #include <sys/stat.h>
15 
16 #include <cstdlib>
17 #include <unistd.h>
18 #include <fstream>
19 
20 #include <unordered_map>
21 
22 #include <boost/filesystem.hpp>
23 
24 #include <mx/mxlib.hpp>
25 #include <mx/app/application.hpp>
26 #include <mx/environment.hpp>
27 
28 
29 
30 #include "../common/environment.hpp"
31 #include "../common/paths.hpp"
32 #include "../common/defaults.hpp"
33 #include "../common/config.hpp"
34 
35 #include "../logger/logFileRaw.hpp"
36 #include "../logger/logManager.hpp"
37 
38 
39 
40 #include "stateCodes.hpp"
41 #include "indiDriver.hpp"
42 #include "indiMacros.hpp"
43 
44 //#include "../../INDI/libcommon/System.hpp"
45 
46 using namespace MagAOX::logger;
47 
48 namespace MagAOX
49 {
50 namespace app
51 {
52 
53 /// The base-class for MagAO-X applications.
54 /**
55  * You can define a base configuration file for this class by writing
56  * \code
57  * #define MAGAOX_configBase "relative/path/from/configDir.conf"
58  * \endcode
59  * before including MagAOXApp.hpp. This would be used, for instance to have a config common to
60  * all filter wheels.
61  *
62  *
63  * \todo do we need libMagAOX error handling? (a stack?)
64  *
65  * \ingroup magaoxapp
66  */
67 template< bool _useINDI = true >
68 class MagAOXApp : public mx::application
69 {
70 
71 public:
72 
73  ///The log manager type.
75 
76 protected:
77 
78  std::string MagAOXPath; ///< The base path of the MagAO-X system.
79 
80  std::string m_configName; ///< The name of the configuration file (minus .conf).
81 
82  std::string sysPath; ///< The path to the system directory, for PID file, etc.
83 
84  std::string secretsPath; ///< Path to the secrets directory, where passwords, etc, are stored.
85 
86  unsigned long loopPause {MAGAOX_default_loopPause}; ///< The time in nanoseconds to pause the main loop. The appLogic() function of the derived class is called every loopPause nanoseconds. Default is 1,000,000,000 ns. Config with loopPause=X.
87 
88  int m_shutdown {0}; ///< Flag to signal it's time to shutdown. When not 0, the main loop exits.
89 
90 private:
91 
92  ///Default c'tor is deleted.
93  MagAOXApp() = delete;
94 
95 public:
96 
97  /// Public c'tor. Handles uid, logs git repo status, and initializes static members.
98  /**
99  * Only one MagAOXApp can be instantiated per program. Hence this c'tor will issue exit(-1)
100  * if the static self-pointer m_self is already initialized.
101  *
102  * euid is set to 'real' to ensure that the application has normal privileges unless
103  * explicitly needed.
104  *
105  * Reference: http://man7.org/linux/man-pages/man2/getresuid.2.html
106  *
107  * The git repository status is required to create a MagAOXApp. Derived classes
108  * should include the results of running `gengithead.sh` and pass the defined
109  * sha1 and modified flags.
110  *
111  */
112  MagAOXApp( const std::string & git_sha1, ///< [in] The current SHA1 hash of the git repository
113  const bool git_modified ///< [in] Whether or not the repo is modified.
114  );
115 
116  ~MagAOXApp() noexcept(true);
117 
118  /// Set the paths for config files
119  /** Replaces the mx::application defaults with the MagAO-X config system.
120  *
121  * This function parses the CL for "-n" or "--name".
122  *
123  *
124  * Do not override this unless you intend to depart from the MagAO-X standard.
125  */
126  virtual void setDefaults( int argc, ///< [in] standard command line result specifying number of arguments in argv
127  char ** argv ///< [in] standard command line result containing the arguments.
128  );
129 
130  /// The basic MagAO-X configuration setup method. Should not normally be overridden.
131  /** This method sets up the config system with the standard MagAO-X key=value pairs.
132  *
133  * Though it is virtual, it should not normally be overridden unless you need
134  * to depart from the MagAO-X standard.
135  *
136  * Setting up app specific config goes in setupConfig() implemented in the derived class.
137  */
138  virtual void setupBasicConfig();
139 
140  /// The basic MagAO-X configuration processing method. Should not normally be overridden.
141  /** This method processes the standard MagAO-X key=value pairs.
142  *
143  * Though it is virtual, it should not normally be overridden unless you need
144  * to depart from the MagAO-X standard.
145  *
146  * Processing of app specific config goes in loadConfig() implemented by the derived class.
147  */
148  virtual void loadBasicConfig();
149 
150  /// The execute method implementing the standard main loop. Should not normally be overridden.
151  /**
152  *
153  */
154  virtual int execute();
155 
156 
157  /** \name Pure Virtual Functions
158  * Derived applications must implement these.
159  * @{
160  */
161 
162  /// Any tasks to perform prior to the main event loop go here.
163  /** This is called after signal handling is installed. FSM state is
164  * stateCodes::INITIALIZED when this is called.
165  *
166  * Set m_shutdown = 1 on any fatal errors here.
167  */
168  virtual int appStartup() = 0;
169 
170  /// This is where derived applications implement their main FSM logic.
171  /** This will be called every loopPause nanoseconds until the application terminates.
172  *
173  * FSM state will be whatever it is on exti from appStartup.
174  *
175  * Should return -1 on an any unrecoverable errors which will caues app to terminate. Could also set m_shutdown=1.
176  * Return 0 on success, or at least intent to continue.
177  *
178  */
179  virtual int appLogic() = 0;
180 
181  /// Any tasks to perform after main loop exit go here.
182  /** Should be able to handle case where appStartup and/or appLogic have not run.
183  */
184  virtual int appShutdown() = 0;
185 
186  ///@} -- Pure Virtual Functions
187 
188  /** \name Logging
189  * @{
190  */
191 public:
192  static logManagerT m_log;
193 
194  /// Make a log entry
195  /** Wrapper for logManager::log
196  *
197  * \tparam logT the log entry type
198  * \tparam retval the value returned by this method.
199  *
200  */
201  template<typename logT, int retval=0>
202  static int log( const typename logT::messageT & msg, ///< [in] the message to log
203  logPrioT level = logPrio::LOG_DEFAULT ///< [in] [optional] the log level. The default is used if not specified.
204  );
205 
206  /// Make a log entry
207  /** Wrapper for logManager::log
208  *
209  * \tparam logT the log entry type
210  * \tparam retval the value returned by this method.
211  *
212  */
213  template<typename logT, int retval=0>
214  static int log( logPrioT level = logPrio::LOG_DEFAULT /**< [in] [optional] the log level. The default is used if not specified.*/);
215 
216  ///@} -- logging
217 
218  /** \name Signal Handling
219  * @{
220  */
221 private:
222 
223  static MagAOXApp * m_self; ///< Static pointer to this (set in constructor). Used to test whether a a MagAOXApp is already instatiated (a fatal error) and used for getting out of static signal handlers.
224 
225  ///Sets the handler for SIGTERM, SIGQUIT, and SIGINT.
226  int setSigTermHandler();
227 
228  ///The handler called when SIGTERM, SIGQUIT, or SIGINT is received. Just a wrapper for handlerSigTerm.
229  static void _handlerSigTerm( int signum, ///< [in] specifies the signal.
230  siginfo_t *siginf, ///< [in] ignored by MagAOXApp
231  void *ucont ///< [in] ignored by MagAOXApp
232  );
233 
234  ///Handles SIGTERM, SIGQUIT, and SIGINT. Sets m_shutdown to 1 and logs the signal.
235  void handlerSigTerm( int signum, ///< [in] specifies the signal.
236  siginfo_t *siginf, ///< [in] ignored by MagAOXApp
237  void *ucont ///< [in] ignored by MagAOXApp
238  );
239 
240  ///@} -- Signal Handling
241 
242  /** \name Privilege Management
243  * @{
244  */
245 private:
246  uid_t m_euidReal; ///< The real user id of the proces (i.e. the lower privileged id of the user)
247  uid_t m_euidCalled; ///< The user id of the process as called (i.e. the higher privileged id of the owner, root if setuid).
248  uid_t m_suid; ///< The save-set user id of the process
249 
250 protected:
251 
252  /// Set the effective user ID to the called value, i.e. the highest possible.
253  /** If setuid is set on the file, this will be super-user privileges.
254  *
255  * Reference: http://pubs.opengroup.org/onlinepubs/009695399/functions/seteuid.html
256  *
257  * \returns 0 on success
258  * \returns -1 on error from setuid().
259  */
260  int euidCalled();
261 
262  /// Set the effective user ID to the real value, i.e. the file owner.
263  /**
264  * Reference: http://pubs.opengroup.org/onlinepubs/009695399/functions/seteuid.html
265  *
266  * \returns 0 on success
267  * \returns -1 on error from setuid().
268  */
269  int euidReal();
270 
271 
272  ///@} -- Privilege Management
273 
274  /** \name RT Priority
275  * @{
276  */
277 private:
278  int m_RTPriority {0}; ///< The real-time scheduling priority. Default is 0.
279 
280 protected:
281  /// Set the real-time priority of this process.
282  /** This method attempts to set euid to 'called' with \ref euidCalled. It then sets the priority
283  * but will fail if it does not have sufficient privileges. Regardless, it will then restore
284  * privileges with \ref euidReal.
285  *
286  * If prio < 0, it is changed to 0. If prio is > 99, then it is changed to 99.
287  *
288  * \returns 0 on success.
289  * \returns -1 on an error. In this case priority will not have been changed.
290  */
291  int RTPriority( int prio /**< [in] the desired new RT priority */ );
292 
293  ///@} -- RT Priority
294 
295  /** \name PID Locking
296  *
297  * Each MagAOXApp has a PID lock file in the system directory. The app will not
298  * startup if it detects that the PID is already locked, preventing duplicates. This is
299  * based on the configured name, not the invoked name (argv[0]).
300  *
301  * @{
302  */
303 
304  std::string pidFileName; ///<The name of the PID file
305 
306  pid_t m_pid {0}; ///< This process's PID
307 
308  /// Attempt to lock the PID by writing it to a file. Fails if a process is already running with the same config name.
309  /** First checks the PID file for an existing PID. If found, interrogates /proc to determine if that process is
310  * running and if so if the command line matches. If a matching process is currently running, then this returns an error.
311  *
312  * Will not fail if a PID file exists but the stored PID does not correspond to a running process with the same command line name.
313  *
314  * Reference: https://linux.die.net/man/3/getpid
315  *
316  * \returns 0 on success.
317  * \returns -1 on any error, including creating the PID file or if this app is already running.
318  */
319  int lockPID();
320 
321  /// Remove the PID file.
322  int unlockPID();
323 
324  ///@} -- PID Locking
325 
326  /** \name Application State
327  *
328  * @{
329  */
330 private:
331  stateCodes::stateCodeT m_state {stateCodes::UNINITIALIZED}; ///< The application's state. Never ever set this directly, use state(const stateCodeT & s).
332 
333  int m_stateLogged {0} ;///< Counter and flag for use to log errors just once. Never ever access directly, use stateLogged().
334 
335 public:
336  /// Get the current state code
337  /** \returns m_state
338  */
339  stateCodes::stateCodeT state();
340 
341  /// Set the current state code
342  /** If no change, returns immediately with no actions.
343  *
344  * If it is a change, the state change is logged. Also resets m_stateLogged to 0.
345  */
346  void state(const stateCodes::stateCodeT & s /**< [in] The new application state */);
347 
348  /// Updates and returns the value of m_stateLogged. Will be 0 on first call after a state change, >0 afterwards.
349  /** This method exists to facilitate logging the reason for a state change once, but not
350  * logging it on subsequent event loops. Returns the current value upon entry, but updates
351  * before returning so that the next call returns the incremented value. Example usage:
352  * \code
353  if( connection_failed ) //some condition set this to true
354  {
355  state( stateCodes::NOTCONNECTED );
356  if(!stateLogged()) log<text_log>("Not connected");
357  }
358  \endcode
359  * In this example, the log entry is made the first time the state changes. If there are no changes to a
360  * different state in the mean time, then when the event loop gets here again and decides it is not connected,
361  * the log entry will not be made.
362  *
363  * \returns current value of m_stateLogged, that is the value before it is incremented.
364  */
365  int stateLogged();
366 
367  ///@} --Application State
368 
369  /** \name INDI Interface
370  *
371  * For reference: "Get" and "New" refer to properties we own. "Set" refers to properties owned by others.
372  * So we respond to GetProperties by listing our own properties, and NewProperty is a request to change
373  * a property we own. Whereas SetProperty is a notification that someone else has changed a property.
374  *
375  * @{
376  */
377 protected:
378 
379  ///Flag controlling whether INDI is used. If false, then no INDI code ipRecv.getName()executes.
380  constexpr static bool m_useINDI = _useINDI;
381 
382  ///The INDI driver wrapper. Constructed and initialized by execute, which starts and stops communications.
383  indiDriver<MagAOXApp> * m_indiDriver {nullptr};
384 
385  ///Mutex for locking INDI communications.
386  std::mutex m_indiMutex;
387 
388  ///Structure to hold the call-back details for handling INDI communications.
390  {
391  pcf::IndiProperty * property {0}; ///< A pointer to an INDI property.
392  int (*callBack)( void *, const pcf::IndiProperty &) {0}; ///< The function to call for a new or set property.
393  bool m_defReceived {false}; ///< Flag indicating that a DefProperty has been received after a GetProperty.
394  };
395 
396  ///Map to hold the NewProperty indiCallBacks for this App, with fast lookup by property name.
397  /** The key for these is the property name.
398  */
399  std::unordered_map< std::string, indiCallBack> m_indiNewCallBacks;
400 
401  ///Map to hold the SetProperty indiCallBacks for this App, with fast lookup by property name.
402  /** The key for these is device.name
403  */
404  std::unordered_map< std::string, indiCallBack> m_indiSetCallBacks;
405 
406  ///Flat indicating that all registered Set properties have been updated since last Get.
407  bool m_allDefsReceived {false};
408 
409  ///Value type of the indiCallBack map.
410  typedef std::pair<std::string, indiCallBack> callBackValueType;
411 
412  ///Iterator type of the indiCallBack map.
413  typedef typename std::unordered_map<std::string, indiCallBack>::iterator callBackIterator;
414 
415  ///Return type of insert on the indiCallBack map.
416  typedef std::pair<callBackIterator,bool> callBackInsertResult;
417 
418  ///Full path name of the INDI driver input FIFO.
419  std::string m_driverInName;
420 
421  ///Full path name of the INDI driver output FIFO.
422  std::string m_driverOutName;
423 
424  ///Full path name of the INDI driver control FIFO.
425  /** This is currently only used to signal restarts.
426  */
427  std::string m_driverCtrlName;
428 
429  /// Register an INDI property which is exposed for others to request a New Property for.
430  /**
431  *
432  * \returns 0 on success.
433  * \returns -1 on error.
434  *
435  * \todo needs error logging
436  * \todo needs exception handling
437  * \todo is a failure to register a FATAL error?
438  */
439  int registerIndiPropertyNew( pcf::IndiProperty & prop, ///< [out] the property to register
440  const std::string & propName, ///< [in] the name of the property
441  const pcf::IndiProperty::Type & propType, ///< [in] the type of the property
442  const pcf::IndiProperty::PropertyPermType & propPerm, ///< [in] the permissions of the property
443  const pcf::IndiProperty::PropertyStateType & propState, ///< [in] the state of the property
444  int (*)( void *, const pcf::IndiProperty &) ///< [in] the callback for changing the property
445  );
446 
447  /// Register an INDI property which is monitored for updates from others.
448  /**
449  *
450  * \returns 0 on success.
451  * \returns -1 on error.
452  *
453  * \todo needs error logging
454  * \todo needs exception handling
455  * \todo is a failure to register a FATAL error?
456  */
457  int registerIndiPropertySet( pcf::IndiProperty & prop, ///< [out] the property to register
458  const std::string & devName, ///< [in] the device which owns this property
459  const std::string & propName, ///< [in] the name of the property
460  int (*)( void *, const pcf::IndiProperty &) ///< [in] the callback for processing the property change
461  );
462 
463  /// Create the INDI FIFOs
464  /** Changes permissions to max available and creates the
465  * FIFOs at the configured path.
466  */
467  int createINDIFIFOS();
468 
469  /// Start INDI Communications
470  /**
471  * \returns 0 on success
472  * \returns -1 on error. This is fatal.
473  */
474  int startINDI();
475 
476 public:
477 
478  void sendGetPropertySetList(bool all=false);
479 
480  /// Handler for the DEF INDI properties notification
481  /** Uses the properties registered in m_indiSetCallBacks to process the notification. This is called by
482  * m_indiDriver's indiDriver::handleDefProperties.
483  */
484  void handleDefProperty( const pcf::IndiProperty &ipRecv /**< [in] The property being sent. */ );
485 
486  /// Handler for the get INDI properties request
487  /** Uses the properties registered in m_indiCallBacks to respond to the request. This is called by
488  * m_indiDriver's indiDriver::handleGetProperties.
489  */
490  void handleGetProperties( const pcf::IndiProperty &ipRecv /**< [in] The property being requested. */ );
491 
492  /// Handler for the new INDI property request
493  /** Uses the properties registered in m_indiCallBacks to respond to the request, looking up the callback for this
494  * property and calling it.
495  *
496  * This is called by m_indiDriver's indiDriver::handleGetProperties.
497  *
498  * \todo handle errors, are they FATAL?
499  */
500  void handleNewProperty( const pcf::IndiProperty &ipRecv /**< [in] The property being changed. */);
501 
502  /// Handler for the set INDI property request
503  /**
504  *
505  * This is called by m_indiDriver's indiDriver::handleSetProperties.
506  *
507  * \todo handle errors, are they FATAL?
508  */
509  void handleSetProperty( const pcf::IndiProperty &ipRecv /**< [in] The property being changed. */);
510 
511 protected:
512 
513  /// Update an INDI property element value if it has changed.
514  /** Will only peform a SetProperty if the new element value has changed
515  * compared to the stored value. This comparison is done in the true
516  * type of the value.
517  */
518  template<typename T>
519  void updateIfChanged( pcf::IndiProperty & p, ///< [in/out] The property containing the element to possibly update
520  const std::string & el, ///< [in] The element name
521  const T & newVal ///< [in] the new value
522  );
523 
524  /// Send a newProperty command to another device (using the INDI Client interface)
525  /** Copies the input IndiProperty, then updates the element with the new value.
526  *
527  * \returns 0 on success.
528  * \returns -1 on an errory.
529  */
530  template<typename T>
531  int sendNewProperty( const pcf::IndiProperty & ipSend, ///< [in] The property to send a "new" INDI command for
532  const std::string & el, ///< [in] The element of the property to change
533  const T & newVal ///< [in] The value to request for the element.
534  );
535 
536  ///indi Property to report the application state.
537  pcf::IndiProperty m_indiP_state;
538 
539  ///@} --INDI Interface
540 
541  /** \name Power Management
542  * For devices which have remote power management (e.g. from one of the PDUs) we implement
543  * a standard power state monitoring and management component for the FSM. This is only
544  * enabled if the power management configuration options are set.
545  *
546  * If power management is enabled, then while power is off, appLogic will not be called.
547  * Instead a parrallel set of virtual functions is called, onPowerOff (to allow apps to
548  * perform cleanup) and whilePowerOff (to allow apps to keep variables updated, etc).
549  * Note that these could merely call appLogic if desired.
550  *
551  */
552 protected:
553  bool m_powerMgtEnabled {false};
554 
555  std::string m_powerDevice;
556  std::string m_powerOutlet;
557  std::string m_powerElement {"state"};
558 
559  int m_powerState {-1}; ///< Current power state, 1=On, 0=Off, -1=Unk.
560 
561  pcf::IndiProperty m_indiP_powerOutlet;
562 
563  /// This method is called when the change to poweroff is detected.
564  /**
565  * \returns 0 on success.
566  * \returns -1 on any error which means the app should exit.
567  */
568  virtual int onPowerOff() {return 0;}
569 
570  /// This method is called while the power is off, once per FSM loop.
571  /**
572  * \returns 0 on success.
573  * \returns -1 on any error which means the app should exit.
574  */
575  virtual int whilePowerOff() {return 0;}
576 
577 public:
578 
579  INDI_SETCALLBACK_DECL(MagAOXApp, m_indiP_powerOutlet);
580 
581  ///@} Power Management
582 
583 public:
584 
585  /** \name Member Accessors
586  *
587  * @{
588  */
589 
590  ///Get the config name
591  /**
592  * \returns the current value of m_configName
593  */
594  std::string configName();
595 
596  ///Get the INDI input FIFO file name
597  /**
598  * \returns the current value of m_driverInName
599  */
600  std::string driverInName();
601 
602  ///Get the INDI output FIFO file name
603  /**
604  * \returns the current value of m_driverOutName
605  */
606  std::string driverOutName();
607 
608  ///Get the INDI control FIFO file name
609  /**
610  * \returns the current value of m_driverCtrlName
611  */
612  std::string driverCtrlName();
613 
614  ///@} --Member Accessors
615 };
616 
617 //Set self pointer to null so app starts up uninitialized.
618 template<bool _useINDI> MagAOXApp<_useINDI> * MagAOXApp<_useINDI>::m_self = nullptr;
619 
620 //Define the logger
621 template<bool _useINDI> typename MagAOXApp<_useINDI>::logManagerT MagAOXApp<_useINDI>::m_log;
622 
623 template<bool _useINDI>
624 MagAOXApp<_useINDI>::MagAOXApp( const std::string & git_sha1,
625  const bool git_modified
626  )
627 {
628  if( m_self != nullptr )
629  {
630  std::cerr << "Attempt to instantiate 2nd MagAOXApp. Exiting immediately.\n";
631  exit(-1);
632  }
633 
634  m_self = this;
635 
636  //We log the current GIT status.
638  if(git_modified) gl = logPrio::LOG_WARNING;
639  log<git_state>(git_state::messageT("MagAOX", git_sha1, git_modified), gl);
640 
641  gl = logPrio::LOG_INFO;
642  if(MXLIB_UNCOMP_REPO_MODIFIED) gl = logPrio::LOG_WARNING;
643  log<git_state>(git_state::messageT("mxlib", MXLIB_UNCOMP_CURRENT_SHA1, MXLIB_UNCOMP_REPO_MODIFIED), gl);
644 
645  //Get the uids of this process.
646  getresuid(&m_euidReal, &m_euidCalled, &m_suid);
647  euidReal(); //immediately step down to unpriveleged uid.
648 
649 }
650 
651 template<bool _useINDI>
653 {
654  if(m_indiDriver) delete m_indiDriver;
655 
656  MagAOXApp<_useINDI>::m_self = nullptr;
657 }
658 
659 template<bool _useINDI>
661  char ** argv
662  ) //virtual
663 {
664  std::string tmpstr;
665  std::string configDir;
666 
667  tmpstr = mx::getEnv(MAGAOX_env_path);
668  if(tmpstr != "")
669  {
670  MagAOXPath = tmpstr;
671  }
672  else
673  {
674  MagAOXPath = MAGAOX_path;
675  }
676 
677  //Set the config path relative to MagAOXPath
678  tmpstr = mx::getEnv(MAGAOX_env_config);
679  if(tmpstr == "")
680  {
681  tmpstr = MAGAOX_configRelPath;
682  }
683  configDir = MagAOXPath + "/" + tmpstr;
684  configPathGlobal = configDir + "/magaox.conf";
685 
686  //Setup default log path
687  tmpstr = MagAOXPath + "/" + MAGAOX_logRelPath;
688 
689  m_log.logPath(tmpstr);
690 
691  //Setup default sys path
692  tmpstr = MagAOXPath + "/" + MAGAOX_sysRelPath;
693  sysPath = tmpstr;
694 
695  //Setup default secrets path
696  tmpstr = MagAOXPath + "/" + MAGAOX_secretsRelPath;
697  secretsPath = tmpstr;
698 
699 
700  #ifdef MAGAOX_configBase
701  //We use mx::application's configPathUser for this components base config file
702  configPathUser = configDir + "/" + MAGAOX_configBase;
703  #endif
704 
705  //Parse CL just to get the "name".
706  config.add("name","n", "name",mx::argType::Required, "", "name", false, "string", "The name of the application, specifies config.");
707 
708  config.parseCommandLine(argc, argv, "name");
709  config(m_configName, "name");
710 
711  if(m_configName == "")
712  {
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].");
716  }
717 
718  //We use mx::application's configPathLocal for this component's config file
719  configPathLocal = configDir + "/" + m_configName + ".conf";
720 
721  //Now we can setup common INDI properties
722  REG_INDI_NEWPROP_NOCB(m_indiP_state, "state", pcf::IndiProperty::Number);
723  m_indiP_state.add (pcf::IndiElement("current"));
724 
725 
726  return;
727 
728 }
729 
730 template<bool _useINDI>
732 {
733  //App stuff
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)");
736 
737  //Logger Stuff
738  m_log.setupConfig(config);
739 
740  //Power Management
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.");
744 }
745 
746 template<bool _useINDI>
748 {
749  //---------- Setup the logger ----------//
750  m_log.logName(m_configName);
751  m_log.loadConfig(config);
752 
753  //--------- Loop Pause Time --------//
754  config(loopPause, "loopPause");
755 
756  //--------- RT Priority ------------//
757  int prio = m_RTPriority;
758  config(prio, "RTPriority");
759  if(prio != m_RTPriority)
760  {
761  RTPriority(prio);
762  }
763 
764  //--------Power Management --------//
765  config(m_powerDevice, "power.device");
766  config(m_powerOutlet, "power.outlet");
767  config(m_powerElement, "power.element");
768 
769  if(m_powerDevice != "" && m_powerOutlet != "")
770  {
771  log<text_log>("enabling power management: " + m_powerDevice + "." + m_powerOutlet + "." + m_powerElement);
772 
773  m_powerMgtEnabled = true;
774  REG_INDI_SETPROP(m_indiP_powerOutlet, m_powerDevice, m_powerOutlet);
775  }
776 }
777 
778 
779 
780 template<bool _useINDI>
782 {
783  if( lockPID() < 0 )
784  {
785  state(stateCodes::FAILURE);
786  log<text_log>({"Failed to lock PID."}, logPrio::LOG_CRITICAL);
787  //Return immediately, not safe to go on.
788  return -1;
789  }
790 
791  //----------------------------------------//
792  // Begin the logger
793  //----------------------------------------//
794  m_log.logThreadStart();
795 
796  //Give up to 2 secs to make sure log thread has time to get started and try to open a file.
797  for(int w=0;w<4;++w)
798  {
799  //Sleep for 500 msec
800  std::this_thread::sleep_for( std::chrono::duration<unsigned long, std::nano>(500000));
801 
802  //Verify that log thread is still running.
803  if(m_log.logThreadRunning() == true) break;
804  }
805 
806  if(m_log.logThreadRunning() == false)
807  {
808  //We don't log this, because it won't be logged anyway.
809  std::cerr << "\nCRITICAL: log thread not running. Exiting.\n\n";
810  m_shutdown = 1;
811  }
812 
813  //----------------------------------------//
814 
815  setSigTermHandler();
816 
817  if( m_shutdown == 0 )
818  {
819  state(stateCodes::INITIALIZED);
820  if(appStartup() < 0) m_shutdown = 1;
821  }
822 
823  //====Begin INDI Communications
824  if(m_useINDI && m_shutdown == 0) //if we're using INDI and not already dead, that is
825  {
826  if(startINDI() < 0)
827  {
828  state(stateCodes::FAILURE);
829  m_shutdown = 1;
830  }
831  }
832 
833  //We have to wait for power status to become available
834  if(m_powerMgtEnabled)
835  {
836  while(m_powerState < 0)
837  {
838  sleep(1);
839  if(m_powerState < 0)
840  {
841  if(!stateLogged()) log<text_log>("waiting for power state");
842  }
843  }
844  if(m_powerState > 0) state(stateCodes::POWERON);
845  else state(stateCodes::POWEROFF);
846  }
847 
848  //This is the main event loop.
849  /* Conditions on entry:
850  * -- PID locked
851  * -- Log thread running
852  * -- Signal handling installed
853  * -- appStartup() successful
854  * -- INDI communications started successfully (if being used)
855  * -- power state known (if being managed)
856  */
857  while( m_shutdown == 0)
858  {
859  if(m_powerMgtEnabled)
860  {
861  if(state() == stateCodes::POWEROFF)
862  {
863  if(m_powerState == 1)
864  {
865  state(stateCodes::POWERON);
866  }
867  }
868  else //Any other state
869  {
870  if(m_powerState == 0)
871  {
872  state(stateCodes::POWEROFF);
873  if(onPowerOff() < 0)
874  {
875  m_shutdown = 1;
876  continue;
877  }
878  }
879  //We don't do anything if m_powerState is -1, which is a startup condition.
880  }
881  }
882 
883  //Only run appLogic if power is on, or we are not managing power.
884  if( !m_powerMgtEnabled || m_powerState > 0 )
885  {
886  if( appLogic() < 0)
887  {
888  m_shutdown = 1;
889  continue;
890  }
891  }
892  else if(m_powerState == 0)
893  {
894  if( whilePowerOff() < 0)
895  {
896  m_shutdown = 1;
897  continue;
898  }
899  }
900 
901  /** \todo Need a heartbeat update here.
902  */
903 
904  if(m_useINDI)
905  {
906  //Checkup on the INDI properties we're monitoring.
907  //This will make sure we are up-to-date if indiserver restarts without us.
908  //And handles cases where we miss a Def becuase the other driver wasn't started up
909  //when we sent our Get.
910  sendGetPropertySetList(false); //Only does anything if it needs to be done.
911  }
912 
913  // This is purely to make sure INDI is up to date in case
914  // mutex was locked on last attempt.
915  state( state() );
916 
917  //Pause loop unless shutdown is set
918  if( m_shutdown == 0)
919  {
920  std::this_thread::sleep_for( std::chrono::duration<unsigned long, std::nano>(loopPause));
921  }
922  }
923 
924  appShutdown();
925 
926  state(stateCodes::SHUTDOWN);
927 
928  //Stop INDI communications
929  if(m_indiDriver != nullptr)
930  {
931  m_indiDriver->quitProcess();
932  m_indiDriver->deactivate();
933  log<indidriver_stop>();
934  }
935 
936  unlockPID();
937 
938  return 0;
939 }
940 
941 template<bool _useINDI>
942 template<typename logT, int retval>
943 int MagAOXApp<_useINDI>::log( const typename logT::messageT & msg,
944  logPrioT level
945  )
946 {
947  m_log.log<logT>(msg, level);
948  return retval;
949 }
950 
951 template<bool _useINDI>
952 template<typename logT, int retval>
954 {
955  m_log.log<logT>(level);
956  return retval;
957 }
958 
959 template<bool _useINDI>
961 {
962  struct sigaction act;
963  sigset_t set;
964 
965  act.sa_sigaction = &MagAOXApp<_useINDI>::_handlerSigTerm;
966  act.sa_flags = SA_SIGINFO;
967  sigemptyset(&set);
968  act.sa_mask = set;
969 
970  errno = 0;
971  if( sigaction(SIGTERM, &act, 0) < 0 )
972  {
973  std::string logss = "Setting handler for SIGTERM failed. Errno says: ";
974  logss += strerror(errno);
975 
976  log<software_error>({__FILE__, __LINE__, errno, 0, logss});
977 
978  return -1;
979  }
980 
981  errno = 0;
982  if( sigaction(SIGQUIT, &act, 0) < 0 )
983  {
984  std::string logss = "Setting handler for SIGQUIT failed. Errno says: ";
985  logss += strerror(errno);
986 
987  log<software_error>({__FILE__, __LINE__, errno, 0,logss});
988 
989  return -1;
990  }
991 
992  errno = 0;
993  if( sigaction(SIGINT, &act, 0) < 0 )
994  {
995  std::string logss = "Setting handler for SIGINT failed. Errno says: ";
996  logss += strerror(errno);
997 
998  log<software_error>({__FILE__, __LINE__, errno, 0, logss});
999 
1000  return -1;
1001  }
1002 
1003  log<text_log>("Installed SIGTERM/SIGQUIT/SIGINT signal handler.", logPrio::LOG_DEBUG);
1004 
1005  return 0;
1006 }
1007 
1008 template<bool _useINDI>
1010  siginfo_t *siginf,
1011  void *ucont
1012  )
1013 {
1014  m_self->handlerSigTerm(signum, siginf, ucont);
1015 }
1016 
1017 template<bool _useINDI>
1019  siginfo_t *siginf __attribute__((unused)),
1020  void *ucont __attribute__((unused))
1021  )
1022 {
1023  m_shutdown = 1;
1024 
1025  std::string signame;
1026  switch(signum)
1027  {
1028  case SIGTERM:
1029  signame = "SIGTERM";
1030  break;
1031  case SIGINT:
1032  signame = "SIGINT";
1033  break;
1034  case SIGQUIT:
1035  signame = "SIGQUIT";
1036  break;
1037  default:
1038  signame = "OTHER";
1039  }
1040 
1041  std::string logss = "Caught signal ";
1042  logss += signame;
1043  logss += ". Shutting down.";
1044 
1045  std::cerr << "\n" << logss << std::endl;
1046  log<text_log>(logss);
1047 }
1048 
1049 template<bool _useINDI>
1051 {
1052  errno = 0;
1053  if(seteuid(m_euidCalled) < 0)
1054  {
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);
1059 
1060  log<software_error>({__FILE__, __LINE__, errno, 0, logss});
1061 
1062  return -1;
1063  }
1064 
1065  return 0;
1066 }
1067 
1068 template<bool _useINDI>
1070 {
1071  errno = 0;
1072  if(seteuid(m_euidReal) < 0)
1073  {
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);
1078 
1079  log<software_error>({__FILE__, __LINE__, errno, 0, logss});
1080 
1081  return -1;
1082  }
1083 
1084  return 0;
1085 
1086 }
1087 
1088 template<bool _useINDI>
1090 {
1091  struct sched_param schedpar;
1092 
1093  if(prio < 0) prio = 0;
1094  if(prio > 99) prio = 99;
1095  schedpar.sched_priority = prio;
1096 
1097  //Get the maximum privileges available
1098  if( euidCalled() < 0 )
1099  {
1100  log<software_error>({__FILE__, __LINE__, 0, 0,"Seeting euid to called failed."});
1101  return -1;
1102  }
1103 
1104 
1105  //We set return value based on result from sched_setscheduler
1106  //But we make sure to restore privileges no matter what happens.
1107  errno = 0;
1108  int rv = 0;
1109  if(prio > 0) rv = sched_setscheduler(0, MAGAOX_RT_SCHED_POLICY, &schedpar);
1110  else rv = sched_setscheduler(0, SCHED_OTHER, &schedpar);
1111 
1112  if(rv < 0)
1113  {
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()});
1117  }
1118  else
1119  {
1120  m_RTPriority = prio;
1121 
1122  std::stringstream logss;
1123  logss << "Scheduler priority (RT_priority) set to " << m_RTPriority << ".";
1124  log<text_log>(logss.str());
1125  }
1126 
1127  //Go back to regular privileges
1128  if( euidReal() < 0 )
1129  {
1130  log<software_error>({__FILE__, __LINE__, 0, 0, "Setting euid to real failed."});
1131  return -1;
1132  }
1133 
1134  return rv;
1135 }
1136 
1137 template<bool _useINDI>
1139 {
1140  m_pid = getpid();
1141 
1142  std::string statusDir = sysPath;
1143 
1144  //Get the maximum privileges available
1145  if( euidCalled() < 0 )
1146  {
1147  log<software_error>({__FILE__, __LINE__, 0, 0, "Seeting euid to called failed."});
1148  return -1;
1149  }
1150 
1151  // Create statusDir root with read/write/search permissions for owner and group, and with read/search permissions for others.
1152  errno = 0;
1153  if( mkdir(statusDir.c_str(), S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH) < 0 )
1154  {
1155  if( errno != EEXIST)
1156  {
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()});
1160 
1161  //Go back to regular privileges
1162  euidReal();
1163 
1164  return -1;
1165  }
1166 
1167  }
1168 
1169  statusDir += "/";
1170  statusDir += m_configName;
1171 
1172  pidFileName = statusDir + "/pid";
1173 
1174  // Create statusDir with read/write/search permissions for owner and group, and with read/search permissions for others.
1175  errno = 0;
1176  if( mkdir(statusDir.c_str(), S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH) < 0 )
1177  {
1178  if( errno != EEXIST)
1179  {
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()});
1183 
1184  //Go back to regular privileges
1185  euidReal();
1186 
1187  return -1;
1188  }
1189 
1190  //If here, then we need to check the pid file.
1191 
1192  std::ifstream pidIn;
1193  pidIn.open( pidFileName );
1194 
1195  if(pidIn.good()) //PID file exists, now read its contents and compare to proc/<pid>/cmdline
1196  {
1197  //Read PID from file
1198  pid_t testPid;
1199  pidIn >> testPid;
1200  pidIn.close();
1201 
1202  //Get command line used to start this process from /proc
1203  std::stringstream procN;
1204  procN << "/proc/" << testPid << "/cmdline";
1205 
1206  std::ifstream procIn;
1207  std::string pidCmdLine;
1208 
1209  try
1210  {
1211  procIn.open(procN.str());
1212  if(procIn.good()) procIn >> pidCmdLine;
1213  procIn.close();
1214  }
1215  catch( ... )
1216  {
1217  log<software_critical>({__FILE__, __LINE__, 0, 0, "exception caught testing /proc/pid"});
1218  euidReal();
1219  return -1;
1220  }
1221 
1222  //If pidCmdLine == "" at this point we just allow the rest of the
1223  //logic to run...
1224 
1225  //Search for invokedName in command line.
1226  size_t invokedPos = pidCmdLine.find( invokedName );
1227 
1228  //If invokedName found, then we check for configName.
1229  size_t configPos = std::string::npos;
1230  if(invokedPos != std::string::npos) configPos = pidCmdLine.find( m_configName );
1231 
1232  //Check if PID is already locked by this program+config combo:
1233  if( invokedPos != std::string::npos && configPos != std::string::npos)
1234  {
1235  //This means that this app already exists for this config, and we need to die.
1236  std::stringstream logss;
1237  logss << "PID already locked (" << testPid << "). Time to die.";
1238  std::cerr << logss.str() << std::endl;
1239 
1240  log<text_log>(logss.str(), logPrio::LOG_CRITICAL);
1241 
1242  //Go back to regular privileges
1243  euidReal();
1244 
1245  return -1;
1246  }
1247  }
1248  else
1249  {
1250  //No PID File so we should just go on.
1251  pidIn.close();
1252  }
1253  }
1254 
1255  //Now write current PID to file and go on with life.
1256  std::ofstream pidOut;
1257  pidOut.open(pidFileName);
1258 
1259  if(!pidOut.good())
1260  {
1261  log<software_critical>({__FILE__, __LINE__, errno, 0, "could not open pid file for writing."});
1262  euidReal();
1263  return -1;
1264  }
1265 
1266  pidOut << m_pid;
1267 
1268  pidOut.close();
1269 
1270  std::stringstream logss;
1271  logss << "PID (" << m_pid << ") locked.";
1272  log<text_log>(logss.str());
1273 
1274  //Go back to regular privileges
1275  if( euidReal() < 0 )
1276  {
1277  log<software_error>({__FILE__, __LINE__, 0, 0, "Seeting euid to real failed."});
1278  return -1;
1279  }
1280 
1281  return 0;
1282 }
1283 
1284 template<bool _useINDI>
1286 {
1287  if( ::remove(pidFileName.c_str()) < 0)
1288  {
1289  log<software_error>({__FILE__, __LINE__, errno, 0, std::string("Failed to remove PID file: ") + strerror(errno)});
1290  return -1;
1291  }
1292 
1293  std::stringstream logss;
1294  logss << "PID (" << m_pid << ") unlocked.";
1295  log<text_log>(logss.str());
1296 
1297  return 0;
1298 }
1299 
1300 template<bool _useINDI>
1302 {
1303  return m_state;
1304 }
1305 
1306 template<bool _useINDI>
1308 {
1309  //Only do anything if it's a change
1310  if(m_state != s)
1311  {
1313  if(s == stateCodes::ERROR) lvl = logPrio::LOG_ERROR;
1314  if(s == stateCodes::FAILURE) lvl = logPrio::LOG_CRITICAL;
1315 
1316  log<state_change>( {m_state, s}, lvl );
1317 
1318  m_state = s;
1319  m_stateLogged = 0;
1320  }
1321 
1322  //Check to make sure INDI is up to date
1323  std::unique_lock<std::mutex> lock(m_indiMutex, std::try_to_lock); //Lock the mutex before conducting INDI communications.
1324 
1325  if(lock.owns_lock())
1326  {
1327  updateIfChanged(m_indiP_state, "current", m_state);
1328  }
1329 }
1330 
1331 template<bool _useINDI>
1333 {
1334  if(m_stateLogged > 0)
1335  {
1336  ++m_stateLogged;
1337  return m_stateLogged - 1;
1338  }
1339  else
1340  {
1341  m_stateLogged = 1;
1342  return 0;
1343  }
1344 }
1345 
1346 /*-------------------------------------------------------------------------------------*/
1347 /* INDI Support */
1348 /*-------------------------------------------------------------------------------------*/
1349 
1350 template<bool _useINDI>
1351 int MagAOXApp<_useINDI>::registerIndiPropertyNew( pcf::IndiProperty & prop,
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)
1357  )
1358 {
1359  if(!m_useINDI) return 0;
1360 
1361  prop = pcf::IndiProperty (propType);
1362  prop.setDevice(m_configName);
1363  prop.setName(propName);
1364  prop.setPerm(propPerm);
1365  prop.setState( propState);
1366 
1367 
1368  callBackInsertResult result = m_indiNewCallBacks.insert(callBackValueType( propName, {&prop, callBack}));
1369 
1370  if(!result.second)
1371  {
1372  return -1;
1373  }
1374 
1375  return 0;
1376 }
1377 
1378 template<bool _useINDI>
1379 int MagAOXApp<_useINDI>::registerIndiPropertySet( pcf::IndiProperty & prop,
1380  const std::string & devName,
1381  const std::string & propName,
1382  int (*callBack)( void *, const pcf::IndiProperty &ipRecv)
1383  )
1384 {
1385  if(!m_useINDI) return 0;
1386 
1387  prop = pcf::IndiProperty();
1388  prop.setDevice(devName);
1389  prop.setName(propName);
1390 
1391  callBackInsertResult result = m_indiSetCallBacks.insert(callBackValueType( devName + "." + propName, {&prop, callBack}));
1392 
1393  if(!result.second)
1394  {
1395  return -1;
1396  }
1397 
1398  return 0;
1399 }
1400 
1401 template<bool _useINDI>
1403 {
1404  if(!m_useINDI) return 0;
1405 
1406  ///\todo make driver FIFO path full configurable.
1407  std::string driverFIFOPath = MAGAOX_path;
1408  driverFIFOPath += "/";
1409  driverFIFOPath += MAGAOX_driverFIFORelPath;
1410 
1411  m_driverInName = driverFIFOPath + "/" + configName() + ".in";
1412  m_driverOutName = driverFIFOPath + "/" + configName() + ".out";
1413  m_driverCtrlName = driverFIFOPath + "/" + configName() + ".ctrl";
1414 
1415  //Get max permissions
1416  euidCalled();
1417 
1418  //Clear the file mode creation mask so mkfifo does what we want. Don't forget to restore it.
1419  mode_t prev = umask(0);
1420 
1421  errno = 0;
1422  if(mkfifo(m_driverInName.c_str(), S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP) !=0)
1423  {
1424  if(errno != EEXIST)
1425  {
1426  umask(prev);
1427  euidReal();
1428  log<software_critical>({__FILE__, __LINE__, errno, 0, "mkfifo failed"});
1429  log<text_log>("Failed to create input FIFO.", logPrio::LOG_CRITICAL);
1430  return -1;
1431  }
1432  }
1433 
1434  errno = 0;
1435  if(mkfifo(m_driverOutName.c_str(), S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP) !=0 )
1436  {
1437  if(errno != EEXIST)
1438  {
1439  umask(prev);
1440  euidReal();
1441  log<software_critical>({__FILE__, __LINE__, errno, 0, "mkfifo failed"});
1442  log<text_log>("Failed to create ouput FIFO.", logPrio::LOG_CRITICAL);
1443  return -1;
1444  }
1445  }
1446 
1447  errno = 0;
1448  if(mkfifo(m_driverCtrlName.c_str(), S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP) !=0 )
1449  {
1450  if(errno != EEXIST)
1451  {
1452  umask(prev);
1453  euidReal();
1454  log<software_critical>({__FILE__, __LINE__, errno, 0, "mkfifo failed"});
1455  log<text_log>("Failed to create ouput FIFO.", logPrio::LOG_CRITICAL);
1456  return -1;
1457  }
1458  }
1459 
1460  umask(prev);
1461  euidReal();
1462  return 0;
1463 }
1464 
1465 template<bool _useINDI>
1467 {
1468  if(!m_useINDI) return 0;
1469 
1470 
1471  //===== Create the FIFOs for INDI communications ====
1472  if(createINDIFIFOS() < 0)
1473  {
1474  return -1;
1475  }
1476 
1477  //======= Instantiate the indiDriver
1478  try
1479  {
1480  m_indiDriver = new indiDriver<MagAOXApp>(this, m_configName, "0", "0");
1481  }
1482  catch(...)
1483  {
1484  log<software_critical>({__FILE__, __LINE__, 0, 0, "INDI Driver construction exception."});
1485  return -1;
1486  }
1487 
1488  //Check for INDI failure
1489  if(m_indiDriver == nullptr)
1490  {
1491  log<software_critical>({__FILE__, __LINE__, 0, 0, "INDI Driver construction failed."});
1492  return -1;
1493  }
1494 
1495  //Check for INDI failure to open the FIFOs
1496  if(m_indiDriver->good() == false)
1497  {
1498  log<software_critical>({__FILE__, __LINE__, 0, 0, "INDI Driver failed to open FIFOs."});
1499  delete m_indiDriver;
1500  m_indiDriver = nullptr;
1501  return -1;
1502  }
1503 
1504  //======= Now we start talkin'
1505  m_indiDriver->activate();
1506  log<indidriver_start>();
1507 
1508  sendGetPropertySetList();
1509 
1510  return 0;
1511 }
1512 
1513 template<bool _useINDI>
1515 {
1516  //Unless forced by all, we only do anything if allDefs are not received yet
1517  if(!all && m_allDefsReceived) return;
1518 
1519  callBackIterator it = m_indiSetCallBacks.begin();
1520 
1521  int nowFalse = 0;
1522  while(it != m_indiSetCallBacks.end() )
1523  {
1524  if(all || it->second.m_defReceived == false)
1525  {
1526  if( it->second.property )
1527  {
1528  m_indiDriver->sendGetProperties( *(it->second.property) );
1529  }
1530 
1531  it->second.m_defReceived = false;
1532  ++nowFalse;
1533 
1534  }
1535  ++it;
1536  }
1537  if(nowFalse != 0) m_allDefsReceived = false;
1538  if(nowFalse == 0) m_allDefsReceived = true;
1539 }
1540 
1541 template<bool _useINDI>
1542 void MagAOXApp<_useINDI>::handleDefProperty( const pcf::IndiProperty &ipRecv )
1543 {
1544  handleSetProperty(ipRecv); //We have the same response to both Def and Set.
1545 }
1546 
1547 template<bool _useINDI>
1548 void MagAOXApp<_useINDI>::handleGetProperties( const pcf::IndiProperty &ipRecv )
1549 {
1550  if(!m_useINDI) return;
1551  if(m_indiDriver == nullptr) return;
1552 
1553  //Ignore if not our device
1554  if (ipRecv.hasValidDevice() && ipRecv.getDevice() != m_indiDriver->getName())
1555  {
1556  return;
1557  }
1558 
1559  //Send all properties if requested.
1560  if( !ipRecv.hasValidName() )
1561  {
1562  callBackIterator it = m_indiNewCallBacks.begin();
1563 
1564  while(it != m_indiNewCallBacks.end() )
1565  {
1566  if( it->second.property )
1567  {
1568  m_indiDriver->sendDefProperty( *(it->second.property) );
1569  }
1570  ++it;
1571  }
1572 
1573  //This is a possible INDI server restart, so we re-register for all notifications.
1574  sendGetPropertySetList(true);
1575 
1576  return;
1577  }
1578 
1579  //Check if we actually have this.
1580  if( m_indiNewCallBacks.count(ipRecv.getName()) == 0)
1581  {
1582  return;
1583  }
1584 
1585  //Otherwise send just the requested property, if property is not null
1586  if(m_indiNewCallBacks[ ipRecv.getName() ].property)
1587  {
1588  m_indiDriver->sendDefProperty( *(m_indiNewCallBacks[ ipRecv.getName() ].property) );
1589  }
1590  return;
1591 }
1592 
1593 template<bool _useINDI>
1594 void MagAOXApp<_useINDI>::handleNewProperty( const pcf::IndiProperty &ipRecv )
1595 {
1596  if(!m_useINDI) return;
1597  if(m_indiDriver == nullptr) return;
1598 
1599  //Check if this is a valid name for us.
1600  if( m_indiNewCallBacks.count(ipRecv.getName()) == 0 )
1601  {
1602  ///\todo log invalid NewProperty request, though it probably can't get this far.
1603  return;
1604  }
1605 
1606  int (*callBack)(void *, const pcf::IndiProperty &) = m_indiNewCallBacks[ ipRecv.getName() ].callBack;
1607 
1608  if(callBack) callBack( this, ipRecv);
1609 
1610  ///\todo log an error here because callBack should not be null
1611 
1612  return;
1613 }
1614 
1615 template<bool _useINDI>
1616 void MagAOXApp<_useINDI>::handleSetProperty( const pcf::IndiProperty &ipRecv )
1617 {
1618  if(!m_useINDI) return;
1619  if(m_indiDriver == nullptr) return;
1620 
1621  std::string key = ipRecv.getDevice() + "." + ipRecv.getName();
1622 
1623  //Check if this is valid
1624  if( m_indiSetCallBacks.count(key) > 0 )
1625  {
1626  m_indiSetCallBacks[ key ].m_defReceived = true; //record that we got this Def/Set
1627 
1628  //And call the callback
1629  int (*callBack)(void *, const pcf::IndiProperty &) = m_indiSetCallBacks[ key ].callBack;
1630  if(callBack) callBack( this, ipRecv);
1631 
1632  ///\todo log an error here because callBack should not be null
1633  }
1634  else
1635  {
1636  ///\todo log invalid SetProperty request.
1637  }
1638 
1639  return;
1640 }
1641 
1642 template<bool _useINDI>
1643 template<typename T>
1644 void MagAOXApp<_useINDI>::updateIfChanged( pcf::IndiProperty & p,
1645  const std::string & el,
1646  const T & newVal
1647  )
1648 {
1649  if(!_useINDI) return;
1650 
1651  if(!m_indiDriver) return;
1652 
1653  T oldVal = p[el].get<T>();
1654 
1655  if(oldVal != newVal)
1656  {
1657  p[el].set(newVal);
1658  p.setState (pcf::IndiProperty::Ok);
1659  m_indiDriver->sendSetProperty (p);
1660  }
1661 }
1662 
1663 /// \todo move propType to an INDI utils file, and document.
1664 
1665 template<typename T>
1666 pcf::IndiProperty::Type propType()
1667 {
1668  return pcf::IndiProperty::Unknown;
1669 }
1670 
1671 
1672 template<>
1673 inline
1674 pcf::IndiProperty::Type propType<char *>()
1675 {
1676  return pcf::IndiProperty::Text;
1677 }
1678 
1679 template<>
1680 inline
1681 pcf::IndiProperty::Type propType<std::string>()
1682 {
1683  return pcf::IndiProperty::Text;
1684 }
1685 
1686 template<>
1687 inline
1688 pcf::IndiProperty::Type propType<int>()
1689 {
1690  return pcf::IndiProperty::Number;
1691 }
1692 
1693 template<>
1694 inline
1695 pcf::IndiProperty::Type propType<double>()
1696 {
1697  return pcf::IndiProperty::Number;
1698 }
1699 
1700 template<bool _useINDI>
1701 template<typename T>
1702 int MagAOXApp<_useINDI>::sendNewProperty( const pcf::IndiProperty & ipSend,
1703  const std::string & el,
1704  const T & newVal
1705  )
1706 {
1707  if(!_useINDI) return 0;
1708 
1709  if(!m_indiDriver)
1710  {
1711  log<software_error>({__FILE__, __LINE__, "INDI communications not initialized."});
1712  return -1;
1713  }
1714  pcf::IndiProperty ipToSend = ipSend;
1715 
1716  try
1717  {
1718  ipToSend[el].setValue(newVal);
1719  }
1720  catch(...)
1721  {
1722  log<software_error>({__FILE__, __LINE__, "Exception caught setting " + ipSend.getDevice() + "." + ipSend.getName() + "." + el});
1723  return -1;
1724  }
1725 
1726  int rv = m_indiDriver->sendNewProperty(ipToSend);
1727  if(rv < 0)
1728  {
1729  log<software_error>({__FILE__, __LINE__});
1730  return -1;
1731  }
1732 
1733  return 0;
1734 }
1735 
1736 template<bool _useINDI>
1737 INDI_SETCALLBACK_DEFN( MagAOXApp<_useINDI>, m_indiP_powerOutlet)(const pcf::IndiProperty &ipRecv)
1738 {
1739  //m_indiP_powerOutlet = ipRecv;
1740 
1741  std::string ps;
1742 
1743  try
1744  {
1745  ps = ipRecv[m_powerElement].get<std::string>();
1746  }
1747  catch(...)
1748  {
1749  log<software_error>({__FILE__, __LINE__, "Exception caught."});
1750  return -1;
1751  }
1752 
1753  if(ps == "On")
1754  {
1755  m_powerState = 1;
1756  }
1757  else if (ps == "Off")
1758  {
1759  m_powerState = 0;
1760  }
1761  else
1762  {
1763  m_powerState = -1;
1764  }
1765 
1766  return 0;
1767 }
1768 
1769 template<bool _useINDI>
1771 {
1772  return m_configName;
1773 }
1774 
1775 template<bool _useINDI>
1777 {
1778  return m_driverInName;
1779 }
1780 
1781 template<bool _useINDI>
1783 {
1784  return m_driverOutName;
1785 }
1786 
1787 template<bool _useINDI>
1789 {
1790  return m_driverCtrlName;
1791 }
1792 
1793 } //namespace app
1794 } //namespace MagAOX
1795 
1796 #endif //app_MagAOXApp_hpp
std::string m_driverOutName
Full path name of the INDI driver output FIFO.
Definition: MagAOXApp.hpp:422
pcf::IndiProperty::Type propType()
Definition: MagAOXApp.hpp:1666
int logThreadStart()
Start the logger thread.
Definition: logManager.hpp:357
pcf::IndiProperty::Type propType< int >()
Definition: MagAOXApp.hpp:1688
virtual int whilePowerOff()
This method is called while the power is off, once per FSM loop.
Definition: MagAOXApp.hpp:575
static constexpr logPrioT LOG_INFO
Informational. The info log level is the lowest level recorded during normal operations.
Definition: logPriority.hpp:49
The standard MagAOX log manager, used for both process logs and telemetry streams.
Definition: logManager.hpp:68
#define MAGAOX_env_config
Environment variable setting the relative config path.
Definition: environment.hpp:25
#define MAGAOX_logRelPath
The relative path to the log directory.
Definition: paths.hpp:43
static constexpr logPrioT LOG_WARNING
A condition has occurred which may become an error, but the process continues.
Definition: logPriority.hpp:43
#define REG_INDI_SETPROP(prop, devName, propName)
Register a SET INDI property with the class, using the standard callback name.
Definition: indiMacros.hpp:173
std::unordered_map< std::string, indiCallBack > m_indiNewCallBacks
Map to hold the NewProperty indiCallBacks for this App, with fast lookup by property name...
Definition: MagAOXApp.hpp:399
void log(const typename logT::messageT &msg, logPrioT level=logPrio::LOG_DEFAULT)
Make a log entry, including a message.
Definition: logManager.hpp:475
std::string pidFileName
The name of the PID file.
Definition: MagAOXApp.hpp:304
#define REG_INDI_NEWPROP_NOCB(prop, propName, type)
Register a NEW INDI property with the class, with no callback.
Definition: indiMacros.hpp:160
std::string m_configName
The name of the configuration file (minus .conf).
Definition: MagAOXApp.hpp:80
Macros for INDI.
#define INDI_SETCALLBACK_DEFN(class, prop)
Define the callback for a set property request.
Definition: indiMacros.hpp:116
pcf::IndiProperty::Type propType< double >()
Definition: MagAOXApp.hpp:1695
std::string m_powerDevice
Definition: MagAOXApp.hpp:555
std::string m_powerOutlet
Definition: MagAOXApp.hpp:556
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)
Definition: MagAOXApp.hpp:1702
static constexpr logPrioT LOG_ERROR
An error has occured which the software will attempt to correct.
Definition: logPriority.hpp:40
static MagAOXApp * m_self
Static pointer to this (set in constructor). Used to test whether a a MagAOXApp is already instatiate...
Definition: MagAOXApp.hpp:223
std::string m_driverCtrlName
Full path name of the INDI driver control FIFO.
Definition: MagAOXApp.hpp:427
#define MAGAOX_default_loopPause
The default application loopPause.
Definition: defaults.hpp:48
The base-class for MagAO-X applications.
Definition: MagAOXApp.hpp:68
std::unordered_map< std::string, indiCallBack >::iterator callBackIterator
Iterator type of the indiCallBack map.
Definition: MagAOXApp.hpp:413
#define MAGAOX_secretsRelPath
The relative path to the secrets directory. Used for storing passwords, etc.
Definition: paths.hpp:57
std::string sysPath
The path to the system directory, for PID file, etc.
Definition: MagAOXApp.hpp:82
std::string m_driverInName
Full path name of the INDI driver input FIFO.
Definition: MagAOXApp.hpp:419
pcf::IndiProperty m_indiP_powerOutlet
Definition: MagAOXApp.hpp:561
#define MAGAOX_RT_SCHED_POLICY
The real-time scheduling policy.
Definition: config.hpp:21
virtual int onPowerOff()
This method is called when the change to poweroff is detected.
Definition: MagAOXApp.hpp:568
#define MAGAOX_driverFIFORelPath
The relative path to the INDI driver FIFOs.
Definition: paths.hpp:71
uid_t m_euidCalled
The user id of the process as called (i.e. the higher privileged id of the owner, root if setuid)...
Definition: MagAOXApp.hpp:247
std::unordered_map< std::string, indiCallBack > m_indiSetCallBacks
Map to hold the SetProperty indiCallBacks for this App, with fast lookup by property name...
Definition: MagAOXApp.hpp:404
static logManagerT m_log
Definition: MagAOXApp.hpp:192
MagAO-X INDI Driver Wrapper.
int8_t logPrioT
The type of the log priority code.
Definition: logDefs.hpp:19
std::pair< callBackIterator, bool > callBackInsertResult
Return type of insert on the indiCallBack map.
Definition: MagAOXApp.hpp:416
MagAO-X Application States.
uid_t m_euidReal
The real user id of the proces (i.e. the lower privileged id of the user)
Definition: MagAOXApp.hpp:246
#define MAGAOX_sysRelPath
The relative path to the system directory.
Definition: paths.hpp:50
int loadConfig(mx::appConfigurator &config)
Load the logger section from an application configurator.
Definition: logManager.hpp:309
std::string MagAOXPath
The base path of the MagAO-X system.
Definition: MagAOXApp.hpp:78
#define MAGAOX_configRelPath
The relative path to the configuration files.
Definition: paths.hpp:29
int16_t stateCodeT
The type of the state code.
Definition: stateCodes.hpp:28
uid_t m_suid
The save-set user id of the process.
Definition: MagAOXApp.hpp:248
#define MAGAOX_path
The path to the MagAO-X system files.
Definition: paths.hpp:22
The type of the input message.
Definition: git_state.hpp:35
#define INDI_SETCALLBACK_DECL(class, prop)
Declare the callback for a set property request, and declare and define the static wrapper...
Definition: indiMacros.hpp:57
bool logThreadRunning()
Get status of the log thread running flag.
Definition: logManager.hpp:290
static constexpr logPrioT LOG_CRITICAL
The process can not continue and will shut down (fatal)
Definition: logPriority.hpp:37
std::mutex m_indiMutex
Mutex for locking INDI communications.
Definition: MagAOXApp.hpp:386
logger::logManager< logFileRaw > logManagerT
The log manager type.
Definition: MagAOXApp.hpp:74
void handlerSigTerm(int signum, siginfo_t *siginf, void *ucont)
Handles SIGTERM, SIGQUIT, and SIGINT. Sets m_shutdown to 1 and logs the signal.
Definition: MagAOXApp.hpp:1018
Structure to hold the call-back details for handling INDI communications.
Definition: MagAOXApp.hpp:389
static constexpr logPrioT LOG_DEFAULT
Used to denote "use the default level for this log type".
Definition: logPriority.hpp:58
#define MAGAOX_env_path
Environment variable setting the MagAO-X path.
Definition: environment.hpp:20
std::pair< std::string, indiCallBack > callBackValueType
Value type of the indiCallBack map.
Definition: MagAOXApp.hpp:410
static constexpr logPrioT LOG_DEBUG
Used for debugging.
Definition: logPriority.hpp:52
std::string secretsPath
Path to the secrets directory, where passwords, etc, are stored.
Definition: MagAOXApp.hpp:84
int setupConfig(mx::appConfigurator &config)
Setup an application configurator for the logger section.
Definition: logManager.hpp:296
pcf::IndiProperty m_indiP_state
indi Property to report the application state.
Definition: MagAOXApp.hpp:537