0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2025-01-16 17:46:54 +01:00
construct/include/ircd/stats.h

277 lines
6 KiB
C
Raw Normal View History

// The Construct
//
// Copyright (C) The Construct Developers, Authors & Contributors
// Copyright (C) 2016-2020 Jason Volk <jason@zemos.net>
//
// Permission to use, copy, modify, and/or distribute this software for any
// purpose with or without fee is hereby granted, provided that the above
// copyright notice and this permission notice is present in all copies. The
// full license for this software is available in the LICENSE file.
#pragma once
#define HAVE_IRCD_STATS_H
/// Statistics & Metrics
///
/// This is a central collection of registered items each representing a
/// counter or metric of some kind. To collect items of various types we orient
/// the collection around an abstract item<void> template class. To keep things
/// simple, the abstract instance holds a typeinfo provided by the derived
/// instance. User can then downcast the item<void> to a derived template.
///
/// There are two levels in the class hierarchy under the abstract root
/// item<void>. The next level is a pointer-to-value such that the item
/// registers the location of an existing value. This is useful for
/// incorporating external values in existing structures into this collection
/// non-intrusively; for example in third-party libraries etc.
///
/// The next derived level after this is where the item instance itself
/// roots (contains) the value and its parent class at the pointer-level points
/// down to the derived class. This is a convenience for developers so that
/// extern values don't have to be created separately for every stats item.
/// Note that when this subsystem works abstractly with items, it considers the
/// pointer-level templates to be the principal level of derivation; in other
/// words there is no reason to downcast to a value-level template when working
/// with this system, and every value-level template must have a matching
/// pointer-level parent.
namespace ircd::stats
{
IRCD_EXCEPTION(ircd::error, error)
IRCD_EXCEPTION(error, not_found)
IRCD_PANICKING(error, invalid)
// Abstract item
template<class T = void> struct item;
template<> struct item<void>;
// Pointer-to-value items
template<> struct item<uint64_t *>;
template<> struct item<uint32_t *>;
template<> struct item<uint16_t *>;
// Value-carrying items
template<> struct item<uint64_t>;
template<> struct item<uint32_t>;
template<> struct item<uint16_t>;
extern const size_t NAME_MAX_LEN;
extern std::vector<item<void> *> items;
string_view string(const mutable_buffer &, const item<void> &);
std::ostream &operator<<(std::ostream &, const item<void> &);
}
/// Abstract stats item.
///
/// This object contains type information about its derived class. There is no
/// known use for constructing this on its own without a derived item.
///
/// Feature information must contain a 'name' string. It is advised that this
/// is appropriately namespaced i.e "ircd.net.socket.xxx" and when third-party
/// values are gathered i.e "rocksdb.xxx." such that the entire map of items
/// can be serialized into a single JSON object tree.
///
/// Feature information can also contain a 'desc' string describing more about
/// the value to administrators and developers.
template<>
struct ircd::stats::item<void>
{
std::type_index type {typeid(void)};
json::strung feature;
json::string name;
public:
// Access features
string_view operator[](const string_view &key) const noexcept;
virtual bool operator!() const = 0;
item() = default;
item(const std::type_index &, const json::members &);
item(item &&) = delete;
item(const item &) = delete;
virtual ~item() noexcept;
};
template<>
struct ircd::stats::item<uint64_t *>
:item<void>
{
uint64_t *val {nullptr};
public:
bool operator!() const override
{
return !val || !*val;
}
operator const uint64_t &() const
{
assert(val);
return *val;
}
operator uint64_t &()
{
assert(val);
return *val;
}
item(uint64_t *const &, const json::members &);
item() = default;
item &operator=(const uint64_t &val) &
{
static_cast<uint64_t &>(*this) = val;
return *this;
}
};
template<>
struct ircd::stats::item<uint32_t *>
:item<void>
{
uint32_t *val {nullptr};
public:
bool operator!() const override
{
return !val || !*val;
}
operator const uint32_t &() const
{
assert(val);
return *val;
}
operator uint32_t &()
{
assert(val);
return *val;
}
item(uint32_t *const &, const json::members &);
item() = default;
item &operator=(const uint32_t &val) &
{
static_cast<uint32_t &>(*this) = val;
return *this;
}
};
template<>
struct ircd::stats::item<uint16_t *>
:item<void>
{
uint16_t *val {nullptr};
public:
bool operator!() const override
{
return !val || !*val;
}
operator const uint16_t &() const
{
assert(val);
return *val;
}
operator uint16_t &()
{
assert(val);
return *val;
}
item(uint16_t *const &, const json::members &);
item() = default;
item &operator=(const uint16_t &val) &
{
static_cast<uint16_t &>(*this) = val;
return *this;
}
};
template<> struct ircd::stats::item<uint64_t>
:item<uint64_t *>
{
uint64_t val {0};
public:
operator const uint64_t &() const noexcept
{
return val;
}
operator uint64_t &() noexcept
{
return val;
}
item(const json::members &);
item() = default;
item &operator=(const uint64_t &val) &
{
static_cast<uint64_t &>(*this) = val;
return *this;
}
};
template<>
struct ircd::stats::item<uint32_t>
:item<uint32_t *>
{
uint32_t val {0};
public:
operator const uint32_t &() const noexcept
{
return val;
}
operator uint32_t &() noexcept
{
return val;
}
item(const json::members &);
item() = default;
item &operator=(const uint32_t &val) &
{
static_cast<uint32_t &>(*this) = val;
return *this;
}
};
template<>
struct ircd::stats::item<uint16_t>
:item<uint16_t *>
{
uint16_t val {0};
public:
operator const uint16_t &() const noexcept
{
return val;
}
operator uint16_t &() noexcept
{
return val;
}
item(const json::members &);
item() = default;
item &operator=(const uint16_t &val) &
{
static_cast<uint16_t &>(*this) = val;
return *this;
}
};