API
 
Loading...
Searching...
No Matches
observerCtrl.hpp
Go to the documentation of this file.
1/** \file observerCtrl.hpp
2 * \brief The MagAO-X Observer Controller header file
3 *
4 * \ingroup observerCtrl_files
5 */
6
7#ifndef observerCtrl_hpp
8#define observerCtrl_hpp
9
10#include <map>
11#include <mx/math/geo.hpp>
12
13#include "../../libMagAOX/libMagAOX.hpp" //Note this is included on command line to trigger pch
14#include "../../magaox_git_version.h"
15
16/** \defgroup observerCtrl
17 * \brief The MagAO-X Observer Controller application
18 *
19 * <a href="../handbook/operating/software/apps/observerCtrl.html">Application Documentation</a>
20 *
21 * \ingroup apps
22 *
23 */
24
25/** \defgroup observerCtrl_files
26 * \ingroup observerCtrl
27 */
28
29namespace MagAOX
30{
31namespace app
32{
33
34/// The MagAO-X Observer Controller
35/**
36 * \ingroup observerCtrl
37 */
38class observerCtrl : public MagAOXApp<true>, public dev::telemeter<observerCtrl>
39{
40
41 // Give the test harness access.
42 friend class observerCtrl_test;
43
44 friend class dev::telemeter<observerCtrl>;
45
47
48 typedef std::chrono::time_point<std::chrono::steady_clock> timePointT;
49 typedef std::chrono::duration<double> durationT;
50
51 protected:
52 /** \name Configurable Parameters
53 *@{
54 */
55
56 std::vector<std::string> m_streamWriters; ///< The stream writers to stop and start
57
58 std::string m_tcsDev{ "tcsi" };
59 std::string m_catalogProp{ "catalog" };
60 std::string m_objEl{ "object" };
61 std::string m_catdataProp{ "catdata" };
62 std::string m_raEl{ "ra" };
63 std::string m_decEl{ "dec" };
64 std::string m_labModeProp{"labMode"};
65
66 std::string m_teldataProp{ "teldata" };
67 std::string m_parangEl{ "pa" };
68
69 std::string m_loopDev{ "holoop" };
70 std::string m_loopStateProp{ "loop_state" };
71
72 ///@}
73
74 /// The observer specification
75 struct observer
76 {
77 std::string m_fullName; ///< Obsever's full name
78 std::string m_pfoa; ///< Observer's preferred forma of address
79 std::string m_pronunciation; ///< Guide for the TTS to pronounced the pfoa (defaults to pfoa)
80 std::string m_email; ///< Observer's email. Must be unique.
81 std::string m_sanitizedEmail; ///< Observer's email sanitized for use in INDI properties
82 std::string m_institution; ///< The observer's institution
83 };
84
85 typedef std::map<std::string, observer> observerMapT;
86
87 observerMapT m_observers; ///< The observers from the configuration file
88
89 observer m_currentObserver; ///< The current selected observer
90
91 std::string m_obsName; ///< The name of the observation.
92 double m_obsDuration{ 0 }; ///< The desired duration of the observation. If 0 then until stopped.
93
94 bool m_observing{ false }; ///< Flag indicating whether or not we are in an observation
95
96 std::string m_target;
97
98 std::string m_catObj;
99
100 std::string m_catRA;
101 std::string m_catDec;
102
103 bool m_loop {false}; ///< Flag tracking loop state. true is loop closed.
104
105 bool m_labMode {false}; ///< Flag tracking whether the TCS interface is in lab mode.
106
107
108 bool m_newTargetBlock{ true }; /**< Flag to indicate that this is a new target block. This starts out as true
109 but becomes false on the first observation. Then becomes true when the
110 loop closes for the first time after a target change. */
111 bool m_newTarget{ false }; /**< Flag to track when the target changes. Occurs either automatically on a TCS update
112 or on a user override.*/
113
114
115
116 /// The start time of the current observation
118
119 /// The parallactic angle at the start of the observation
120 double m_obsStartParang{ 0 };
121
122 /// The start time of the current target
124
125 /// Teh parallactic angle at the start of observing the current target
126 double m_tgtStartParang{ 0 };
127
128 /// The current parallactic angle
129 double m_parang;
130
131 /// The current target time. Only updated while observing.
133
134 /// The current target angle. Only updated while observing.
135 double m_tgtAng {0};
136
137
138 public:
139 /// Default c'tor.
140 observerCtrl();
141
142 /// D'tor, declared and defined for noexcept.
144 {
145 }
146
147 virtual void setupConfig();
148
149 /// Implementation of loadConfig logic, separated for testing.
150 /** This is called by loadConfig().
151 */
152 int loadConfigImpl( mx::app::appConfigurator &_config /**< [in] an application configuration
153 from which to load values*/
154 );
155
156 virtual void loadConfig();
157
158 /// Startup function
159 /**
160 *
161 */
162 virtual int appStartup();
163
164 /// Implementation of the FSM for observerCtrl.
165 /**
166 * \returns 0 on no critical error
167 * \returns -1 on an error requiring shutdown
168 */
169 virtual int appLogic();
170
171 /// Shutdown the app.
172 /**
173 *
174 */
175 virtual int appShutdown();
176
177 void startObserving();
178
179 void stopObserving();
180
181 ///\name INDI
182 /** @{
183 */
184 protected:
185 pcf::IndiProperty m_indiP_observers; ///< Selection switch to allow selection of the observer
186 pcf::IndiProperty m_indiP_observer; ///< Text which contains the specifications of the current observer
187 pcf::IndiProperty m_indiP_obsName; /**< The current observation name, used to specify the
188 purpose of the observation*/
189 pcf::IndiProperty m_indiP_observing; ///< Toggle switch to trigger observation
190 pcf::IndiProperty m_indiP_obsDuration; ///< Number to set the desired duration of observation
191 pcf::IndiProperty m_indiP_obsTime; ///< Number tracking the elapsed time
192 pcf::IndiProperty m_indiP_obsAngle; ///< Number tracking the change in angle
193 pcf::IndiProperty m_indiP_sws; ///< Selection to switch to define which stream writers are enabled
194 pcf::IndiProperty m_indiP_userlog; ///< Text to enter a user log
195
196 pcf::IndiProperty m_indiP_resetTarget; ///< Reset the target statistics
197
198 pcf::IndiProperty m_indiP_target; ///< The target name, which can be overridden by the user
199
200 pcf::IndiProperty m_indiP_catalog; ///< Catalog text data
201 pcf::IndiProperty m_indiP_catdata; ///< Catalog numeric data
202 pcf::IndiProperty m_indiP_teldata; ///< Telescope data (for parang)
203
204 pcf::IndiProperty m_indiP_labMode; ///< Tracks whether TCS is in lab mode.
205
206 pcf::IndiProperty m_indiP_loop; ///< Tracks the loop state
207
208
209 public:
211
215
217
219
221
223
225
227
229
231
233
234
235 ///@}
236
237 /** \name Telemeter Interface
238 *
239 * @{
240 */
241 int checkRecordTimes();
242
243 int recordTelem( const telem_observer * );
244
245 int recordObserver( bool force = false );
246
247 ///@}
248};
249
250observerCtrl::observerCtrl() : MagAOXApp( MAGAOX_CURRENT_SHA1, MAGAOX_REPO_MODIFIED )
251{
252 return;
253}
254
256{
257 config.add( "stream.writers",
258 "",
259 "stream.writers",
260 argType::Required,
261 "stream",
262 "writers",
263 false,
264 "string",
265 "The device names of the stream writers to control." );
266
268}
269
270int observerCtrl::loadConfigImpl( mx::app::appConfigurator &_config )
271{
272 _config( m_streamWriters, "stream.writers" );
273
274 std::vector<std::string> sections;
275
276 _config.unusedSections( sections );
277
278 if( sections.size() == 0 )
279 {
280 log<text_log>( "no observers found in config", logPrio::LOG_CRITICAL );
281 return -1;
282 }
283
284 for( size_t i = 0; i < sections.size(); ++i )
285 {
286 bool pfoaSet = _config.isSetUnused( mx::app::iniFile::makeKey( sections[i], "pfoa" ) );
287 if( !pfoaSet )
288 continue;
289
290 std::string email = sections[i];
291
292 std::string pfoa;
293 _config.configUnused( pfoa, mx::app::iniFile::makeKey( sections[i], "pfoa" ) );
294
295 std::string pronunciation;
296 _config.configUnused( pronunciation, mx::app::iniFile::makeKey( sections[i], "pronunciation" ) );
297
298 if( pronunciation == "" )
299 {
301 }
302
303 std::string fullName;
304 _config.configUnused( fullName, mx::app::iniFile::makeKey( sections[i], "full_name" ) );
305
306 std::string institution;
307 _config.configUnused( institution, mx::app::iniFile::makeKey( sections[i], "institution" ) );
308
309 std::string sanitizedEmail = "";
310 for( size_t n = 0; n < email.size(); ++n )
311 {
312 if( email[n] == '@' )
313 {
315 }
316 else if( email[n] == '.' )
317 {
318 sanitizedEmail = sanitizedEmail + "-dot-";
319 }
320 else
321 {
322 sanitizedEmail.push_back( email[n] );
323 }
324 }
325 m_observers[email] = observer( { fullName, pfoa, pronunciation, email, sanitizedEmail, institution } );
326 }
327
328 return 0;
329}
330
332{
333 if( loadConfigImpl( config ) < 0 )
334 {
335 m_shutdown = 1;
336 return;
337 }
338
339 if( m_observers.size() < 1 )
340 {
341 log<text_log>( "no observers found in config", logPrio::LOG_CRITICAL );
342 m_shutdown = 1;
343 return;
344 }
345
347}
348
350{
351 std::vector<std::string> sanitizedEmails;
352 std::vector<std::string> emails;
353 for( auto it = m_observers.begin(); it != m_observers.end(); ++it )
354 {
355 sanitizedEmails.push_back( it->second.m_sanitizedEmail );
356 emails.push_back( it->second.m_email );
357 }
358
360 {
362 return -1;
363 }
364
365 // Set to default user of jared
366 ///\todo do something else. maybe a default user is specified in the config?
367 for( auto &it : m_observers )
368 {
369 if( it.first.find( "jrmales" ) != std::string::npos )
370 {
371 m_indiP_observers[it.second.m_sanitizedEmail].setSwitchState( pcf::IndiElement::On );
372 m_currentObserver = it.second;
373 }
374 else
375 {
376 m_indiP_observers[it.second.m_sanitizedEmail].setSwitchState( pcf::IndiElement::Off );
377 }
378 }
379
381
382 CREATE_REG_INDI_NEW_TEXT( m_indiP_obsName, "obs_name", "Observation Name", "Observer" );
383
385
386 CREATE_REG_INDI_NEW_NUMBERD( m_indiP_obsDuration, "obs_duration", 0, 300, 0.1, "%0.1f", "Duration", "Observer" );
387
388 CREATE_REG_INDI_RO_NUMBER( m_indiP_obsTime, "obs_time", "Observation Time", "Observer" );
389 indi::addNumberElement<double>( m_indiP_obsTime, "observation", 0, 14400, 0.1, "%0.1f", "Current Obs" );
390 indi::addNumberElement<double>( m_indiP_obsTime, "target", 0, 14400, 0.1, "%0.1f", "Target" );
391
392 CREATE_REG_INDI_RO_NUMBER( m_indiP_obsAngle, "obs_delta_parang", "Change in Par. Ang.", "Observer" );
393 indi::addNumberElement<double>( m_indiP_obsAngle, "observation", 0, 360, 0.1, "%0.1f", "Current Obs" );
394 indi::addNumberElement<double>( m_indiP_obsAngle, "target", 0, 360, 0.1, "%0.1f", "Target" );
395
396 REG_INDI_NEWPROP_NOCB( m_indiP_observer, "current_observer", pcf::IndiProperty::Text );
400 indi::addTextElement( m_indiP_observer, "pronunciation" );
401 indi::addTextElement( m_indiP_observer, "institution" );
402
403 m_indiP_sws = pcf::IndiProperty( pcf::IndiProperty::Switch );
404 m_indiP_sws.setDevice( configName() );
405 m_indiP_sws.setName( "writers" );
406 m_indiP_sws.setPerm( pcf::IndiProperty::ReadWrite );
407 m_indiP_sws.setState( pcf::IndiProperty::Idle );
408 m_indiP_sws.setRule( pcf::IndiProperty::AnyOfMany );
409
410 for( size_t n = 0; n < m_streamWriters.size(); ++n )
411 {
412 m_indiP_sws.add( pcf::IndiElement( m_streamWriters[n], pcf::IndiElement::Off ) );
413 }
414
416
417 m_indiP_userlog = pcf::IndiProperty( pcf::IndiProperty::Text );
418 m_indiP_userlog.setDevice( configName() );
419 m_indiP_userlog.setName( "user_log" );
420 m_indiP_userlog.setPerm( pcf::IndiProperty::ReadWrite );
421 m_indiP_userlog.setState( pcf::IndiProperty::Idle );
422 m_indiP_userlog.add( pcf::IndiElement( "email" ) );
423 m_indiP_userlog.add( pcf::IndiElement( "message" ) );
424 m_indiP_userlog.add( pcf::IndiElement( "time_s" ) );
425 m_indiP_userlog.add( pcf::IndiElement( "time_ns" ) );
426
428
430
431 CREATE_REG_INDI_NEW_TEXT( m_indiP_target, "target", "Target", "Observer" );
432
437
439
441
443 return 0;
444}
445
447{
448
449 std::unique_lock<std::mutex> lock( m_indiMutex, std::try_to_lock );
450
451 if( lock.owns_lock() )
452 {
454 { "full_name", "email", "pfoa", "pronunciation", "institution" },
460
461 for( auto it = m_observers.begin(); it != m_observers.end(); ++it )
462 {
463 if( it->first == m_currentObserver.m_email )
464 {
466 m_indiP_observers, it->second.m_sanitizedEmail, pcf::IndiElement::On, INDI_IDLE );
467 }
468 else
469 {
471 m_indiP_observers, it->second.m_sanitizedEmail, pcf::IndiElement::Off, INDI_IDLE );
472 }
473 }
474
475 updatesIfChanged<std::string>( m_indiP_obsName, { "current", "target" }, { m_obsName, m_obsName } );
476
478
479 if( m_observing )
480 {
481 timePointT ct = std::chrono::steady_clock::now();
482 const std::chrono::duration<double> obstime = ct - m_obsStartTime;
484
485 double obsang = mx::math::angleDiff<mx::math::degreesT<double>>( m_parang, m_obsStartParang );
486 m_tgtAng = mx::math::angleDiff<mx::math::degreesT<double>>( m_parang, m_tgtStartParang );
487
488 updateSwitchIfChanged( m_indiP_observing, "toggle", pcf::IndiElement::On, INDI_OK );
490 m_indiP_obsTime, { "observation", "target" }, { obstime.count(), m_tgtTime.count() } );
491
492 updatesIfChanged<double>( m_indiP_obsAngle, { "observation", "target" }, { obsang, m_tgtAng } );
493
494 if( m_obsDuration > 0.0 && obstime.count() > m_obsDuration )
495 {
497 }
498 }
499 else
500 {
501 updateSwitchIfChanged( m_indiP_observing, "toggle", pcf::IndiElement::Off, INDI_IDLE );
502 updatesIfChanged<double>( m_indiP_obsTime, { "observation", "target" }, { 0.0, m_tgtTime.count() } );
503 updatesIfChanged<double>( m_indiP_obsAngle, { "observation", "target" }, { 0.0, m_tgtAng } );
504 }
505 }
506
508
509 return 0;
510}
511
513{
515
516 return 0;
517}
518
520{
521 for( size_t n = 0; n < m_streamWriters.size(); ++n )
522 {
523 if( m_indiP_sws[m_streamWriters[n]].getSwitchState() == pcf::IndiElement::On )
524 {
525 pcf::IndiProperty ip( pcf::IndiProperty::Switch );
526
527 ip.setDevice( m_streamWriters[n] + "-sw" );
528 ip.setName( "writing" );
529 ip.add( pcf::IndiElement( "toggle" ) );
530 ip["toggle"].setSwitchState( pcf::IndiElement::On );
531
533 }
534 }
535
536 mx::sys::sleep( 1 );
537
538 m_obsStartTime = std::chrono::steady_clock::now();
540
541 if( m_newTargetBlock || m_labMode ) //We always reset if in lab mode
542 {
545 m_newTargetBlock = false;
546 }
547
548 m_observing = true;
549 recordObserver( true );
550}
551
553{
554 m_observing = false;
555 recordObserver( true );
556
557 for( size_t n = 0; n < m_streamWriters.size(); ++n )
558 {
559 if( m_indiP_sws[m_streamWriters[n]].getSwitchState() == pcf::IndiElement::On )
560 {
561 pcf::IndiProperty ip( pcf::IndiProperty::Switch );
562
563 ip.setDevice( m_streamWriters[n] + "-sw" );
564 ip.setName( "writing" );
565 ip.add( pcf::IndiElement( "toggle" ) );
566 ip["toggle"].setSwitchState( pcf::IndiElement::Off );
567
569 }
570 }
571}
572
573INDI_NEWCALLBACK_DEFN( observerCtrl, m_indiP_observers )( const pcf::IndiProperty &ipRecv )
574{
575
576 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_observers, ipRecv );
577
578 // look for selected mode switch which matches a known mode. Make sure only one is selected.
579 std::string newEmail = "";
580 for( auto it = m_observers.begin(); it != m_observers.end(); ++it )
581 {
582 if( !ipRecv.find( it->second.m_sanitizedEmail ) )
583 continue;
584
585 if( ipRecv[it->second.m_sanitizedEmail].getSwitchState() == pcf::IndiElement::On )
586 {
587 if( newEmail != "" )
588 {
589 log<text_log>( "More than one observer selected", logPrio::LOG_ERROR );
590 return -1;
591 }
592
593 newEmail = it->first;
594 }
595 }
596
597 if( newEmail == "" )
598 {
599 std::cerr << "nothing\n";
600 return 0;
601 }
602
603 {
604 std::unique_lock<std::mutex> lock( m_indiMutex );
605
606 m_currentObserver = m_observers[newEmail];
607
608 for( auto it = m_observers.begin(); it != m_observers.end(); ++it )
609 {
610 if( it->first == m_currentObserver.m_sanitizedEmail )
611 {
612 updateSwitchIfChanged( m_indiP_observers, it->second.m_sanitizedEmail, pcf::IndiElement::On, INDI_IDLE );
613 }
614 else
615 {
616 updateSwitchIfChanged( m_indiP_observers, it->second.m_sanitizedEmail, pcf::IndiElement::Off, INDI_IDLE );
617 }
618 }
619 }
620
621 log<logger::observer>( { m_currentObserver.m_fullName,
622 m_currentObserver.m_pfoa,
623 m_currentObserver.m_email,
624 m_currentObserver.m_institution } );
625
626 return 0;
627}
628
629INDI_NEWCALLBACK_DEFN( observerCtrl, m_indiP_obsName )( const pcf::IndiProperty &ipRecv )
630{
631 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_obsName, ipRecv );
632
633 std::string target;
634
635 std::unique_lock<std::mutex> lock( m_indiMutex );
636
637 if( indiTargetUpdate( m_indiP_obsName, target, ipRecv, true ) < 0 )
638 {
639 log<software_error>( { __FILE__, __LINE__ } );
640 return -1;
641 }
642
643 m_obsName = target;
644
645 return 0;
646}
647
648INDI_NEWCALLBACK_DEFN( observerCtrl, m_indiP_observing )( const pcf::IndiProperty &ipRecv )
649{
650 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_observing, ipRecv );
651
652 if( !ipRecv.find( "toggle" ) )
653 {
654 return 0;
655 }
656
657 recordObserver( true );
658 if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::On )
659 {
660 std::unique_lock<std::mutex> lock( m_indiMutex );
661 startObserving();
662 updateSwitchIfChanged( m_indiP_observing, "toggle", pcf::IndiElement::On, INDI_OK );
663 }
664 else if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::Off )
665 {
666 std::unique_lock<std::mutex> lock( m_indiMutex );
667 stopObserving();
668 updateSwitchIfChanged( m_indiP_observing, "toggle", pcf::IndiElement::Off, INDI_IDLE );
669 }
670
671 return 0;
672}
673
674INDI_NEWCALLBACK_DEFN( observerCtrl, m_indiP_obsDuration )( const pcf::IndiProperty &ipRecv )
675{
676 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_obsDuration, ipRecv );
677
678 return indiTargetUpdate( m_indiP_obsDuration, m_obsDuration, ipRecv, false );
679}
680
681INDI_NEWCALLBACK_DEFN( observerCtrl, m_indiP_sws )( const pcf::IndiProperty &ipRecv )
682{
683 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_sws, ipRecv );
684
685 if( m_observing == true )
686 {
687 log<text_log>( { "Can't change stream writers while observing" }, logPrio::LOG_WARNING );
688 return 0;
689 }
690
691 for( size_t n = 0; n < m_streamWriters.size(); ++n )
692 {
693 if( !ipRecv.find( m_streamWriters[n] ) )
694 {
695 continue;
696 }
697
698 std::unique_lock<std::mutex> lock( m_indiMutex );
699 updateSwitchIfChanged( m_indiP_sws, m_streamWriters[n], ipRecv[m_streamWriters[n]].getSwitchState() );
700 }
701
702 return 0;
703}
704
705INDI_NEWCALLBACK_DEFN( observerCtrl, m_indiP_userlog )( const pcf::IndiProperty &ipRecv )
706{
707 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_userlog, ipRecv );
708
709 std::string email;
710 std::string message;
711
712 timespecX ts{};
713
714 if( ipRecv.find( "email" ) )
715 {
716 email = ipRecv["email"].get();
717 }
718
719 if( ipRecv.find( "message" ) )
720 {
721 message = ipRecv["message"].get();
722 }
723
724 if( message == "" )
725 {
726 return 0;
727 }
728
729 if( email == "" )
730 {
731 email = m_currentObserver.m_email;
732 }
733
734 if( ipRecv.find( "time_s" ) )
735 {
736 ts.time_s = ipRecv["time_s"].get<flatlogs::secT>();
737 }
738
739 if( ipRecv.find( "time_ns" ) )
740 {
741 ts.time_ns = ipRecv["time_ns"].get<flatlogs::nanosecT>();
742 }
743
744 if( ts.time_s != 0 )
745 {
746 m_log.template log<user_log>( ts, { email, message }, logPrio::LOG_INFO );
747 }
748 else
749 {
750 log<user_log>( { email, message } );
751 }
752
753 return 0;
754}
755
756INDI_NEWCALLBACK_DEFN( observerCtrl, m_indiP_resetTarget )( const pcf::IndiProperty &ipRecv )
757{
758 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_resetTarget, ipRecv );
759
760 if( !ipRecv.find( "request" ) )
761 {
762 return 0;
763 }
764
765 if( ipRecv["request"].getSwitchState() == pcf::IndiElement::On )
766 {
767 if(m_observing)
768 {
769 m_tgtStartTime = m_obsStartTime;
770 m_tgtStartParang = m_obsStartParang;
771 }
772 else
773 {
774 m_newTargetBlock = true;
775 }
776
777 }
778
779 return 0;
780}
781
782INDI_NEWCALLBACK_DEFN( observerCtrl, m_indiP_target )( const pcf::IndiProperty &ipRecv )
783{
784 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_target, ipRecv );
785
786 std::string target;
787 if( indiTargetUpdate( m_indiP_target, target, ipRecv ) < 0 )
788 {
789 return log<software_error, -1>( { __FILE__, __LINE__ } );
790 }
791
792 if( target != m_target )
793 {
794 m_target = target;
795 m_newTarget = true;
796
797 log<text_log>( "Target updated by observer to " + m_target, logPrio::LOG_NOTICE );
798
799 std::unique_lock<std::mutex> lock( m_indiMutex );
800 updatesIfChanged<std::string>( m_indiP_target, { "current", "target" }, { m_target, m_target } );
801 }
802
803 return 0;
804}
805
806INDI_SETCALLBACK_DEFN( observerCtrl, m_indiP_catalog )( const pcf::IndiProperty &ipRecv )
807{
808 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_catalog, ipRecv );
809
810 if( !ipRecv.find( "object" ) )
811 {
812 return -1;
813 }
814
815 std::string object = ipRecv["object"].get();
816
817 if( object != m_catObj )
818 {
819 m_catObj = object;
820 m_target = object;
821 m_newTarget = true;
822
823 // Always log change in name (different from RA and DEC)
824 log<text_log>( "Target updated by TCS to " + m_target, logPrio::LOG_NOTICE );
825
826 std::unique_lock<std::mutex> lock( m_indiMutex );
827 updatesIfChanged<std::string>( m_indiP_target, { "current", "target" }, { m_target, m_target } );
828 }
829
830 return 0;
831}
832
833INDI_SETCALLBACK_DEFN( observerCtrl, m_indiP_catdata )( const pcf::IndiProperty &ipRecv )
834{
835 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_catdata, ipRecv );
836
837 bool change = false;
838 if( ipRecv.find( "ra" ) )
839 {
840 std::string ra = ipRecv["ra"].get();
841
842 if( ra != m_catRA )
843 {
844 m_catRA = ra;
845 m_target = m_catObj;
846
847 if( !m_newTarget ) // Only log if not already new
848 {
849 change = true;
850 m_newTarget = true;
851 }
852
853 std::unique_lock<std::mutex> lock( m_indiMutex );
854 updatesIfChanged<std::string>( m_indiP_target, { "current", "target" }, { m_target, m_target } );
855 }
856 }
857
858 if( ipRecv.find( "dec" ) )
859 {
860 std::string dec = ipRecv["dec"].get();
861
862 if( dec != m_catDec )
863 {
864 m_catDec = dec;
865 m_target = m_catObj;
866
867 if( !m_newTarget ) // Only log if not already new
868 {
869 change = true;
870 m_newTarget = true;
871 }
872
873 std::unique_lock<std::mutex> lock( m_indiMutex );
874 updatesIfChanged<std::string>( m_indiP_target, { "current", "target" }, { m_target, m_target } );
875 }
876 }
877
878 if( change ) // Only log if not already new
879 {
880 log<text_log>( "Target updated by TCS", logPrio::LOG_NOTICE );
881 }
882
883 return 0;
884}
885
886INDI_SETCALLBACK_DEFN( observerCtrl, m_indiP_teldata )( const pcf::IndiProperty &ipRecv )
887{
888 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_teldata, ipRecv );
889
890 if( ipRecv.find( "pa" ) )
891 {
892 m_parang = ipRecv["pa"].get<double>();
893 }
894
895 return 0;
896}
897
898INDI_SETCALLBACK_DEFN( observerCtrl, m_indiP_labMode )( const pcf::IndiProperty &ipRecv )
899{
900 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_labMode, ipRecv );
901
902 if( ipRecv.find( "toggle" ) )
903 {
904 if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::On )
905 {
906 m_labMode = true;
907 }
908 else
909 {
910 m_newTargetBlock = true;
911 m_labMode = false;
912 }
913 }
914
915 return 0;
916}
917
918INDI_SETCALLBACK_DEFN( observerCtrl, m_indiP_loop )( const pcf::IndiProperty &ipRecv )
919{
920 INDI_VALIDATE_CALLBACK_PROPS( m_indiP_loop, ipRecv );
921
922 if( ipRecv.find( "toggle" ) )
923 {
924 if( ipRecv["toggle"].getSwitchState() == pcf::IndiElement::On )
925 {
926 if( m_newTarget == true && !m_loop)
927 {
928 m_newTargetBlock = true;
929 m_newTarget = false;
930 }
931
932 m_loop = true;
933 }
934 else
935 {
936 m_loop = false;
937 }
938 }
939
940 return 0;
941}
942
947
949{
950 return recordObserver( true );
951}
952
953inline int observerCtrl::recordObserver( bool force )
954{
955 static std::string last_email;
956 static std::string last_obsName;
957 static bool last_observing;
958 static std::string last_target;
959
962 {
964
969 }
970
971 return 0;
972}
973
974} // namespace app
975} // namespace MagAOX
976
977#endif // observerCtrl_hpp
The base-class for MagAO-X applications.
Definition MagAOXApp.hpp:73
stateCodes::stateCodeT state()
Get the current state code.
int m_shutdown
Flag to signal it's time to shutdown. When not 0, the main loop exits.
void updateSwitchIfChanged(pcf::IndiProperty &p, const std::string &el, const pcf::IndiElement::SwitchStateType &newVal, pcf::IndiProperty::PropertyStateType ipState=pcf::IndiProperty::Ok)
Update an INDI switch element value if it has changed.
static int log(const typename logT::messageT &msg, logPrioT level=logPrio::LOG_DEFAULT)
Make a log entry.
std::mutex m_indiMutex
Mutex for locking INDI communications.
std::string configName()
Get the config name.
int createStandardIndiSelectionSw(pcf::IndiProperty &prop, const std::string &name, const std::vector< std::string > &elements, const std::vector< std::string > &elementLabels, const std::string &label="", const std::string &group="")
Create a standard R/W INDI selection (one of many) switch with vector of elements and element labels.
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)
The MagAO-X Observer Controller.
INDI_SETCALLBACK_DECL(observerCtrl, m_indiP_loop)
INDI_NEWCALLBACK_DECL(observerCtrl, m_indiP_resetTarget)
pcf::IndiProperty m_indiP_observer
Text which contains the specifications of the current observer.
std::string m_pronunciation
Guide for the TTS to pronounced the pfoa (defaults to pfoa)
std::chrono::time_point< std::chrono::steady_clock > timePointT
pcf::IndiProperty m_indiP_obsName
INDI_NEWCALLBACK_DECL(observerCtrl, m_indiP_obsDuration)
double m_parang
The current parallactic angle.
int recordTelem(const telem_observer *)
timePointT m_obsStartTime
The start time of the current observation.
pcf::IndiProperty m_indiP_resetTarget
Reset the target statistics.
std::string m_sanitizedEmail
Observer's email sanitized for use in INDI properties.
pcf::IndiProperty m_indiP_obsTime
Number tracking the elapsed time.
pcf::IndiProperty m_indiP_obsAngle
Number tracking the change in angle.
durationT m_tgtTime
The current target time. Only updated while observing.
virtual int appLogic()
Implementation of the FSM for observerCtrl.
INDI_SETCALLBACK_DECL(observerCtrl, m_indiP_labMode)
INDI_NEWCALLBACK_DECL(observerCtrl, m_indiP_observing)
std::string m_pfoa
Observer's preferred forma of address.
pcf::IndiProperty m_indiP_observers
Selection switch to allow selection of the observer.
int recordObserver(bool force=false)
std::string m_institution
The observer's institution.
INDI_NEWCALLBACK_DECL(observerCtrl, m_indiP_sws)
double m_obsDuration
The desired duration of the observation. If 0 then until stopped.
pcf::IndiProperty m_indiP_labMode
Tracks whether TCS is in lab mode.
~observerCtrl() noexcept
D'tor, declared and defined for noexcept.
double m_obsStartParang
The parallactic angle at the start of the observation.
pcf::IndiProperty m_indiP_catalog
Catalog text data.
std::chrono::duration< double > durationT
INDI_NEWCALLBACK_DECL(observerCtrl, m_indiP_obsName)
INDI_NEWCALLBACK_DECL(observerCtrl, m_indiP_observers)
INDI_SETCALLBACK_DECL(observerCtrl, m_indiP_catalog)
dev::telemeter< observerCtrl > telemeterT
std::vector< std::string > m_streamWriters
The stream writers to stop and start.
pcf::IndiProperty m_indiP_obsDuration
Number to set the desired duration of observation.
INDI_SETCALLBACK_DECL(observerCtrl, m_indiP_teldata)
std::string m_fullName
Obsever's full name.
pcf::IndiProperty m_indiP_loop
Tracks the loop state.
pcf::IndiProperty m_indiP_target
The target name, which can be overridden by the user.
virtual int appStartup()
Startup function.
bool m_observing
Flag indicating whether or not we are in an observation.
INDI_NEWCALLBACK_DECL(observerCtrl, m_indiP_userlog)
double m_tgtStartParang
Teh parallactic angle at the start of observing the current target.
pcf::IndiProperty m_indiP_userlog
Text to enter a user log.
pcf::IndiProperty m_indiP_observing
Toggle switch to trigger observation.
pcf::IndiProperty m_indiP_catdata
Catalog numeric data.
INDI_SETCALLBACK_DECL(observerCtrl, m_indiP_catdata)
timePointT m_tgtStartTime
The start time of the current target.
double m_tgtAng
The current target angle. Only updated while observing.
virtual int appShutdown()
Shutdown the app.
bool m_labMode
Flag tracking whether the TCS interface is in lab mode.
bool m_loop
Flag tracking loop state. true is loop closed.
int loadConfigImpl(mx::app::appConfigurator &_config)
Implementation of loadConfig logic, separated for testing.
std::string m_obsName
The name of the observation.
std::string m_email
Observer's email. Must be unique.
observerMapT m_observers
The observers from the configuration file.
INDI_NEWCALLBACK_DECL(observerCtrl, m_indiP_target)
observer m_currentObserver
The current selected observer.
std::map< std::string, observer > observerMapT
pcf::IndiProperty m_indiP_sws
Selection to switch to define which stream writers are enabled.
pcf::IndiProperty m_indiP_teldata
Telescope data (for parang)
The observer specification.
#define INDI_NEWCALLBACK_DEFN(class, prop)
Define the callback for a new property request.
#define REG_INDI_NEWPROP_NOCB(prop, propName, type)
Register a NEW INDI property with the class, with no callback.
#define CREATE_REG_INDI_NEW_TOGGLESWITCH(prop, name)
Create and register a NEW INDI property as a standard toggle switch, using the standard callback name...
#define CREATE_REG_INDI_NEW_REQUESTSWITCH(prop, name)
Create and register a NEW INDI property as a standard request switch, using the standard callback nam...
#define INDI_SETCALLBACK_DEFN(class, prop)
Define the callback for a set property request.
#define REG_INDI_SETPROP(prop, devName, propName)
Register a SET INDI property with the class, using the standard callback name.
#define CREATE_REG_INDI_NEW_NUMBERD(prop, name, min, max, step, format, label, group)
Create and register a NEW INDI property as a standard number as double, using the standard callback n...
#define REG_INDI_NEWPROP_NOSETUP(prop)
Register a NEW INDI property with the class, using the standard callback name.
#define CREATE_REG_INDI_NEW_TEXT(prop, name, label, group)
Create and register a NEW INDI property as a standard text, using the standard callback name.
#define CREATE_REG_INDI_RO_NUMBER(prop, name, label, group)
Create and register a RO INDI property as a number, using the standard callback name.
uint32_t secT
The type used for seconds.
Definition logDefs.hpp:29
@ READY
The device is ready for operation, but is not operating.
#define INDI_VALIDATE_CALLBACK_PROPS(prop1, prop2)
Standard check for matching INDI properties in a callback.
#define INDI_IDLE
Definition indiUtils.hpp:27
#define INDI_OK
Definition indiUtils.hpp:28
int addTextElement(pcf::IndiProperty &prop, const std::string &name, const std::string &label="")
Add a standard INDI Text element.
Definition indiUtils.hpp:37
const pcf::IndiProperty & ipRecv
std::unique_lock< std::mutex > lock(m_indiMutex)
Definition dm.hpp:24
static constexpr logPrioT LOG_NOTICE
A normal but significant condition.
static constexpr logPrioT LOG_INFO
Informational. The info log level is the lowest level recorded during normal operations.
static constexpr logPrioT LOG_CRITICAL
The process can not continue and will shut down (fatal)
static constexpr logPrioT LOG_WARNING
A condition has occurred which may become an error, but the process continues.
static constexpr logPrioT LOG_ERROR
An error has occured which the software will attempt to correct.
uint32_t nanosecT
The type used for nanoseconds.
Definition logDefs.hpp:34
A device base class which saves telemetry.
Definition telemeter.hpp:69
int loadConfig(appConfigurator &config)
Load the device section from an application configurator.
int setupConfig(appConfigurator &config)
Setup an application configurator for the device section.
Software ERR log entry.
Log entry recording the build-time git state.
A fixed-width timespec structure.
Definition timespecX.hpp:35
secT time_s
Time since the Unix epoch.
Definition timespecX.hpp:36
#define TELEMETER_APP_LOGIC
Call telemeter::appLogic with error checking.
#define TELEMETER_APP_STARTUP
Call telemeter::appStartup with error checking.
#define TELEMETER_APP_SHUTDOWN
Call telemeter::appShutdown with error checking.