2013-12-16 06:29:26 +01:00
|
|
|
/** ==========================================================================
|
|
|
|
* 2013 by KjellKod.cc. This is PUBLIC DOMAIN to use at your own risk and comes
|
|
|
|
* with no warranties. This code is yours to share, use and modify with no
|
|
|
|
* strings attached and no restrictions or obligations.
|
|
|
|
* ============================================================================*/
|
|
|
|
|
2013-08-21 07:40:39 +02:00
|
|
|
#include <gtest/gtest.h>
|
|
|
|
#include <iostream>
|
|
|
|
#include <atomic>
|
|
|
|
#include <vector>
|
|
|
|
#include <memory>
|
|
|
|
#include <thread>
|
|
|
|
#include <chrono>
|
2014-02-12 04:00:20 +01:00
|
|
|
#include <string>
|
2013-11-04 05:46:19 +01:00
|
|
|
|
2013-08-21 07:40:39 +02:00
|
|
|
#include "testing_helpers.h"
|
2013-11-04 05:46:19 +01:00
|
|
|
#include "g2logmessage.hpp"
|
2013-10-05 06:14:35 +02:00
|
|
|
#include "g2logworker.hpp"
|
2014-02-12 04:00:20 +01:00
|
|
|
#include "std2_make_unique.hpp"
|
2013-07-14 01:57:26 +02:00
|
|
|
|
2013-08-21 07:40:39 +02:00
|
|
|
using namespace testing_helpers;
|
|
|
|
using namespace std;
|
|
|
|
TEST(Sink, OneSink) {
|
2013-11-12 10:21:40 +01:00
|
|
|
using namespace g2;
|
2013-11-04 05:46:19 +01:00
|
|
|
AtomicBoolPtr flag = make_shared < atomic<bool >> (false);
|
|
|
|
AtomicIntPtr count = make_shared < atomic<int >> (0);
|
|
|
|
{
|
2013-11-12 10:21:40 +01:00
|
|
|
auto worker = g2::LogWorker::createWithNoSink();
|
2013-11-04 05:46:19 +01:00
|
|
|
auto handle = worker->addSink(std2::make_unique<ScopedSetTrue>(flag, count), &ScopedSetTrue::ReceiveMsg);
|
|
|
|
EXPECT_FALSE(flag->load());
|
|
|
|
EXPECT_TRUE(0 == count->load());
|
2013-11-12 10:21:40 +01:00
|
|
|
LogMessagePtr message{std2::make_unique<LogMessage>("test", 0, "test", DEBUG)};
|
2013-11-15 10:02:30 +01:00
|
|
|
message.get()->write().append("this message should trigger an atomic increment at the sink");
|
2013-11-04 05:46:19 +01:00
|
|
|
worker->save(message);
|
|
|
|
}
|
|
|
|
EXPECT_TRUE(flag->load());
|
|
|
|
EXPECT_TRUE(1 == count->load());
|
2013-08-21 07:40:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-09-01 08:44:30 +02:00
|
|
|
namespace {
|
2013-11-04 05:46:19 +01:00
|
|
|
typedef std::shared_ptr<std::atomic<bool >> AtomicBoolPtr;
|
|
|
|
typedef std::shared_ptr<std::atomic<int >> AtomicIntPtr;
|
|
|
|
typedef vector<AtomicBoolPtr> BoolList;
|
|
|
|
typedef vector<AtomicIntPtr> IntVector;
|
2013-09-01 08:44:30 +02:00
|
|
|
}
|
2013-11-04 05:46:19 +01:00
|
|
|
|
2013-09-01 08:44:30 +02:00
|
|
|
TEST(ConceptSink, OneHundredSinks) {
|
2013-11-12 10:21:40 +01:00
|
|
|
using namespace g2;
|
2013-11-04 05:46:19 +01:00
|
|
|
BoolList flags;
|
|
|
|
IntVector counts;
|
|
|
|
|
2013-11-12 10:45:09 +01:00
|
|
|
size_t NumberOfItems = 100;
|
2013-11-04 05:46:19 +01:00
|
|
|
for (size_t index = 0; index < NumberOfItems; ++index) {
|
|
|
|
flags.push_back(make_shared < atomic<bool >> (false));
|
|
|
|
counts.push_back(make_shared < atomic<int >> (0));
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
2013-12-13 21:58:36 +01:00
|
|
|
RestoreFileLogger logger{"./"};
|
2013-11-12 10:21:40 +01:00
|
|
|
g2::LogWorker* worker = logger._scope->get(); //g2LogWorker::createWithNoSink();
|
2013-11-04 05:46:19 +01:00
|
|
|
size_t index = 0;
|
|
|
|
for (auto& flag : flags) {
|
|
|
|
auto& count = counts[index++];
|
|
|
|
// ignore the handle
|
|
|
|
worker->addSink(std2::make_unique<ScopedSetTrue>(flag, count), &ScopedSetTrue::ReceiveMsg);
|
|
|
|
}
|
|
|
|
LOG(DEBUG) << "start message";
|
2013-11-12 10:21:40 +01:00
|
|
|
LogMessagePtr message1{std2::make_unique<LogMessage>("test", 0, "test", DEBUG)};
|
|
|
|
LogMessagePtr message2{std2::make_unique<LogMessage>("test", 0, "test", DEBUG)};
|
2013-11-15 10:02:30 +01:00
|
|
|
auto& write1 = message1.get()->write();
|
|
|
|
write1.append("Hello to 100 receivers :)");
|
2013-11-12 10:21:40 +01:00
|
|
|
worker->save(message1);
|
|
|
|
|
2013-11-15 10:02:30 +01:00
|
|
|
auto& write2 = message2.get()->write();
|
|
|
|
write2.append("Hello to 100 receivers :)");
|
2013-11-12 10:21:40 +01:00
|
|
|
worker->save(message2);
|
2013-11-04 05:46:19 +01:00
|
|
|
LOG(INFO) << "end message";
|
2013-12-13 21:58:36 +01:00
|
|
|
logger.reset();
|
2013-11-04 05:46:19 +01:00
|
|
|
}
|
|
|
|
// at the curly brace above the ScopedLogger will go out of scope and all the
|
|
|
|
// 100 logging receivers will get their message to exit after all messages are
|
|
|
|
// are processed
|
2013-09-01 08:44:30 +02:00
|
|
|
size_t index = 0;
|
2013-11-04 05:46:19 +01:00
|
|
|
for (auto& flag : flags) {
|
2013-09-01 08:44:30 +02:00
|
|
|
auto& count = counts[index++];
|
2013-11-04 05:46:19 +01:00
|
|
|
ASSERT_TRUE(flag->load()) << ", count : " << (index - 1);
|
|
|
|
ASSERT_TRUE(4 == count->load()) << ", count : " << (index - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
cout << "test one hundred sinks is finished finished\n";
|
|
|
|
}
|
2014-02-12 04:00:20 +01:00
|
|
|
|
|
|
|
struct VoidReceiver {
|
|
|
|
std::atomic<int>* _atomicCounter;
|
|
|
|
explicit VoidReceiver(std::atomic<int>* counter) : _atomicCounter(counter){}
|
|
|
|
|
|
|
|
void receiveMsg(std::string msg){ /*ignored*/}
|
|
|
|
void incrementAtomic(){
|
|
|
|
(*_atomicCounter)++;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST(ConceptSink, VoidCall__NoCall_ExpectingNoAdd) {
|
|
|
|
std::atomic<int> counter{0};
|
|
|
|
{
|
|
|
|
std::unique_ptr<g2::LogWorker> worker{g2::LogWorker::createWithNoSink()};
|
|
|
|
auto handle = worker->addSink(std2::make_unique<VoidReceiver>(&counter), &VoidReceiver::receiveMsg);
|
|
|
|
}
|
|
|
|
EXPECT_EQ(counter, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ConceptSink, VoidCall__OneCall_ExpectingOneAdd) {
|
|
|
|
std::atomic<int> counter{0};
|
|
|
|
{
|
|
|
|
std::unique_ptr<g2::LogWorker> worker{g2::LogWorker::createWithNoSink()};
|
|
|
|
auto handle = worker->addSink(std2::make_unique<VoidReceiver>(&counter), &VoidReceiver::receiveMsg);
|
|
|
|
std::future<void> ignored = handle->call(&VoidReceiver::incrementAtomic);
|
|
|
|
}
|
|
|
|
EXPECT_EQ(counter, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(ConceptSink, VoidCall__TwoCalls_ExpectingTwoAdd) {
|
|
|
|
std::atomic<int> counter{0};
|
|
|
|
{
|
|
|
|
std::unique_ptr<g2::LogWorker> worker{g2::LogWorker::createWithNoSink()};
|
|
|
|
auto handle = worker->addSink(std2::make_unique<VoidReceiver>(&counter), &VoidReceiver::receiveMsg);
|
|
|
|
auto voidFuture1 = handle->call(&VoidReceiver::incrementAtomic);
|
|
|
|
auto voidFuture2 = handle->call(&VoidReceiver::incrementAtomic);
|
|
|
|
voidFuture1.wait();
|
|
|
|
EXPECT_TRUE(counter >= 1);
|
|
|
|
}
|
|
|
|
EXPECT_EQ(counter, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
struct IntReceiver {
|
|
|
|
std::atomic<int>* _atomicCounter;
|
|
|
|
explicit IntReceiver(std::atomic<int>* counter) : _atomicCounter(counter){}
|
|
|
|
|
|
|
|
void receiveMsg(std::string msg){ /*ignored*/}
|
|
|
|
int incrementAtomic(){
|
|
|
|
(*_atomicCounter)++;
|
|
|
|
int value = *_atomicCounter;
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
TEST(ConceptSink, IntCall__TwoCalls_ExpectingTwoAdd) {
|
|
|
|
std::atomic<int> counter{0};
|
|
|
|
{
|
|
|
|
std::unique_ptr<g2::LogWorker> worker{g2::LogWorker::createWithNoSink()};
|
|
|
|
auto handle = worker->addSink(std2::make_unique<IntReceiver>(&counter), &IntReceiver::receiveMsg);
|
|
|
|
std::future<int> intFuture1 = handle->call(&IntReceiver::incrementAtomic);
|
|
|
|
EXPECT_EQ(intFuture1.get(), 1);
|
|
|
|
EXPECT_EQ(counter, 1);
|
|
|
|
|
|
|
|
auto intFuture2 = handle->call(&IntReceiver::incrementAtomic);
|
|
|
|
EXPECT_EQ(intFuture2.get(), 2);
|
|
|
|
|
|
|
|
}
|
|
|
|
EXPECT_EQ(counter, 2);
|
|
|
|
}
|
|
|
|
|