Add deletion, more stats to card fraud and RWLock
Summary:
^^
this is a sample config to be used:
```
{
"num_workers": 1,
"cards_per_worker": 10001,
"pos_per_worker": 10001,
"fraud_probability": 0.01,
"hop_probability": 0.05,
"cleanup": {
"check_interval_sec": 10,
"tx_hi": 150000,
"tx_lo": 100000
},
"analytic": {
"query_interval_ms": 500,
"pos_limit": 10
}
}
```
I also added `RWLock` --- a wrapper around `pthread_rwlock`
Reviewers: mferencevic, mculinovic, florijan, teon.banek
Reviewed By: florijan
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1280
2018-03-14 20:47:18 +08:00
|
|
|
#include <shared_mutex>
|
|
|
|
#include <thread>
|
|
|
|
|
|
|
|
#include "glog/logging.h"
|
|
|
|
#include "gtest/gtest.h"
|
|
|
|
|
2018-05-30 19:00:25 +08:00
|
|
|
#include "utils/thread/sync.hpp"
|
Add deletion, more stats to card fraud and RWLock
Summary:
^^
this is a sample config to be used:
```
{
"num_workers": 1,
"cards_per_worker": 10001,
"pos_per_worker": 10001,
"fraud_probability": 0.01,
"hop_probability": 0.05,
"cleanup": {
"check_interval_sec": 10,
"tx_hi": 150000,
"tx_lo": 100000
},
"analytic": {
"query_interval_ms": 500,
"pos_limit": 10
}
}
```
I also added `RWLock` --- a wrapper around `pthread_rwlock`
Reviewers: mferencevic, mculinovic, florijan, teon.banek
Reviewed By: florijan
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1280
2018-03-14 20:47:18 +08:00
|
|
|
#include "utils/timer.hpp"
|
|
|
|
|
|
|
|
using namespace std::chrono_literals;
|
|
|
|
|
2018-05-30 19:00:25 +08:00
|
|
|
using utils::RWLock;
|
|
|
|
using utils::RWLockPriority;
|
Add deletion, more stats to card fraud and RWLock
Summary:
^^
this is a sample config to be used:
```
{
"num_workers": 1,
"cards_per_worker": 10001,
"pos_per_worker": 10001,
"fraud_probability": 0.01,
"hop_probability": 0.05,
"cleanup": {
"check_interval_sec": 10,
"tx_hi": 150000,
"tx_lo": 100000
},
"analytic": {
"query_interval_ms": 500,
"pos_limit": 10
}
}
```
I also added `RWLock` --- a wrapper around `pthread_rwlock`
Reviewers: mferencevic, mculinovic, florijan, teon.banek
Reviewed By: florijan
Subscribers: pullbot
Differential Revision: https://phabricator.memgraph.io/D1280
2018-03-14 20:47:18 +08:00
|
|
|
|
|
|
|
TEST(RWLock, MultipleReaders) {
|
|
|
|
RWLock rwlock(RWLockPriority::READ);
|
|
|
|
|
|
|
|
std::vector<std::thread> threads;
|
|
|
|
utils::Timer timer;
|
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
threads.push_back(std::thread([&rwlock] {
|
|
|
|
std::shared_lock<RWLock> lock(rwlock);
|
|
|
|
std::this_thread::sleep_for(100ms);
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
threads[i].join();
|
|
|
|
}
|
|
|
|
|
|
|
|
EXPECT_LE(timer.Elapsed(), 150ms);
|
|
|
|
EXPECT_GE(timer.Elapsed(), 90ms);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(RWLock, SingleWriter) {
|
|
|
|
RWLock rwlock(RWLockPriority::READ);
|
|
|
|
|
|
|
|
std::vector<std::thread> threads;
|
|
|
|
utils::Timer timer;
|
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
threads.push_back(std::thread([&rwlock] {
|
|
|
|
std::unique_lock<RWLock> lock(rwlock);
|
|
|
|
std::this_thread::sleep_for(100ms);
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
threads[i].join();
|
|
|
|
}
|
|
|
|
|
|
|
|
EXPECT_LE(timer.Elapsed(), 350ms);
|
|
|
|
EXPECT_GE(timer.Elapsed(), 290ms);
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(RWLock, ReadPriority) {
|
|
|
|
/*
|
|
|
|
* - Main thread is holding a shared lock until T = 100ms.
|
|
|
|
* - Thread 1 tries to acquire an unique lock at T = 30ms.
|
|
|
|
* - Thread 2 successfuly acquires a shared lock at T = 60ms, even though
|
|
|
|
* there's a writer waiting.
|
|
|
|
*/
|
|
|
|
RWLock rwlock(RWLockPriority::READ);
|
|
|
|
rwlock.lock_shared();
|
|
|
|
bool first = true;
|
|
|
|
|
|
|
|
std::thread t1([&rwlock, &first] {
|
|
|
|
std::this_thread::sleep_for(30ms);
|
|
|
|
std::unique_lock<RWLock> lock(rwlock);
|
|
|
|
EXPECT_FALSE(first);
|
|
|
|
});
|
|
|
|
|
|
|
|
std::thread t2([&rwlock, &first] {
|
|
|
|
std::this_thread::sleep_for(60ms);
|
|
|
|
std::shared_lock<RWLock> lock(rwlock);
|
|
|
|
EXPECT_TRUE(first);
|
|
|
|
first = false;
|
|
|
|
});
|
|
|
|
|
|
|
|
std::this_thread::sleep_for(100ms);
|
|
|
|
rwlock.unlock_shared();
|
|
|
|
t1.join();
|
|
|
|
t2.join();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(RWLock, WritePriority) {
|
|
|
|
/*
|
|
|
|
* - Main thread is holding a shared lock until T = 100ms.
|
|
|
|
* - Thread 1 tries to acquire an unique lock at T = 30ms.
|
|
|
|
* - Thread 2 tries to acquire a shared lock at T = 60ms, but it is not able
|
|
|
|
* to because of write priority.
|
|
|
|
*/
|
|
|
|
RWLock rwlock(RWLockPriority::WRITE);
|
|
|
|
rwlock.lock_shared();
|
|
|
|
bool first = true;
|
|
|
|
|
|
|
|
std::thread t1([&rwlock, &first] {
|
|
|
|
std::this_thread::sleep_for(30ms);
|
|
|
|
std::unique_lock<RWLock> lock(rwlock);
|
|
|
|
EXPECT_TRUE(first);
|
|
|
|
first = false;
|
|
|
|
});
|
|
|
|
|
|
|
|
std::thread t2([&rwlock, &first] {
|
|
|
|
std::this_thread::sleep_for(60ms);
|
|
|
|
std::shared_lock<RWLock> lock(rwlock);
|
|
|
|
EXPECT_FALSE(first);
|
|
|
|
});
|
|
|
|
|
|
|
|
std::this_thread::sleep_for(100ms);
|
|
|
|
rwlock.unlock_shared();
|
|
|
|
|
|
|
|
t1.join();
|
|
|
|
t2.join();
|
|
|
|
}
|
|
|
|
|
|
|
|
TEST(RWLock, TryLock) {
|
|
|
|
RWLock rwlock(RWLockPriority::WRITE);
|
|
|
|
rwlock.lock();
|
|
|
|
|
|
|
|
std::thread t1([&rwlock] { EXPECT_FALSE(rwlock.try_lock()); });
|
|
|
|
t1.join();
|
|
|
|
|
|
|
|
std::thread t2([&rwlock] { EXPECT_FALSE(rwlock.try_lock_shared()); });
|
|
|
|
t2.join();
|
|
|
|
|
|
|
|
rwlock.unlock();
|
|
|
|
|
|
|
|
std::thread t3([&rwlock] {
|
|
|
|
EXPECT_TRUE(rwlock.try_lock());
|
|
|
|
rwlock.unlock();
|
|
|
|
});
|
|
|
|
t3.join();
|
|
|
|
|
|
|
|
std::thread t4([&rwlock] {
|
|
|
|
EXPECT_TRUE(rwlock.try_lock_shared());
|
|
|
|
rwlock.unlock_shared();
|
|
|
|
});
|
|
|
|
t4.join();
|
|
|
|
|
|
|
|
rwlock.lock_shared();
|
|
|
|
|
|
|
|
std::thread t5([&rwlock] {
|
|
|
|
EXPECT_TRUE(rwlock.try_lock_shared());
|
|
|
|
rwlock.unlock_shared();
|
|
|
|
});
|
|
|
|
t5.join();
|
|
|
|
}
|