#pragma once #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include /// \brief Base class for applications that can run as daemons. /// /// \code /// # Some possible command line options: /// # --config-file, -C or --config - path to configuration file. By default - config.xml in the current directory. /// # --log-file /// # --errorlog-file /// # --daemon - run as daemon; without this option, the program will be attached to the terminal and also output logs to stderr. /// --daemon --config-file=localfile.xml --log-file=log.log --errorlog-file=error.log /// \endcode /// /// You can configure different log options for different loggers used inside program /// by providing subsections to "logger" in configuration file. class BaseDaemon : public Poco::Util::ServerApplication, public Loggers { friend class SignalListener; public: static constexpr char DEFAULT_GRAPHITE_CONFIG_NAME[] = "graphite"; BaseDaemon(); ~BaseDaemon() override; /// Load configuration, prepare loggers, etc. void initialize(Poco::Util::Application &) override; void reloadConfiguration(); /// Process command line parameters void defineOptions(Poco::Util::OptionSet & new_options) override; /// Graceful shutdown static void terminate(); /// Forceful shutdown [[noreturn]] void kill(); /// Cancellation request has been received. bool isCancelled() const { return is_cancelled; } static BaseDaemon & instance() { return dynamic_cast(Poco::Util::Application::instance()); } /// return none if daemon doesn't exist, reference to the daemon otherwise static std::optional> tryGetInstance() { return tryGetInstance(); } /// Graphite metric name has components separated by dots. /// We used to have the following format: root_path.hostname_clickhouse_com.key /// root_path - one_min by default /// key - something that makes sense. Examples: "meminfo.cached" or "meminfo.free", "meminfo.total". template void writeToGraphite(const std::string & key, const T & value, const std::string & config_name = DEFAULT_GRAPHITE_CONFIG_NAME, time_t timestamp = 0, const std::string & custom_root_path = "") { auto *writer = getGraphiteWriter(config_name); if (writer) writer->write(key, value, timestamp, custom_root_path); } template void writeToGraphite(const GraphiteWriter::KeyValueVector & key_vals, const std::string & config_name = DEFAULT_GRAPHITE_CONFIG_NAME, time_t timestamp = 0, const std::string & custom_root_path = "") { auto *writer = getGraphiteWriter(config_name); if (writer) writer->write(key_vals, timestamp, custom_root_path); } template void writeToGraphite(const GraphiteWriter::KeyValueVector & key_vals, const std::chrono::system_clock::time_point & current_time, const std::string & custom_root_path) { auto *writer = getGraphiteWriter(); if (writer) writer->write(key_vals, std::chrono::system_clock::to_time_t(current_time), custom_root_path); } GraphiteWriter * getGraphiteWriter(const std::string & config_name = DEFAULT_GRAPHITE_CONFIG_NAME) { if (graphite_writers.contains(config_name)) return graphite_writers[config_name].get(); return nullptr; } /// close all process FDs except /// 0-2 -- stdin, stdout, stderr /// also doesn't close global internal pipes for signal handling static void closeFDs(); /// If this method is called after initialization and before run, /// will fork child process and setup watchdog that will print diagnostic info, if the child terminates. /// argv0 is needed to change process name (consequently, it is needed for scripts involving "pgrep", "pidof" to work correctly). void shouldSetupWatchdog(char * argv0_); /// Hash of the binary for integrity checks. String getStoredBinaryHash() const; protected: virtual void logRevision() const; /// thread safe virtual void handleSignal(int signal_id); /// initialize termination process and signal handlers virtual void initializeTerminationAndSignalProcessing(); /// fork the main process and watch if it was killed void setupWatchdog(); void waitForTerminationRequest() override; /// thread safe virtual void onInterruptSignals(int signal_id); template static std::optional> tryGetInstance(); virtual std::string getDefaultCorePath() const; virtual std::string getDefaultConfigFileName() const; std::optional pid_file; std::atomic_bool is_cancelled{false}; bool log_to_console = false; /// A thread that acts on HUP and USR1 signal (close logs). Poco::Thread signal_listener_thread; std::unique_ptr signal_listener; std::map> graphite_writers; std::mutex signal_handler_mutex; std::condition_variable signal_event; std::atomic_size_t terminate_signals_counter{0}; std::string config_path; DB::ConfigProcessor::LoadedConfig loaded_config; Poco::Util::AbstractConfiguration * last_configuration = nullptr; String build_id; String stored_binary_hash; bool should_setup_watchdog = false; char * argv0 = nullptr; }; template std::optional> BaseDaemon::tryGetInstance() { Daemon * ptr = nullptr; try { ptr = dynamic_cast(&Poco::Util::Application::instance()); } catch (const Poco::NullPointerException &) /// NOLINT(bugprone-empty-catch) { /// if daemon doesn't exist than instance() throw NullPointerException } if (ptr) return std::optional>(*ptr); return {}; } #if defined(OS_LINUX) /// Sends notification (e.g. "server is ready") to systemd, analogous to sd_notify from libsystemd. /// See https://www.freedesktop.org/software/systemd/man/sd_notify.html for more information on the supported notifications. void systemdNotify(const std::string_view & command); #endif