Make work with string and number keys, cmake
This commit is contained in:
parent
da9a917945
commit
e659a78bd1
1
.gitignore
vendored
1
.gitignore
vendored
@ -0,0 +1 @@
|
|||||||
|
build/
|
52
.vscode/settings.json
vendored
52
.vscode/settings.json
vendored
@ -8,6 +8,56 @@
|
|||||||
"*.tcc": "cpp",
|
"*.tcc": "cpp",
|
||||||
"deque": "cpp",
|
"deque": "cpp",
|
||||||
"unordered_map": "cpp",
|
"unordered_map": "cpp",
|
||||||
"system_error": "cpp"
|
"system_error": "cpp",
|
||||||
|
"atomic": "cpp",
|
||||||
|
"shared_mutex": "cpp",
|
||||||
|
"mutex": "cpp",
|
||||||
|
"cctype": "cpp",
|
||||||
|
"clocale": "cpp",
|
||||||
|
"cmath": "cpp",
|
||||||
|
"cstdarg": "cpp",
|
||||||
|
"cstddef": "cpp",
|
||||||
|
"cstdio": "cpp",
|
||||||
|
"cstdlib": "cpp",
|
||||||
|
"ctime": "cpp",
|
||||||
|
"cwchar": "cpp",
|
||||||
|
"cwctype": "cpp",
|
||||||
|
"array": "cpp",
|
||||||
|
"bit": "cpp",
|
||||||
|
"chrono": "cpp",
|
||||||
|
"compare": "cpp",
|
||||||
|
"concepts": "cpp",
|
||||||
|
"condition_variable": "cpp",
|
||||||
|
"cstdint": "cpp",
|
||||||
|
"exception": "cpp",
|
||||||
|
"algorithm": "cpp",
|
||||||
|
"functional": "cpp",
|
||||||
|
"iterator": "cpp",
|
||||||
|
"memory": "cpp",
|
||||||
|
"memory_resource": "cpp",
|
||||||
|
"numeric": "cpp",
|
||||||
|
"optional": "cpp",
|
||||||
|
"random": "cpp",
|
||||||
|
"ratio": "cpp",
|
||||||
|
"string_view": "cpp",
|
||||||
|
"tuple": "cpp",
|
||||||
|
"type_traits": "cpp",
|
||||||
|
"utility": "cpp",
|
||||||
|
"fstream": "cpp",
|
||||||
|
"initializer_list": "cpp",
|
||||||
|
"iosfwd": "cpp",
|
||||||
|
"istream": "cpp",
|
||||||
|
"limits": "cpp",
|
||||||
|
"new": "cpp",
|
||||||
|
"numbers": "cpp",
|
||||||
|
"semaphore": "cpp",
|
||||||
|
"sstream": "cpp",
|
||||||
|
"stdexcept": "cpp",
|
||||||
|
"stop_token": "cpp",
|
||||||
|
"streambuf": "cpp",
|
||||||
|
"thread": "cpp",
|
||||||
|
"cinttypes": "cpp",
|
||||||
|
"typeinfo": "cpp",
|
||||||
|
"variant": "cpp"
|
||||||
}
|
}
|
||||||
}
|
}
|
30
CMakeLists.txt
Normal file
30
CMakeLists.txt
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
cmake_minimum_required(VERSION 3.10)
|
||||||
|
|
||||||
|
project(metrics)
|
||||||
|
|
||||||
|
# Postavi verziju projekta
|
||||||
|
set(CMAKE_CXX_STANDARD 17)
|
||||||
|
set(CMAKE_CXX_STANDARD_REQUIRED ON)
|
||||||
|
|
||||||
|
# Pronađi Boost biblioteku (ako nije uobičajeni direktorijum, postavi put)
|
||||||
|
# find_package(Boost REQUIRED COMPONENTS system)
|
||||||
|
|
||||||
|
# Dodaj direktorijume sa zaglavljima
|
||||||
|
include_directories(lib)
|
||||||
|
|
||||||
|
# Dodaj biblioteku
|
||||||
|
add_library(metrics STATIC
|
||||||
|
src/metrics.cpp
|
||||||
|
)
|
||||||
|
|
||||||
|
# # Linkaj log biblioteku sa Boost-om
|
||||||
|
# target_link_libraries(log Boost::system)
|
||||||
|
|
||||||
|
# Dodaj testove
|
||||||
|
add_subdirectory(test)
|
||||||
|
|
||||||
|
|
||||||
|
# Instaliraj biblioteku
|
||||||
|
# install(TARGETS log DESTINATION lib)
|
||||||
|
# install(FILES lib/log.hpp lib/define.hpp lib/engine.hpp lib/filesystem.hpp lib/timers.hpp lib/trigger.hpp DESTINATION include/log)
|
||||||
|
#
|
@ -27,7 +27,7 @@ using namespace marcelb;
|
|||||||
|
|
||||||
```c++
|
```c++
|
||||||
// init
|
// init
|
||||||
Metrics stats;
|
Metrics<string> stats;
|
||||||
// operator [] and increment ++
|
// operator [] and increment ++
|
||||||
stats["access"]++;
|
stats["access"]++;
|
||||||
// print couter access
|
// print couter access
|
||||||
@ -39,7 +39,7 @@ auto data = stats.get_data();
|
|||||||
// reset counters
|
// reset counters
|
||||||
stats.clear();
|
stats.clear();
|
||||||
// set counters
|
// set counters
|
||||||
map<string, uint> MyStats = {{"access", 3},{ "error", 0}};
|
map<string, uint64_t> MyStats = {{"access", 3},{ "error", 0}};
|
||||||
stats.set(MyStats);
|
stats.set(MyStats);
|
||||||
// get and clear all counters
|
// get and clear all counters
|
||||||
auto dataWithClear = stats.get_data_and_clear();
|
auto dataWithClear = stats.get_data_and_clear();
|
||||||
|
@ -6,77 +6,69 @@
|
|||||||
#include <vector>
|
#include <vector>
|
||||||
#include <map>
|
#include <map>
|
||||||
#include <mutex>
|
#include <mutex>
|
||||||
|
#include <shared_mutex>
|
||||||
|
#include <atomic>
|
||||||
|
|
||||||
using namespace std;
|
using namespace std;
|
||||||
|
|
||||||
namespace marcelb {
|
namespace marcelb {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A class for measuring arbitrary statistics
|
* A template class for measuring arbitrary statistics
|
||||||
*/
|
*/
|
||||||
|
template<typename K>
|
||||||
class Metrics {
|
class Metrics {
|
||||||
mutex io;
|
mutable shared_mutex mtx; // Mutex for concurrent read, exclusive write
|
||||||
map<string, uint> counters;
|
map<K, atomic<uint64_t>> counters; // Supports any K (e.g., string, int)
|
||||||
|
|
||||||
public:
|
|
||||||
|
|
||||||
|
public:
|
||||||
/**
|
/**
|
||||||
* Constructor, without predefining the name of the counter
|
* Constructor, without predefining the name of the counter
|
||||||
*/
|
*/
|
||||||
Metrics ();
|
Metrics();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Constructor, with predefined counters from the passed object
|
* Constructor, with predefined counters from the passed object
|
||||||
*/
|
*/
|
||||||
Metrics (map<string, uint> _counters);
|
Metrics(map<K, uint64_t> _counters);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Operator[] to access each measurement counter
|
* Operator[] to access each measurement counter
|
||||||
*/
|
*/
|
||||||
uint& operator[](const string& key) {
|
atomic<uint64_t>& operator[](const K& key);
|
||||||
return counters[key];
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Operator++ to increment the measurement counter incrementally
|
* Operator++ to increment the measurement counter incrementally
|
||||||
*/
|
*/
|
||||||
Metrics& operator++ (int n) {
|
Metrics& operator++(int n);
|
||||||
lock_guard<mutex> _io(io);
|
|
||||||
n++;
|
|
||||||
return *this;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Method to set the counter from the passed object
|
* Method to set the counter from the passed object
|
||||||
*/
|
*/
|
||||||
void set(map<string, uint> _counters) {
|
void set(map<K, uint64_t> _counters);
|
||||||
counters = _counters;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A method to reset the counter
|
* A method to reset the counter
|
||||||
*/
|
*/
|
||||||
void clear();
|
void clear();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A method that returns a vector os strings of all counter names
|
* A method that returns a vector of strings of all counter names
|
||||||
*/
|
*/
|
||||||
vector<string> keys();
|
vector<K> keys() const;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The method returns a map<string, uint> of all measurements
|
* The method returns a map<K, uint64_t> of all measurements
|
||||||
*/
|
*/
|
||||||
map<string, uint> get_data();
|
map<K, uint64_t> get_data() const;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The method returns a map<string, uint> of all measurements and resets the counters
|
* The method returns a map<K, uint64_t> of all measurements and resets the counters
|
||||||
*/
|
*/
|
||||||
map<string, uint> get_data_and_clear();
|
map<K, uint64_t> get_data_and_clear();
|
||||||
|
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
@ -1,35 +1,78 @@
|
|||||||
#include "../lib/metrics.hpp"
|
#include "../lib/metrics.hpp"
|
||||||
|
|
||||||
marcelb::Metrics::Metrics() {
|
namespace marcelb {
|
||||||
}
|
|
||||||
|
|
||||||
marcelb::Metrics::Metrics(map<string, uint> _counters) {
|
template<typename K>
|
||||||
|
Metrics<K>::Metrics() {}
|
||||||
|
|
||||||
|
template<typename K>
|
||||||
|
Metrics<K>::Metrics(map<K, uint64_t> _counters) {
|
||||||
set(_counters);
|
set(_counters);
|
||||||
}
|
}
|
||||||
|
|
||||||
void marcelb::Metrics::clear() {
|
template<typename K>
|
||||||
lock_guard<mutex> _io(io);
|
void Metrics<K>::clear() {
|
||||||
|
unique_lock<shared_mutex> lock(mtx);
|
||||||
counters.clear();
|
counters.clear();
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
vector<string> marcelb::Metrics::keys() {
|
template<typename K>
|
||||||
vector<string> _keys;
|
vector<K> Metrics<K>::keys() const {
|
||||||
for (auto counter : counters) {
|
shared_lock<shared_mutex> lock(mtx);
|
||||||
|
vector<K> _keys;
|
||||||
|
for (auto& counter : counters) {
|
||||||
_keys.push_back(counter.first);
|
_keys.push_back(counter.first);
|
||||||
}
|
}
|
||||||
|
|
||||||
return _keys;
|
return _keys;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<typename K>
|
||||||
map<string, uint> marcelb::Metrics::get_data() {
|
map<K, uint64_t> Metrics<K>::get_data() const {
|
||||||
return counters;
|
shared_lock<shared_mutex> lock(mtx);
|
||||||
|
map<K, uint64_t> data;
|
||||||
|
for (const auto& counter : counters) {
|
||||||
|
data[counter.first] = counter.second.load();
|
||||||
|
}
|
||||||
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
template<typename K>
|
||||||
map<string, uint> marcelb::Metrics::get_data_and_clear() {
|
map<K, uint64_t> Metrics<K>::get_data_and_clear() {
|
||||||
auto data = counters;
|
map<K, uint64_t> data;
|
||||||
|
{
|
||||||
|
shared_lock<shared_mutex> lock(mtx);
|
||||||
|
auto data = get_data();
|
||||||
|
}
|
||||||
clear();
|
clear();
|
||||||
return data;
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename K>
|
||||||
|
atomic<uint64_t>& Metrics<K>::operator[](const K& key) {
|
||||||
|
shared_lock<shared_mutex> lock(mtx);
|
||||||
|
return counters[key];
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename K>
|
||||||
|
Metrics<K>& Metrics<K>::operator++(int n) {
|
||||||
|
shared_lock<shared_mutex> lock(mtx);
|
||||||
|
// Increment all counters by n as an example
|
||||||
|
for (auto& counter : counters) {
|
||||||
|
counter.second.fetch_add(n, memory_order_relaxed);
|
||||||
|
}
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
template<typename K>
|
||||||
|
void Metrics<K>::set(map<K, uint64_t> _counters) {
|
||||||
|
unique_lock<shared_mutex> lock(mtx);
|
||||||
|
counters.clear();
|
||||||
|
for (auto& counter : _counters) {
|
||||||
|
counters[counter.first].store(counter.second);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template class Metrics<string>;
|
||||||
|
template class Metrics<uint32_t>;
|
||||||
|
|
||||||
}
|
}
|
3
test/CMakeLists.txt
Normal file
3
test/CMakeLists.txt
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
add_executable(metrics_test test.cpp)
|
||||||
|
|
||||||
|
target_link_libraries(metrics_test metrics)
|
@ -1,21 +1,54 @@
|
|||||||
|
#include <iostream>
|
||||||
|
#include <string>
|
||||||
|
#include <atomic>
|
||||||
|
|
||||||
|
using namespace std;
|
||||||
|
|
||||||
#include "../lib/metrics.hpp"
|
#include "../lib/metrics.hpp"
|
||||||
|
|
||||||
using namespace marcelb;
|
using namespace marcelb;
|
||||||
|
|
||||||
int main() {
|
int main() {
|
||||||
|
|
||||||
Metrics metrika;
|
// atomic<int> a(0);
|
||||||
|
|
||||||
|
// a++;
|
||||||
|
|
||||||
|
// cout << a << endl;
|
||||||
|
|
||||||
|
Metrics<string> metrika;
|
||||||
|
|
||||||
cout << " access " << metrika["access"]++ << endl;
|
cout << " access " << metrika["access"]++ << endl;
|
||||||
cout << " access " << metrika["access"]++ << endl;
|
cout << " access " << metrika["access"]++ << endl;
|
||||||
cout << " error " << metrika["error"] << endl;
|
cout << " error " << metrika["error"] << endl;
|
||||||
cout << " access " << metrika["access"] << endl;
|
cout << " access " << metrika["access"] << endl;
|
||||||
|
|
||||||
metrika.keys();
|
// metrika.keys();
|
||||||
|
|
||||||
cout << " error " << metrika["error"] << endl;
|
cout << " error " << metrika["error"] << endl;
|
||||||
cout << " access " << metrika["access"] << endl;
|
cout << " access " << metrika["access"] << endl;
|
||||||
|
|
||||||
|
Metrics<uint32_t> metrika2;
|
||||||
|
|
||||||
|
|
||||||
|
enum keys {
|
||||||
|
access = 1,
|
||||||
|
error
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
cout << " access " << metrika2[keys::access]++ << endl;
|
||||||
|
cout << " access " << metrika2[keys::access]++ << endl;
|
||||||
|
cout << " error " << metrika2[keys::error] << endl;
|
||||||
|
cout << " access " << metrika2[keys::access] << endl;
|
||||||
|
|
||||||
|
// metrika2.keys();
|
||||||
|
|
||||||
|
cout << " error " << metrika2[keys::error] << endl;
|
||||||
|
cout << " access " << metrika2[keys::access] << endl;
|
||||||
|
|
||||||
|
|
||||||
// // init
|
// // init
|
||||||
// Metrics stats;
|
// Metrics stats;
|
||||||
// // operator [] and increment ++
|
// // operator [] and increment ++
|
||||||
@ -29,7 +62,7 @@ int main() {
|
|||||||
// // reset counters
|
// // reset counters
|
||||||
// stats.clear();
|
// stats.clear();
|
||||||
// // set counters
|
// // set counters
|
||||||
// map<string, uint> MyStats = {{"access", 3},{ "error", 0}};
|
// map<string, uint64_t> MyStats = {{"access", 3},{ "error", 0}};
|
||||||
// stats.set(MyStats);
|
// stats.set(MyStats);
|
||||||
// // get and clear all counters
|
// // get and clear all counters
|
||||||
// auto dataWithClear = stats.get_data_and_clear();
|
// auto dataWithClear = stats.get_data_and_clear();
|
||||||
|
Loading…
x
Reference in New Issue
Block a user