Skip to content

Commit

Permalink
CONFIG
Browse files Browse the repository at this point in the history
  • Loading branch information
pwojcikdev committed Feb 10, 2024
1 parent 5bdbd22 commit f1b3569
Show file tree
Hide file tree
Showing 6 changed files with 91 additions and 109 deletions.
16 changes: 6 additions & 10 deletions nano/core_test/toml.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -222,12 +222,10 @@ TEST (toml, daemon_config_deserialize_defaults)
ASSERT_EQ (conf.node.diagnostics_config.txn_tracking.min_read_txn_time, defaults.node.diagnostics_config.txn_tracking.min_read_txn_time);
ASSERT_EQ (conf.node.diagnostics_config.txn_tracking.min_write_txn_time, defaults.node.diagnostics_config.txn_tracking.min_write_txn_time);

ASSERT_EQ (conf.node.stats_config.sampling_enabled, defaults.node.stats_config.sampling_enabled);
ASSERT_EQ (conf.node.stats_config.interval, defaults.node.stats_config.interval);
ASSERT_EQ (conf.node.stats_config.capacity, defaults.node.stats_config.capacity);
ASSERT_EQ (conf.node.stats_config.max_samples, defaults.node.stats_config.max_samples);
ASSERT_EQ (conf.node.stats_config.log_rotation_count, defaults.node.stats_config.log_rotation_count);
ASSERT_EQ (conf.node.stats_config.log_interval_samples, defaults.node.stats_config.log_interval_samples);
ASSERT_EQ (conf.node.stats_config.log_interval_counters, defaults.node.stats_config.log_interval_counters);
ASSERT_EQ (conf.node.stats_config.log_samples_interval, defaults.node.stats_config.log_samples_interval);
ASSERT_EQ (conf.node.stats_config.log_counters_interval, defaults.node.stats_config.log_counters_interval);
ASSERT_EQ (conf.node.stats_config.log_headers, defaults.node.stats_config.log_headers);
ASSERT_EQ (conf.node.stats_config.log_counters_filename, defaults.node.stats_config.log_counters_filename);
ASSERT_EQ (conf.node.stats_config.log_samples_filename, defaults.node.stats_config.log_samples_filename);
Expand Down Expand Up @@ -641,12 +639,10 @@ TEST (toml, daemon_config_deserialize_no_defaults)
ASSERT_NE (conf.node.diagnostics_config.txn_tracking.min_read_txn_time, defaults.node.diagnostics_config.txn_tracking.min_read_txn_time);
ASSERT_NE (conf.node.diagnostics_config.txn_tracking.min_write_txn_time, defaults.node.diagnostics_config.txn_tracking.min_write_txn_time);

ASSERT_NE (conf.node.stats_config.sampling_enabled, defaults.node.stats_config.sampling_enabled);
ASSERT_NE (conf.node.stats_config.interval, defaults.node.stats_config.interval);
ASSERT_NE (conf.node.stats_config.capacity, defaults.node.stats_config.capacity);
ASSERT_NE (conf.node.stats_config.max_samples, defaults.node.stats_config.max_samples);
ASSERT_NE (conf.node.stats_config.log_rotation_count, defaults.node.stats_config.log_rotation_count);
ASSERT_NE (conf.node.stats_config.log_interval_samples, defaults.node.stats_config.log_interval_samples);
ASSERT_NE (conf.node.stats_config.log_interval_counters, defaults.node.stats_config.log_interval_counters);
ASSERT_NE (conf.node.stats_config.log_samples_interval, defaults.node.stats_config.log_samples_interval);
ASSERT_NE (conf.node.stats_config.log_counters_interval, defaults.node.stats_config.log_counters_interval);
ASSERT_NE (conf.node.stats_config.log_headers, defaults.node.stats_config.log_headers);
ASSERT_NE (conf.node.stats_config.log_counters_filename, defaults.node.stats_config.log_counters_filename);
ASSERT_NE (conf.node.stats_config.log_samples_filename, defaults.node.stats_config.log_samples_filename);
Expand Down
155 changes: 71 additions & 84 deletions nano/lib/stats.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -10,59 +10,6 @@
#include <fstream>
#include <sstream>

/*
* stats_config
*/

nano::error nano::stats_config::deserialize_toml (nano::tomlconfig & toml)
{
auto sampling_l (toml.get_optional_child ("sampling"));
if (sampling_l)
{
sampling_l->get<bool> ("enable", sampling_enabled);
sampling_l->get<size_t> ("capacity", capacity);
sampling_l->get<size_t> ("interval", interval);
}

auto log_l (toml.get_optional_child ("log"));
if (log_l)
{
log_l->get<bool> ("headers", log_headers);
log_l->get<size_t> ("interval_counters", log_interval_counters);
log_l->get<size_t> ("interval_samples", log_interval_samples);
log_l->get<size_t> ("rotation_count", log_rotation_count);
log_l->get<std::string> ("filename_counters", log_counters_filename);
log_l->get<std::string> ("filename_samples", log_samples_filename);

// Don't allow specifying the same file name for counter and samples logs
if (log_counters_filename == log_samples_filename)
{
toml.get_error ().set ("The statistics counter and samples config values must be different");
}
}

return toml.get_error ();
}

nano::error nano::stats_config::serialize_toml (nano::tomlconfig & toml) const
{
nano::tomlconfig sampling_l;
sampling_l.put ("enable", sampling_enabled, "Enable or disable sampling.\ntype:bool");
sampling_l.put ("capacity", capacity, "How many sample intervals to keep in the ring buffer.\ntype:uint64");
sampling_l.put ("interval", interval, "Sample interval.\ntype:milliseconds");
toml.put_child ("sampling", sampling_l);

nano::tomlconfig log_l;
log_l.put ("headers", log_headers, "If true, write headers on each counter or samples writeout.\nThe header contains log type and the current wall time.\ntype:bool");
log_l.put ("interval_counters", log_interval_counters, "How often to log counters. 0 disables logging.\ntype:milliseconds");
log_l.put ("interval_samples", log_interval_samples, "How often to log samples. 0 disables logging.\ntype:milliseconds");
log_l.put ("rotation_count", log_rotation_count, "Maximum number of log outputs before rotating the file.\ntype:uint64");
log_l.put ("filename_counters", log_counters_filename, "Log file name for counters.\ntype:string");
log_l.put ("filename_samples", log_samples_filename, "Log file name for samples.\ntype:string");
toml.put_child ("log", log_l);
return toml.get_error ();
}

/*
* stat_log_sink
*/
Expand Down Expand Up @@ -179,7 +126,7 @@ class file_writer : public nano::stat_log_sink
*/

nano::stats::stats (nano::stats_config config) :
config (config)
config{ std::move (config) }
{
}

Expand Down Expand Up @@ -222,7 +169,7 @@ auto nano::stats::count (stat::type type, stat::detail detail, stat::dir dir) co
void nano::stats::sample (stat::sample type, nano::stats::sampler_value_t value)
{
update_sampler (sampler_key{ type }, [this, value] (sampler_entry & sampler) {
sampler.add (value, config.capacity);
sampler.add (value, config.max_samples);
});
}

Expand Down Expand Up @@ -380,41 +327,29 @@ void nano::stats::update ()
std::lock_guard guard{ mutex };
if (!stopped)
{
auto has_interval_counter = [&] () {
return config.log_interval_counters > 0;
};
auto has_sampling = [&] () {
return config.sampling_enabled && config.interval > 0;
};

if (has_interval_counter () || has_sampling ())
{
auto now = std::chrono::steady_clock::now (); // Only sample clock if necessary as this impacts node performance due to frequent usage
auto now = std::chrono::steady_clock::now (); // Only sample clock if necessary as this impacts node performance due to frequent usage

// TODO: Replace with a proper std::chrono time
std::time_t time = std::chrono::system_clock::to_time_t (std::chrono::system_clock::now ());
tm local_tm = *localtime (&time);
// TODO: Replace with a proper std::chrono time
std::time_t time = std::chrono::system_clock::to_time_t (std::chrono::system_clock::now ());
tm local_tm = *localtime (&time);

// Counters
if (has_interval_counter ())
// Counters
if (config.log_counters_interval.count () > 0)
{
if (nano::elapsed (log_last_count_writeout, config.log_counters_interval))
{
std::chrono::duration<double, std::milli> duration = now - log_last_count_writeout;
if (duration.count () > config.log_interval_counters)
{
log_counters_impl (log_count, local_tm);
log_last_count_writeout = now;
}
log_counters_impl (log_count, local_tm);
log_last_count_writeout = now;
}
}

// Samples
if (has_sampling ())
// Samples
if (config.log_samples_interval.count () > 0)
{
if (nano::elapsed (log_last_sample_writeout, config.log_samples_interval))
{
std::chrono::duration<double, std::milli> duration = now - log_last_sample_writeout;
if (duration.count () > config.log_interval_samples)
{
log_samples_impl (log_sample, local_tm);
log_last_sample_writeout = now;
}
log_samples_impl (log_sample, local_tm);
log_last_sample_writeout = now;
}
}
}
Expand Down Expand Up @@ -465,4 +400,56 @@ auto nano::stats::sampler_entry::collect () -> std::vector<sampler_value_t>
std::vector<sampler_value_t> result{ samples.begin (), samples.end () };
samples.clear ();
return result;
}

/*
* stats_config
*/

nano::error nano::stats_config::serialize_toml (nano::tomlconfig & toml) const
{
toml.put ("max_samples", max_samples, "Maximum number of samples to keep in the ring buffer.\ntype:uint64");

nano::tomlconfig log_l;
log_l.put ("headers", log_headers, "If true, write headers on each counter or samples writeout.\nThe header contains log type and the current wall time.\ntype:bool");
log_l.put ("interval_counters", log_counters_interval.count (), "How often to log counters. 0 disables logging.\ntype:milliseconds");
log_l.put ("interval_samples", log_samples_interval.count (), "How often to log samples. 0 disables logging.\ntype:milliseconds");
log_l.put ("rotation_count", log_rotation_count, "Maximum number of log outputs before rotating the file.\ntype:uint64");
log_l.put ("filename_counters", log_counters_filename, "Log file name for counters.\ntype:string");
log_l.put ("filename_samples", log_samples_filename, "Log file name for samples.\ntype:string");
toml.put_child ("log", log_l);

return toml.get_error ();
}

nano::error nano::stats_config::deserialize_toml (nano::tomlconfig & toml)
{
toml.get ("max_samples", max_samples);

if (auto maybe_log_l = toml.get_optional_child ("log"))
{
auto log_l = *maybe_log_l;

log_l.get ("headers", log_headers);

auto counters_interval_l = log_counters_interval.count ();
log_l.get ("interval_counters", counters_interval_l);
log_counters_interval = std::chrono::milliseconds{ counters_interval_l };

auto samples_interval_l = log_samples_interval.count ();
log_l.get ("interval_samples", samples_interval_l);
log_samples_interval = std::chrono::milliseconds{ samples_interval_l };

log_l.get ("rotation_count", log_rotation_count);
log_l.get ("filename_counters", log_counters_filename);
log_l.get ("filename_samples", log_samples_filename);

// Don't allow specifying the same file name for counter and samples logs
if (log_counters_filename == log_samples_filename)
{
toml.get_error ().set ("The statistics counter and samples config values must be different");
}
}

return toml.get_error ();
}
21 changes: 8 additions & 13 deletions nano/lib/stats.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -33,20 +33,15 @@ class stats_config final
nano::error deserialize_toml (nano::tomlconfig & toml);
nano::error serialize_toml (nano::tomlconfig & toml) const;

/** If true, sampling of counters is enabled */
bool sampling_enabled{ false };

/** How many sample intervals to keep in the ring buffer */
size_t capacity{ 0 };

/** Sample interval in milliseconds */
size_t interval{ 0 };
public:
/** Maximum number samples to keep in the ring buffer */
size_t max_samples{ 1024 * 16 };

/** How often to log sample array, in milliseconds. Default is 0 (no logging) */
size_t log_interval_samples{ 0 };
std::chrono::milliseconds log_samples_interval{ 0 };

/** How often to log counters, in milliseconds. Default is 0 (no logging) */
size_t log_interval_counters{ 0 };
std::chrono::milliseconds log_counters_interval{ 0 };

/** Maximum number of log outputs before rotating the file */
size_t log_rotation_count{ 100 };
Expand Down Expand Up @@ -220,12 +215,12 @@ class stats final
/** Unlocked implementation of log_samples() to avoid using recursive locking */
void log_samples_impl (stat_log_sink & sink, tm & tm);

private:
nano::stats_config const config;

/** Time of last clear() call */
std::chrono::steady_clock::time_point timestamp{ std::chrono::steady_clock::now () };

/** Configuration deserialized from config.json */
nano::stats_config config;

std::chrono::steady_clock::time_point log_last_count_writeout{ std::chrono::steady_clock::now () };
std::chrono::steady_clock::time_point log_last_sample_writeout{ std::chrono::steady_clock::now () };

Expand Down
1 change: 1 addition & 0 deletions nano/lib/tomlconfig.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ class tomlconfig : public nano::configbase
public:
tomlconfig ();
tomlconfig (std::shared_ptr<cpptoml::table> const & tree_a, std::shared_ptr<nano::error> const & error_a = nullptr);

void doc (std::string const & key, std::string const & doc);
nano::error & read (std::filesystem::path const & path_a);
nano::error & read (std::istream & stream_overrides, std::filesystem::path const & path_a);
Expand Down
6 changes: 4 additions & 2 deletions nano/lib/utility.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -197,18 +197,20 @@ using clock = std::chrono::steady_clock;

/**
* Check whether time elapsed between `last` and `now` is greater than `duration`
* Force usage of steady clock
*/
template <typename Duration>
bool elapsed (nano::clock::time_point const & last, Duration duration, nano::clock::time_point const & now)
bool elapsed (nano::clock::time_point const & last, Duration const & duration, nano::clock::time_point const & now)
{
return last + duration < now;
}

/**
* Check whether time elapsed since `last` is greater than `duration`
* Force usage of steady clock
*/
template <typename Duration>
bool elapsed (nano::clock::time_point const & last, Duration duration)
bool elapsed (nano::clock::time_point const & last, Duration const & duration)
{
return elapsed (last, duration, nano::clock::now ());
}
Expand Down
1 change: 1 addition & 0 deletions nano/node/json_handler.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -3958,6 +3958,7 @@ void nano::json_handler::stats ()
}
if (!ec && use_sink)
{
// TODO: Clearly someone gave up on designing this properly here
auto stat_tree_l (*static_cast<boost::property_tree::ptree *> (sink->to_object ()));
stat_tree_l.put ("stat_duration_seconds", node.stats.last_reset ().count ());
std::stringstream ostream;
Expand Down

0 comments on commit f1b3569

Please sign in to comment.