Wire Sysio Wire Sysion 1.0.0
Loading...
Searching...
No Matches
resource_monitor_plugin.cpp
Go to the documentation of this file.
1
21
23
25#include <fc/log/logger_config.hpp> // set_os_thread_name
26
27#include <boost/filesystem.hpp>
28
29#include <thread>
30
31#include <sys/stat.h>
32
33using namespace sysio::resource_monitor;
34
35namespace bfs = boost::filesystem;
36
37namespace sysio {
38 static appbase::abstract_plugin& _resource_monitor_plugin = app().register_plugin<resource_monitor_plugin>();
39
41public:
43 :space_handler(system_file_space_provider(), ctx)
44 {
45 }
46
47 void set_program_options(options_description&, options_description& cfg) {
48 cfg.add_options()
49 ( "resource-monitor-interval-seconds", bpo::value<uint32_t>()->default_value(def_interval_in_secs),
50 "Time in seconds between two consecutive checks of resource usage. Should be between 1 and 300" )
51 ( "resource-monitor-space-threshold", bpo::value<uint32_t>()->default_value(def_space_threshold),
52 "Threshold in terms of percentage of used space vs total space. If used space is above (threshold - 5%), a warning is generated. Unless resource-monitor-not-shutdown-on-threshold-exceeded is enabled, a graceful shutdown is initiated if used space is above the threshold. The value should be between 6 and 99" )
53 ( "resource-monitor-not-shutdown-on-threshold-exceeded",
54 "Used to indicate nodeop will not shutdown when threshold is exceeded." )
55 ( "resource-monitor-warning-interval", bpo::value<uint32_t>()->default_value(def_monitor_warning_interval),
56 "Number of resource monitor intervals between two consecutive warnings when the threshold is hit. Should be between 1 and 450" )
57 ;
58 }
59
60 void plugin_initialize(const appbase::variables_map& options) {
61 dlog("plugin_initialize");
62
63 auto interval = options.at("resource-monitor-interval-seconds").as<uint32_t>();
64 SYS_ASSERT(interval >= monitor_interval_min && interval <= monitor_interval_max, chain::plugin_config_exception,
65 "\"resource-monitor-interval-seconds\" must be between ${monitor_interval_min} and ${monitor_interval_max}", ("monitor_interval_min", monitor_interval_min) ("monitor_interval_max", monitor_interval_max));
66 space_handler.set_sleep_time(interval);
67 ilog("Monitoring interval set to ${interval}", ("interval", interval));
68
69 auto threshold = options.at("resource-monitor-space-threshold").as<uint32_t>();
70 SYS_ASSERT(threshold >= space_threshold_min && threshold <= space_threshold_max, chain::plugin_config_exception,
71 "\"resource-monitor-space-threshold\" must be between ${space_threshold_min} and ${space_threshold_max}", ("space_threshold_min", space_threshold_min) ("space_threshold_max", space_threshold_max));
72 space_handler.set_threshold(threshold, threshold - space_threshold_warning_diff);
73 ilog("Space usage threshold set to ${threshold}", ("threshold", threshold));
74
75 if (options.count("resource-monitor-not-shutdown-on-threshold-exceeded")) {
76 // If set, not shutdown
77 space_handler.set_shutdown_on_exceeded(false);
78 ilog("Shutdown flag when threshold exceeded set to false");
79 } else {
80 // Default will shut down
81 space_handler.set_shutdown_on_exceeded(true);
82 ilog("Shutdown flag when threshold exceeded set to true");
83 }
84
85 auto warning_interval = options.at("resource-monitor-warning-interval").as<uint32_t>();
86 SYS_ASSERT(warning_interval >= warning_interval_min && warning_interval <= warning_interval_max, chain::plugin_config_exception,
87 "\"resource-monitor-warning-interval\" must be between ${warning_interval_min} and ${warning_interval_max}", ("warning_interval_min", warning_interval_min) ("warning_interval_max", warning_interval_max));
88 space_handler.set_warning_interval(warning_interval);
89 ilog("Warning interval set to ${warning_interval}", ("warning_interval", warning_interval));
90 }
91
92 // Start main thread
94 ilog("Creating and starting monitor thread");
95
96 // By now all plugins are initialized.
97 // Find out filesystems containing the directories requested
98 // so far.
99 for ( auto& dir: directories_registered ) {
100 space_handler.add_file_system( dir );
101
102 // A directory like "data" contains subdirectories like
103 // "block". Those subdirectories can mount on different
104 // file systems. Make sure they are taken care of.
105 for (bfs::directory_iterator itr(dir); itr != bfs::directory_iterator(); ++itr) {
106 if (fc::is_directory(itr->path())) {
107 space_handler.add_file_system( itr->path() );
108 }
109 }
110 }
111
112 monitor_thread = std::thread( [this] {
113 fc::set_os_thread_name( "resmon" ); // console_appender uses 9 chars for thread name reporting.
114 space_handler.space_monitor_loop();
115
116 ctx.run();
117 } );
118 }
119
120 // System is shutting down.
122 ilog("shutdown...");
123
124 ctx.stop();
125
126 // Wait for the thread to end
127 monitor_thread.join();
128
129 ilog("exit shutdown");
130 }
131
132 void monitor_directory(const bfs::path& path) {
133 dlog("${path} registered to be monitored", ("path", path.string()));
134 directories_registered.push_back(path);
135 }
136
137private:
138 std::thread monitor_thread;
139 std::vector<bfs::path> directories_registered;
140
141 static constexpr uint32_t def_interval_in_secs = 2;
142 static constexpr uint32_t monitor_interval_min = 1;
143 static constexpr uint32_t monitor_interval_max = 300;
144
145 static constexpr uint32_t def_space_threshold = 90; // in percentage
146 static constexpr uint32_t space_threshold_min = 6; // in percentage
147 static constexpr uint32_t space_threshold_max = 99; // in percentage
148 static constexpr uint32_t space_threshold_warning_diff = 5; // Warning issued when space used reached (threshold - space_threshold_warning_diff). space_threshold_warning_diff must be smaller than space_threshold_min
149
150 static constexpr uint32_t def_monitor_warning_interval = 30; // After this number of monitor intervals, warning is output if the threshold is hit
151 static constexpr uint32_t warning_interval_min = 1;
152 static constexpr uint32_t warning_interval_max = 450; // e.g. if the monitor interval is 2 sec, the warning interval is at most 15 minutes
153
154 boost::asio::io_context ctx;
155
156 using file_space_handler_t = file_space_handler<system_file_space_provider>;
157 file_space_handler_t space_handler;
158};
159
161
163
164void resource_monitor_plugin::set_program_options(options_description& cli, options_description& cfg) {
165 my->set_program_options(cli, cfg);
166}
167
168void resource_monitor_plugin::plugin_initialize(const variables_map& options) {
169 my->plugin_initialize(options);
170}
171
173 my->plugin_startup();
174}
175
177 my->plugin_shutdown();
178}
179
181 my->monitor_directory( path );
182}
183
184} // namespace
#define SYS_ASSERT(expr, exc_type, FORMAT,...)
Definition exceptions.hpp:7
wraps boost::filesystem::path to provide platform independent path manipulation.
std::string string() const
void set_threshold(uint32_t new_threshold, uint32_t new_warning_threshold)
void set_warning_interval(uint32_t new_warning_interval)
void add_file_system(const bfs::path &path_name)
void set_shutdown_on_exceeded(bool new_shutdown_on_exceeded)
void plugin_initialize(const appbase::variables_map &options)
void set_program_options(options_description &, options_description &cfg)
virtual void set_program_options(options_description &, options_description &cfg) override
void monitor_directory(const bfs::path &path)
void plugin_initialize(const variables_map &options)
Defines exception's used by fc.
#define dlog(FORMAT,...)
Definition logger.hpp:101
#define ilog(FORMAT,...)
Definition logger.hpp:118
application & app()
bool is_directory(const path &p)
void set_os_thread_name(const string &name)
Definition name.hpp:106
unsigned int uint32_t
Definition stdint.h:126
void cli()