-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathserver.cc
93 lines (71 loc) · 2.54 KB
/
server.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
/*
Copyright 2021 Simon (psyomn) Symeonidis
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
#include "dump.h"
#include "message.h"
#include "server.h"
#include "utils.h"
#include <fstream>
#include <locale>
namespace psy::psycal {
void Server::Start() {
auto constexpr max_sz = psy::net::kMaxUDPSize;
auto listen_fn = [this](std::uint8_t bf[max_sz]) noexcept -> void {
std::lock_guard guard(lock_);
for (const auto& received_event : Message::FromBuffer(bf, max_sz))
events_.push(std::move(received_event));
make_snapshot_ = true;
snapshot_cv_.notify_one();
};
/* load previous run's untriggered events */
std::ifstream is(Utils::EventsFilePath(), std::fstream::binary);
for (auto& el : ReadEventsFrom(is))
events_.push(std::move(el));
server_.ListenWith(listen_fn);
}
void Server::Tick() {
namespace sc = std::chrono;
while (run_server_) {
std::this_thread::sleep_for(sc::seconds(1));
std::lock_guard guard{lock_};
if (events_.size() == 0) continue;
const sc::time_point<sc::system_clock> now = sc::system_clock::now();
sc::system_clock::time_point event_timepoint{sc::seconds{events_.top().GetUnixTimestamp()}};
bool popped_event = false;
while (events_.size() > 0 && now > event_timepoint) {
old_.push_back(std::move(events_.top()));
events_.pop();
// TODO: libnotify or other hooks can/should be added here
popped_event = true;
// load next timepoint
event_timepoint =
sc::system_clock::time_point{
sc::seconds{events_.top().GetUnixTimestamp()}
};
}
if (popped_event) {
make_snapshot_ = true;
snapshot_cv_.notify_one();
}
}
}
void Server::Snapshot() {
while (run_server_) {
std::unique_lock<std::mutex> lk(lock_);
snapshot_cv_.wait(lk, [this]{return make_snapshot_ == true;});
SaveCurrentEvents(events_);
SaveOld(old_);
std::vector<Event>().swap(old_);
make_snapshot_ = false;
}
}
}