25#ifndef GOBY_MIDDLEWARE_APPLICATION_INTERFACE_H 
   26#define GOBY_MIDDLEWARE_APPLICATION_INTERFACE_H 
   34#include <boost/format.hpp> 
   52template <
typename App>
 
   61template <
typename App,
 
   62          typename Configurator = middleware::ProtobufConfigurator<typename App::ConfigType>>
 
   63int run(
int argc, 
char* argv[])
 
   65    return run<App>(Configurator(argc, argv));
 
 
   99    virtual void run() = 0;
 
  113    void quit(
int return_value = 0)
 
  116        return_value_ = return_value;
 
 
  128            throw(
goby::Exception(
"No lat_origin and lon_origin defined for requested UTMGeodesy"));
 
 
  134    std::string 
app_name() { 
return app3_base_configuration_->name(); }
 
  138    boost::units::quantity<boost::units::si::frequency>
 
  141        if (app3_base_configuration_->has_loop_frequency())
 
  142            return app3_base_configuration_->loop_frequency_with_units();
 
  144            return compiled_loop_freq;
 
 
  151    template <
typename App>
 
  152    friend int ::goby::run(
 
  160    void run_one() { 
run(); }
 
  162    void configure_logger();
 
  163    void configure_glog_file();
 
  164    void check_rotate_glog_file();
 
  168    static std::unique_ptr<Config> app_cfg_;
 
  169    static std::unique_ptr<protobuf::AppConfig> app3_base_configuration_;
 
  175    static std::unique_ptr<std::ofstream> fout_;
 
  179    std::unique_ptr<goby::util::UTMGeodesy> geodesy_;
 
 
  185template <
typename Config>
 
  190template <
typename Config>
 
  191std::unique_ptr<goby::middleware::protobuf::AppConfig>
 
  199    if (app3_base_configuration_->has_geodesy())
 
  201                           app3_base_configuration_->geodesy().lon_origin_with_units()});
 
  203    if (!app3_base_configuration_->IsInitialized())
 
  206    glog.
is_debug2() && 
glog << 
"Application: constructed with PID: " << getpid() << std::endl;
 
  207    glog.
is_debug1() && 
glog << 
"App name is " << app3_base_configuration_->name() << std::endl;
 
  208    glog.
is_debug2() && 
glog << 
"Configuration is: " << app_cfg_->DebugString() << std::endl;
 
 
  218                        app3_base_configuration_->glog_config().tty_verbosity()),
 
  221    if (app3_base_configuration_->glog_config().show_gui())
 
  224    if (app3_base_configuration_->glog_config().has_file_log())
 
  225        configure_glog_file();
 
  227    if (app3_base_configuration_->glog_config().show_dccl_log())
 
  234    if (app3_base_configuration_->glog_config().has_file_log() &&
 
  235        app3_base_configuration_->glog_config().file_log().has_log_rotate_sec() &&
 
  239        configure_glog_file();
 
  245    const auto& file_log = app3_base_configuration_->glog_config().file_log();
 
  246    std::string file_format_str;
 
  248    std::string file_name_format = file_log.file_name();
 
  250    if (!file_log.has_file_name() && file_log.omit().file_timestamp())
 
  251        file_name_format = 
"%2%.txt";
 
  253    if (file_log.has_file_dir() && !file_log.file_dir().empty())
 
  255        auto file_dir = file_log.file_dir();
 
  256        if (file_dir.back() != 
'/')
 
  258        file_format_str = file_dir + file_name_format;
 
  262        file_format_str = file_name_format;
 
  265    boost::format file_format(file_format_str);
 
  267    if (!file_log.omit().file_timestamp())
 
  269        if (file_format_str.find(
"%1") == std::string::npos)
 
  271                glog << 
"file_name string must contain \"%1%\" which is expanded to the current " 
  272                        "application start time (e.g. 20190201T184925), unless omit.file_timestamp " 
  273                        "== true. Erroneous file_name is: " 
  274                     << file_format_str << std::endl;
 
  277    file_format.exceptions(boost::io::all_error_bits ^
 
  278                           (boost::io::too_many_args_bit | boost::io::too_few_args_bit));
 
  280    std::string file_name =
 
  284    fout_.reset(
new std::ofstream(file_name.c_str()));
 
  286    if (!fout_->is_open())
 
  287        glog.
is_die() && 
glog << 
"cannot write glog output to requested file: " << file_name
 
  290    if (!file_log.omit().latest_symlink())
 
  292        std::string file_symlink =
 
  293            (file_format % 
"latest" % app3_base_configuration_->name()).
str();
 
  294        remove(file_symlink.c_str());
 
  295        int result = symlink(realpath(file_name.c_str(), NULL), file_symlink.c_str());
 
  298                glog << 
"Cannot create symlink to latest file. Continuing onwards anyway" 
  304    if (file_log.has_log_rotate_sec())
 
  305        next_log_rotate_time_ =
 
  309template <
typename Config>
 
  319    sigset_t signal_mask;
 
  320    sigemptyset(&signal_mask);
 
  321    sigaddset(&signal_mask, SIGWINCH);
 
  322    pthread_sigmask(SIG_BLOCK, &signal_mask, NULL);
 
  324    this->pre_initialize();
 
  326    this->post_initialize();
 
  331        this->check_rotate_glog_file();
 
  333    this->pre_finalize();
 
  335    this->post_finalize();
 
  336    return return_value_;
 
  339template <
typename App>
 
  342    int return_value = 0;
 
  358            std::cout << cfgtor.
str() << std::endl;
 
  363        App::app_cfg_.reset(
new typename App::ConfigType(cfgtor.
cfg()));
 
  364        App::app3_base_configuration_.reset(
 
  368        if (App::app3_base_configuration_->simulation().time().use_sim_time())
 
  372                App::app3_base_configuration_->simulation().time().warp_factor();
 
  373            if (App::app3_base_configuration_->simulation().time().has_reference_microtime())
 
  375                    std::chrono::system_clock::time_point(std::chrono::microseconds(
 
  376                        App::app3_base_configuration_->simulation().time().reference_microtime()));
 
  381        return_value = app.__run();
 
  383    catch (std::exception& e)
 
  386        std::cerr << 
"Application:: uncaught exception: " << e.what() << std::endl;
 
 
simple exception class for goby applications
Base class for Goby applications. Generally you will want to use SingleThreadApplication or MultiThre...
const util::UTMGeodesy & geodesy()
Accesses the geodetic conversion tool if lat_origin and lon_origin were provided.
virtual void pre_finalize()
Called just before finalize.
virtual void post_initialize()
Called just after initialize.
virtual void initialize()
Perform any initialize tasks that couldn't be done in the constructor.
virtual void pre_initialize()
Called just before initialize.
virtual void run()=0
Runs once.
bool has_geodesy()
Returns if the geodesy tool is configured with a datum.
virtual void finalize()
Perform any final cleanup actions just before the destructor is called.
void quit(int return_value=0)
Requests a clean exit.
void configure_geodesy(goby::util::UTMGeodesy::LatLonPoint datum)
boost::units::quantity< boost::units::si::frequency > choose_loop_freq(boost::units::quantity< boost::units::si::frequency > compiled_loop_freq)
virtual void post_finalize()
Called just after finalize.
const Config & app_cfg()
Accesses configuration object passed at launch.
indicates a problem with the runtime command line or .cfg file configuration (or –help was given)
Defines the interface to a "configurator", a class that can read command line parameters (argc,...
virtual void validate() const
Override to validate the configuration.
virtual void handle_config_error(middleware::ConfigException &e) const
Override to customize how ConfigException errors are handled.
virtual const protobuf::AppConfig & app_configuration() const
Subset of the configuration used to configure the Application itself.
const Config & cfg() const
The configuration object produced from the command line parameters.
virtual std::string str() const =0
Override to output the configuration object as a string.
void remove_stream(std::ostream *os=nullptr)
void set_name(const std::string &s)
Set the name of the application that the logger is serving.
void add_stream(logger::Verbosity verbosity=logger::VERBOSE, std::ostream *os=nullptr)
Attach a stream object (e.g. std::cout, std::ofstream, ...) to the logger with desired verbosity.
std::string str(TimeType value=SystemClock::now< TimeType >())
Returns the provided time (or current time if omitted) as a human-readable string.
std::string file_str(TimeType value=SystemClock::now< TimeType >())
Returns the provided time (or current time if omitted) as an ISO string suitable for file names (no s...
The global namespace for the Goby project.
util::FlexOstream glog
Access the Goby logger through this object.
int run(const goby::middleware::ConfiguratorInterface< typename App::ConfigType > &cfgtor)
Run a Goby application using the provided Configurator.
static void setup_dlog()
Enable dlog output to glog using same verbosity settings as glog.
static bool using_sim_time
Enables simulation time if true (if false, none of the remaining parameters are used)
static std::chrono::system_clock::time_point reference_time
Reference time when calculating SystemClock::now(). If this is unset, the default is 1 January of the...
static int warp_factor
Warp factor to speed up (or slow time) the time values returned by SteadyClock::now() and SystemClock...
std::chrono::time_point< SteadyClock > time_point
static time_point now() noexcept
Returns the current steady time unless SimulatorSettings::using_sim_time == true in which case a simu...