Added basic UDP handling - only sending from client supported.
This commit is contained in:
88
include/columnlynx/common/net/session_registry.hpp
Normal file
88
include/columnlynx/common/net/session_registry.hpp
Normal file
@@ -0,0 +1,88 @@
|
||||
// session_registry.hpp - Session Registry for ColumnLynx
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the GPLv3 license. See LICENSE for details.
|
||||
|
||||
#pragma once
|
||||
#include <unordered_set>
|
||||
#include <shared_mutex>
|
||||
#include <memory>
|
||||
#include <chrono>
|
||||
#include <array>
|
||||
#include <sodium.h>
|
||||
#include <columnlynx/common/utils.hpp>
|
||||
#include <columnlynx/common/libsodium_wrapper.hpp>
|
||||
|
||||
namespace ColumnLynx::Net {
|
||||
struct SessionState {
|
||||
SymmetricKey aesKey; // Immutable after creation
|
||||
std::atomic<uint64_t> send_ctr{0}; // Per-direction counters
|
||||
std::atomic<uint64_t> recv_ctr{0};
|
||||
asio::ip::udp::endpoint udpEndpoint;
|
||||
std::atomic<uint64_t> sendCounter{0};
|
||||
std::chrono::steady_clock::time_point created = std::chrono::steady_clock::now();
|
||||
std::chrono::steady_clock::time_point expires{};
|
||||
Nonce base_nonce{};
|
||||
|
||||
~SessionState() { sodium_memzero(aesKey.data(), aesKey.size()); }
|
||||
SessionState(const SessionState&) = delete;
|
||||
SessionState& operator=(const SessionState&) = delete;
|
||||
SessionState(SessionState&&) = default;
|
||||
SessionState& operator=(SessionState&&) = default;
|
||||
|
||||
explicit SessionState(const SymmetricKey& k, std::chrono::seconds ttl = std::chrono::hours(24)) : aesKey(k) {
|
||||
expires = created + ttl;
|
||||
}
|
||||
|
||||
void setUDPEndpoint(const asio::ip::udp::endpoint& ep) {
|
||||
udpEndpoint = ep;
|
||||
}
|
||||
};
|
||||
|
||||
class SessionRegistry {
|
||||
public:
|
||||
static SessionRegistry& getInstance() { static SessionRegistry instance; return instance; }
|
||||
|
||||
// Insert or replace
|
||||
void put(uint64_t sessionID, std::shared_ptr<SessionState> state) {
|
||||
std::unique_lock lock(mMutex);
|
||||
mSessions[sessionID] = std::move(state);
|
||||
}
|
||||
|
||||
// Lookup
|
||||
std::shared_ptr<const SessionState> get(uint64_t sessionID) const {
|
||||
std::shared_lock lock(mMutex);
|
||||
auto it = mSessions.find(sessionID);
|
||||
return (it == mSessions.end()) ? nullptr : it->second;
|
||||
}
|
||||
|
||||
std::unordered_map<uint64_t, std::shared_ptr<SessionState>> snapshot() const {
|
||||
std::unordered_map<uint64_t, std::shared_ptr<SessionState>> snap;
|
||||
std::shared_lock lock(mMutex);
|
||||
snap = mSessions;
|
||||
return snap;
|
||||
}
|
||||
|
||||
// Remove
|
||||
void erase(uint64_t sessionID) {
|
||||
std::unique_lock lock(mMutex);
|
||||
mSessions.erase(sessionID);
|
||||
}
|
||||
|
||||
// Cleanup expired sessions
|
||||
void cleanupExpired() {
|
||||
std::unique_lock lock(mMutex);
|
||||
auto now = std::chrono::steady_clock::now();
|
||||
for (auto it = mSessions.begin(); it != mSessions.end(); ) {
|
||||
if (it->second && it->second->expires <= now) {
|
||||
it = mSessions.erase(it);
|
||||
} else {
|
||||
++it;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private:
|
||||
mutable std::shared_mutex mMutex;
|
||||
std::unordered_map<uint64_t, std::shared_ptr<SessionState>> mSessions;
|
||||
};
|
||||
}
|
||||
Reference in New Issue
Block a user