Compare commits
6 Commits
dev
...
a2ecc589f8
| Author | SHA1 | Date | |
|---|---|---|---|
| a2ecc589f8 | |||
| 640a751f9b | |||
| a08dba5b59 | |||
| 4ba59fb23f | |||
| 9e5e728438 | |||
| d20bee9e60 |
3
.gitignore
vendored
3
.gitignore
vendored
@@ -12,5 +12,4 @@ _deps
|
||||
CMakeUserPresets.json
|
||||
|
||||
build/
|
||||
.vscode/
|
||||
.DS_Store
|
||||
.vscode/
|
||||
@@ -3,7 +3,7 @@
|
||||
## ASIO C++ Library
|
||||
- **Name:** ASIO (standalone)
|
||||
- **Website:** https://think-async.com/Asio/
|
||||
- **Copyright:** (c) 2003-2026 Christopher M. Kohlhoff
|
||||
- **Copyright:** (c) 2003-2025 Christopher M. Kohlhoff
|
||||
- **License:** Boost Software License, Version 1.0
|
||||
- **License Text:** See `third_party/asio/LICENSE_1_0.txt`
|
||||
|
||||
@@ -12,14 +12,14 @@ This project uses the standalone version of the ASIO C++ library for asynchronou
|
||||
## CXXOPTS C++ Library
|
||||
- **Name:** cxxopts
|
||||
- **Website:** https://github.com/jarro2783/cxxopts/
|
||||
- **Copyright:** (c) 2014-2026 Christopher M. Kohlhoff
|
||||
- **Copyright:** (c) 2014-2025 Christopher M. Kohlhoff
|
||||
- **License:** MIT License
|
||||
- **License Text:** See `third_party/cxxopts/LICENSE_1_0.txt`
|
||||
|
||||
## Wintun C++ Library
|
||||
- **Name:** wintun
|
||||
- **Website:** https://www.wintun.net/
|
||||
- **Copyright:** (c) 2018-2026 WireGuard LLC
|
||||
- **Copyright:** (c) 2018-2025 WireGuard LLC
|
||||
- **License:** MIT License OR GPL-2.0 License
|
||||
- **License Text:** See `third_party/wintun/`
|
||||
- **Utilized Under:** MIT License
|
||||
|
||||
@@ -6,12 +6,10 @@ cmake_minimum_required(VERSION 3.16)
|
||||
# If MAJOR is 0, and MINOR > 0, Version is BETA
|
||||
|
||||
project(ColumnLynx
|
||||
VERSION 1.1.0
|
||||
VERSION 0.1.0
|
||||
LANGUAGES CXX
|
||||
)
|
||||
|
||||
include(GNUInstallDirs)
|
||||
|
||||
# ---------------------------------------------------------
|
||||
# General C++ setup
|
||||
# ---------------------------------------------------------
|
||||
@@ -42,7 +40,7 @@ endif()
|
||||
if(WIN32)
|
||||
add_compile_definitions(_WIN32_WINNT=0x0A00 NOMINMAX WIN32_LEAN_AND_MEAN)
|
||||
elseif(UNIX)
|
||||
add_compile_options(-Wall -Wextra -Wpedantic -O1)
|
||||
add_compile_options(-Wall -Wextra -Wpedantic -O3)
|
||||
add_link_options(-pthread)
|
||||
endif()
|
||||
|
||||
@@ -52,7 +50,7 @@ endif()
|
||||
FetchContent_Declare(
|
||||
Sodium
|
||||
GIT_REPOSITORY https://github.com/robinlinden/libsodium-cmake.git
|
||||
GIT_TAG e5b985ad0dd235d8c4307ea3a385b45e76c74c6a
|
||||
GIT_TAG e5b985ad0dd235d8c4307ea3a385b45e76c74c6a # Last updated at 2025-04-13
|
||||
)
|
||||
|
||||
set(SODIUM_DISABLE_TESTS ON CACHE BOOL "" FORCE)
|
||||
@@ -82,6 +80,15 @@ FetchContent_MakeAvailable(Sodium)
|
||||
FetchContent_MakeAvailable(asio)
|
||||
FetchContent_MakeAvailable(cxxopts)
|
||||
|
||||
# OpenSSL
|
||||
find_package(OpenSSL REQUIRED)
|
||||
if(OPENSSL_FOUND)
|
||||
message(STATUS "Found OpenSSL version ${OPENSSL_VERSION}")
|
||||
include_directories(${OPENSSL_INCLUDE_DIR})
|
||||
else()
|
||||
message(FATAL_ERROR "OpenSSL not found")
|
||||
endif()
|
||||
|
||||
# ---------------------------------------------------------
|
||||
# Output directories
|
||||
# ---------------------------------------------------------
|
||||
@@ -101,17 +108,7 @@ endforeach()
|
||||
# ---------------------------------------------------------
|
||||
file(GLOB_RECURSE COMMON_SRC CONFIGURE_DEPENDS src/common/*.cpp)
|
||||
add_library(common STATIC ${COMMON_SRC})
|
||||
target_link_libraries(common PUBLIC sodium cxxopts::cxxopts)
|
||||
|
||||
if (WIN32)
|
||||
target_link_libraries(common PUBLIC
|
||||
ws2_32
|
||||
iphlpapi
|
||||
advapi32
|
||||
mswsock
|
||||
)
|
||||
endif()
|
||||
|
||||
target_link_libraries(common PUBLIC sodium OpenSSL::SSL OpenSSL::Crypto cxxopts::cxxopts)
|
||||
target_include_directories(common PUBLIC
|
||||
${PROJECT_SOURCE_DIR}/include
|
||||
${sodium_SOURCE_DIR}/src/libsodium/include
|
||||
@@ -125,12 +122,7 @@ target_compile_definitions(common PUBLIC ASIO_STANDALONE)
|
||||
# ---------------------------------------------------------
|
||||
file(GLOB_RECURSE CLIENT_SRC CONFIGURE_DEPENDS src/client/*.cpp)
|
||||
add_executable(client ${CLIENT_SRC})
|
||||
target_link_libraries(client PRIVATE common sodium cxxopts::cxxopts)
|
||||
if (WIN32)
|
||||
target_link_libraries(client PRIVATE
|
||||
dbghelp
|
||||
)
|
||||
endif()
|
||||
target_link_libraries(client PRIVATE common sodium OpenSSL::SSL OpenSSL::Crypto cxxopts::cxxopts)
|
||||
target_include_directories(client PRIVATE
|
||||
${PROJECT_SOURCE_DIR}/include
|
||||
${sodium_SOURCE_DIR}/src/libsodium/include
|
||||
@@ -145,12 +137,7 @@ set_target_properties(client PROPERTIES OUTPUT_NAME "columnlynx_client")
|
||||
# ---------------------------------------------------------
|
||||
file(GLOB_RECURSE SERVER_SRC CONFIGURE_DEPENDS src/server/*.cpp)
|
||||
add_executable(server ${SERVER_SRC})
|
||||
target_link_libraries(server PRIVATE common sodium cxxopts::cxxopts)
|
||||
if (WIN32)
|
||||
target_link_libraries(server PRIVATE
|
||||
dbghelp
|
||||
)
|
||||
endif()
|
||||
target_link_libraries(server PRIVATE common sodium OpenSSL::SSL OpenSSL::Crypto cxxopts::cxxopts)
|
||||
target_include_directories(server PRIVATE
|
||||
${PROJECT_SOURCE_DIR}/include
|
||||
${sodium_SOURCE_DIR}/src/libsodium/include
|
||||
@@ -158,18 +145,4 @@ target_include_directories(server PRIVATE
|
||||
${asio_SOURCE_DIR}/asio/include
|
||||
)
|
||||
target_compile_definitions(server PRIVATE ASIO_STANDALONE)
|
||||
set_target_properties(server PROPERTIES OUTPUT_NAME "columnlynx_server")
|
||||
|
||||
# ---------------------------------------------------------
|
||||
# Install rules
|
||||
# ---------------------------------------------------------
|
||||
|
||||
install(TARGETS
|
||||
client
|
||||
server
|
||||
RUNTIME DESTINATION ${CMAKE_INSTALL_BINDIR}
|
||||
)
|
||||
install(FILES
|
||||
LICENSES/GPL-2.0-only.txt
|
||||
LICENSES/GPL-3.0.txt
|
||||
DESTINATION ${CMAKE_INSTALL_DATAROOTDIR}/licenses/${PROJECT_NAME})
|
||||
set_target_properties(server PROPERTIES OUTPUT_NAME "columnlynx_server")
|
||||
159
README.md
159
README.md
@@ -18,155 +18,22 @@ This simplicity-focused design approach allows us to make an efficient, low-over
|
||||
|
||||
## Configuration
|
||||
|
||||
Configurating the server and client are are relatively easy. Currently (since the project is in alpha), the configuration files **must be in your system-specific config location** (which can be overriden via a CLI argument or the **COLUMNLYNX_CONFIG_DIR** Environment Variable).
|
||||
|
||||
The defaults depends on your system.
|
||||
|
||||
For the server:
|
||||
- Linux: **/etc/columnlynx**
|
||||
- macOS: **/etc/columnlynx**
|
||||
- Windows: **C:\ProgramData\ColumnLynx**
|
||||
|
||||
For the client:
|
||||
- Linux: **~/.config/columnlynx**
|
||||
- macOS: **~/Library/Application Support/columnlynx**
|
||||
- Windows: **C:\Users\USERNAME\AppData\Local\ColumnLynx**
|
||||
|
||||
### Getting a keypair
|
||||
|
||||
Release builds of the software force you to specify your own keypairs. That's why you need to generate a keypair with some other software that you can use.
|
||||
|
||||
This guide will show a generation example with openssl:
|
||||
|
||||
#### Generate a keypair:
|
||||
```bash
|
||||
openssl genpkey -algorithm ED25519 -out key.pem
|
||||
```
|
||||
|
||||
#### Extract the **Private Key Seed**:
|
||||
```bash
|
||||
openssl pkey -in key.pem -outform DER | tail -c 32 | xxd -p -c 32
|
||||
# Output example: 9f3a2b6c0f8e4d1a7c3e9a4b5d2f8c6e1a9d0b7e3f4c2a8e6d5b1f0a3c4e
|
||||
```
|
||||
|
||||
#### Extract the **Raw Public Key**:
|
||||
```bash
|
||||
openssl pkey -in key.pem -pubout -outform DER | tail -c 32 | xxd -p -c 32
|
||||
# Output example: 1c9d4f7a3b2e8a6d0f5c9b1e4d8a7f3c6e2b1a9d5f4c8e0a7b3d6c9f2e
|
||||
```
|
||||
|
||||
You can then set these keys accordingly in the **server_config** and **client_config** files.
|
||||
|
||||
### Server Setup (Linux Server ONLY)
|
||||
|
||||
#### Creating the Tun Interface
|
||||
|
||||
In order for the VPN server to work, you need to create the Tun interface that the VPN will use.
|
||||
|
||||
This is the set of commands to create one on Linux. Replace the example 10.10.0.1/24 IPv4 address with the FIRST IPv4 in the Network and Subnet Mask that you set in server_config.
|
||||
```bash
|
||||
sudo ip tuntap add dev lynx0 mode tun
|
||||
sudo ip addr add 10.10.0.1/24 dev lynx0
|
||||
sudo ip link set dev lynx0 mtu 1420
|
||||
sudo ip link set dev lynx0 up
|
||||
```
|
||||
|
||||
#### Creating the systemd service
|
||||
|
||||
It is highly recommended to **run the server as a systemd service**, as systemd is the primary service manager on Linux.
|
||||
|
||||
**1. Create a file for the service**
|
||||
```bash
|
||||
sudo touch /etc/systemd/system/columnlynx.service
|
||||
```
|
||||
|
||||
**2. Open the file in your editor of choice**
|
||||
```bash
|
||||
sudo nano /etc/systemd/system/columnlynx.service
|
||||
# OR
|
||||
sudo vim /etc/systemd/system/columnlynx.service
|
||||
# OR any other editor of your choice...
|
||||
```
|
||||
|
||||
**3. Configure the service**
|
||||
|
||||
**Replace** the **ExecStart** and **WorkingDirectory** paths with the paths where your binaries are stored.
|
||||
|
||||
If you configured your tun interface to belong to a custom user, you may also replace the **User** and **Group** with that user, however you must ensure that that user owns the **tun interface**, **config directory in /etc/columnlynx** and the **working directory**.
|
||||
|
||||
This is a **simple example** for the **root user** and the executable in **/opt/columnlynx**:
|
||||
|
||||
```
|
||||
[Unit]
|
||||
Description=ColumnLynx Server Service
|
||||
After=network.target
|
||||
|
||||
[Service]
|
||||
Type=simple
|
||||
ExecStart=/opt/columnlynx/columnlynx_server
|
||||
WorkingDirectory=/opt/columnlynx
|
||||
User=root
|
||||
Group=root
|
||||
Restart=on-failure
|
||||
StandardOutput=append:/var/log/columnlynx.log
|
||||
StandardError=append:/var/log/columnlynx.err
|
||||
|
||||
[Install]
|
||||
WantedBy=multi-user.target
|
||||
```
|
||||
|
||||
**4. Reload systemd and enable the service**
|
||||
|
||||
```bash
|
||||
sudo systemctl daemon-reload
|
||||
sudo systemctl enable columnlynx.service
|
||||
sudo systemctl start columnlynx.service
|
||||
```
|
||||
|
||||
#### Set firewall rules
|
||||
|
||||
This part greatly depends on your firewall of choice. Generally you just need to **allow port 48042 on both TCP and UDP** (Both IPv4 and IPv6).
|
||||
|
||||
This example is for **UFW**:
|
||||
|
||||
```bash
|
||||
sudo ufw allow 48042
|
||||
sudo ufw reload
|
||||
```
|
||||
|
||||
|
||||
#### IPTables rules for forwarding (Optional)
|
||||
|
||||
In addition to creating the interface, you'll also need to make some **iptables** rules if you want to be able to **send traffic to foreign networks** (more like a *commercial VPN*).
|
||||
|
||||
You can do these as such (example with NFT IPTABLES):
|
||||
|
||||
- Enable the **generic IPv4 forwarding**:
|
||||
```bash
|
||||
sudo sysctl net.ipv4.ip_forward=1
|
||||
```
|
||||
- Create the masquerade (**Replace the IP subnet** with your own that you set in the config and **replace the interface** with your server's main (NOT *lynx0*) interface):
|
||||
```bash
|
||||
sudo nft add table nat
|
||||
sudo nft add chain nat postroute { type nat hook postrouting priority 100 \; }
|
||||
sudo nft add rule nat postroute ip saddr 10.10.0.0/24 oifname "eth0" masquerade
|
||||
```
|
||||
|
||||
Configurating the server and client are are relatively easy. Currently (since the project is in alpha), the configuration files **must be in the same directory as the working directory**.
|
||||
|
||||
### Server
|
||||
|
||||
"**server_config**" is a file that contains the server configuration, **one variable per line**. These are the current configuration available variables:
|
||||
|
||||
- **SERVER_PUBLIC_KEY** (Hex String): The public key to be used - Used for verification
|
||||
- **SERVER_PRIVATE_KEY** (Hex String): The private key seed to be used
|
||||
- **SERVER_PUBLIC_KEY** (Hex String): The public key to be used
|
||||
- **SERVER_PRIVATE_KEY** (Hex String): The private key to be used
|
||||
- **NETWORK** (IPv4 Format): The network IPv4 to be used (Server Interface still needs to be configured manually)
|
||||
- **SUBNET_MASK** (Integer): The subnet mask to be used (ensure proper length, it will not be checked)
|
||||
|
||||
**Example:**
|
||||
|
||||
```
|
||||
SERVER_PUBLIC_KEY=1c9d4f7a3b2e8a6d0f5c9b1e4d8a7f3c6e2b1a9d5f4c8e0a7b3d6c9f2e
|
||||
SERVER_PRIVATE_KEY=9f3a2b6c0f8e4d1a7c3e9a4b5d2f8c6e1a9d0b7e3f4c2a8e6d5b1f0a3c4e
|
||||
SERVER_PUBLIC_KEY=787B648046F10DDD0B77A6303BE42D859AA65C52F5708CC3C58EB5691F217C7B
|
||||
SERVER_PRIVATE_KEY=778604245F57B847E63BD85DE8208FF1A127FB559895195928C3987E246B77B8787B648046F10DDD0B77A6303BE42D859AA65C52F5708CC3C58EB5691F217C7B
|
||||
NETWORK=10.10.0.0
|
||||
SUBNET_MASK=24
|
||||
```
|
||||
@@ -186,14 +53,14 @@ SUBNET_MASK=24
|
||||
|
||||
"**client_config**" is a file that contains the client configuration, **one variable per line**. These are the current configuration available variables:
|
||||
|
||||
- **CLIENT_PUBLIC_KEY** (Hex String): The public key to be used - Used for verification
|
||||
- **CLIENT_PRIVATE_KEY** (Hex String): The private key seed to be used
|
||||
- **CLIENT_PUBLIC_KEY** (Hex String): The public key to be used
|
||||
- **CLIENT_PRIVATE_KEY** (Hex String): The private key to be used
|
||||
|
||||
**Example:**
|
||||
|
||||
```
|
||||
CLIENT_PUBLIC_KEY=1c9d4f7a3b2e8a6d0f5c9b1e4d8a7f3c6e2b1a9d5f4c8e0a7b3d6c9f2e
|
||||
CLIENT_PRIVATE_KEY=9f3a2b6c0f8e4d1a7c3e9a4b5d2f8c6e1a9d0b7e3f4c2a8e6d5b1f0a3c4e
|
||||
CLIENT_PUBLIC_KEY=8CC8BE1A9D24639D0492EF143E84E2BD4C757C9B3B687E7035173EBFCA8FEDDA
|
||||
CLIENT_PRIVATE_KEY=9B486A5B1509FA216F9EEFED85CACF2384E9D902A76CC979BFA143C18B869F5C8CC8BE1A9D24639D0492EF143E84E2BD4C757C9B3B687E7035173EBFCA8FEDDA
|
||||
```
|
||||
|
||||
<hr></hr>
|
||||
@@ -213,8 +80,6 @@ ColumnLynx makes use of both **TCP** and **UDP**. **TCP** is used for the initia
|
||||
|
||||
It operates on port **48042** for both TCP and UDP.
|
||||
|
||||
Current protocol version is **2**.
|
||||
|
||||
Generally, all transmission is done in **little-endian byte order**, since pretty much every single modern architecture uses it by default. The only exemption to this is the **transmission of IP addresses** (for the **Virtual Interface**), which is **big-endian**.
|
||||
|
||||
### Handshake Procedure
|
||||
@@ -233,7 +98,7 @@ The Client now generates a random aesKey (32 bytes long)
|
||||
|
||||
C: HANDSHAKE_EXCHANGE_KEY <aesKey Encrypted with Server Public Key>
|
||||
|
||||
The Server now assigns a local 4 byte session ID in the Session Registry.
|
||||
The Server now assigns a local 8 byte session ID in the Session Registry.
|
||||
|
||||
S: HANDSHAKE_EXCHANGE_KEY_CONFIRM <Assigned SessionID>
|
||||
```
|
||||
@@ -244,7 +109,7 @@ The **Client** and **Server** have now securely exchanged a symmetric **AES Key*
|
||||
|
||||
Packet exchange and the general data tunneling is done via **Standard UDP** (*see the **UDP Packet** in **Data***).
|
||||
|
||||
The **header** of the sent packet always includes a **12 byte nonce** derived from a random **4 byte base nonce** and the **send count** to ensure a unique nonce, used to obscure the **encrypted payload / data** and the **Session ID** assigned by the server to the client (4 bytes). This makes the header **16 bytes long**.
|
||||
The **header** of the sent packet always includes a **random 12 byte nonce** used to obscure the **encrypted payload / data** and the **Session ID** assigned by the server to the client (8 bytes). This makes the header **20 bytes long**.
|
||||
|
||||
The **payload / data** of the sent packet is **always encrypted** using the exchanged **AES Key** and obscured using the **random nonce**.
|
||||
|
||||
@@ -300,7 +165,7 @@ The **Data** is generally just the **raw underlying packet** forwarded to the se
|
||||
| Type | Length | Name | Description |
|
||||
|:-----|:-------|:-----|:------------|
|
||||
| uint8_t | 12 bytes | **Header** - Nonce | Random nonce to obfuscate encrypted contents |
|
||||
| uint32_t | 4 bytes | **Header** - Session ID | The unique and random session identifier for the client |
|
||||
| uint64_t | 8 bytes | **Header** - Session ID | The unique and random session identifier for the client |
|
||||
| uint8_t | variable | Data | General data / payload |
|
||||
|
||||
## Misc.
|
||||
|
||||
@@ -1,102 +0,0 @@
|
||||
// client_session.hpp - Client Session data for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <memory>
|
||||
#include <columnlynx/common/libsodium_wrapper.hpp>
|
||||
#include <array>
|
||||
#include <columnlynx/common/net/virtual_interface.hpp>
|
||||
#include <shared_mutex>
|
||||
|
||||
namespace ColumnLynx {
|
||||
struct ClientState {
|
||||
std::shared_ptr<Utils::LibSodiumWrapper> sodiumWrapper;
|
||||
SymmetricKey aesKey;
|
||||
bool insecureMode;
|
||||
std::string configPath;
|
||||
std::shared_ptr<Net::VirtualInterface> virtualInterface;
|
||||
uint32_t sessionID;
|
||||
uint64_t recv_cnt;
|
||||
uint64_t send_cnt;
|
||||
uint32_t noncePrefix;
|
||||
|
||||
~ClientState() { sodium_memzero(aesKey.data(), aesKey.size()); }
|
||||
ClientState(const ClientState&) = delete;
|
||||
ClientState& operator=(const ClientState&) = delete;
|
||||
ClientState(ClientState&&) = default;
|
||||
ClientState& operator=(ClientState&&) = default;
|
||||
|
||||
explicit ClientState() = default;
|
||||
|
||||
explicit ClientState(std::shared_ptr<Utils::LibSodiumWrapper> sodium, SymmetricKey& k, bool insecure,
|
||||
std::string& config, std::shared_ptr<Net::VirtualInterface> tun, uint32_t session, uint64_t recv, uint64_t send)
|
||||
: sodiumWrapper(sodium), aesKey(k), insecureMode(insecure), configPath(config), virtualInterface(tun), sessionID(session), recv_cnt(recv), send_cnt(send) {}
|
||||
};
|
||||
|
||||
class ClientSession {
|
||||
public:
|
||||
// Return a reference to the Client Session instance
|
||||
static ClientSession& getInstance() { static ClientSession instance; return instance; }
|
||||
|
||||
// Return the current client state
|
||||
std::shared_ptr<ClientState> getClientState() const;
|
||||
|
||||
// Set the client state
|
||||
void setClientState(std::shared_ptr<ClientState> state);
|
||||
|
||||
// Get the wrapper for libsodium
|
||||
const std::shared_ptr<Utils::LibSodiumWrapper>& getSodiumWrapper() const;
|
||||
// Get the AES key
|
||||
const SymmetricKey& getAESKey() const;
|
||||
// Get whether insecure mode is enabled
|
||||
bool isInsecureMode() const;
|
||||
// Get the config path
|
||||
const std::string& getConfigPath() const;
|
||||
// Get the virtual interface
|
||||
const std::shared_ptr<Net::VirtualInterface>& getVirtualInterface() const;
|
||||
// Get the session ID
|
||||
uint32_t getSessionID() const;
|
||||
uint64_t getRecvCount() const {
|
||||
std::shared_lock lock(mMutex);
|
||||
return mClientState->recv_cnt;
|
||||
}
|
||||
|
||||
uint64_t getSendCount() const {
|
||||
std::shared_lock lock(mMutex);
|
||||
return mClientState->send_cnt;
|
||||
}
|
||||
|
||||
uint32_t getNoncePrefix() const {
|
||||
std::shared_lock lock(mMutex);
|
||||
return mClientState->noncePrefix;
|
||||
}
|
||||
|
||||
// Setters
|
||||
void setSodiumWrapper(std::shared_ptr<Utils::LibSodiumWrapper> sodiumWrapper);
|
||||
void setAESKey(const SymmetricKey& aesKey);
|
||||
void setInsecureMode(bool insecureMode);
|
||||
void setConfigPath(const std::string& configPath);
|
||||
void setVirtualInterface(std::shared_ptr<Net::VirtualInterface> virtualInterface);
|
||||
void setSessionID(uint32_t sessionID);
|
||||
void incrementRecvCount() {
|
||||
std::unique_lock lock(mMutex);
|
||||
mClientState->recv_cnt++;
|
||||
}
|
||||
|
||||
void incrementSendCount() {
|
||||
std::unique_lock lock(mMutex);
|
||||
mClientState->send_cnt++;
|
||||
}
|
||||
|
||||
void setNoncePrefix(uint32_t prefix) {
|
||||
std::unique_lock lock(mMutex);
|
||||
mClientState->noncePrefix = prefix;
|
||||
}
|
||||
|
||||
private:
|
||||
mutable std::shared_mutex mMutex;
|
||||
std::shared_ptr<struct ClientState> mClientState{nullptr};
|
||||
};
|
||||
}
|
||||
@@ -1,5 +1,5 @@
|
||||
// tcp_client.hpp - TCP Client for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
@@ -14,10 +14,8 @@
|
||||
#include <algorithm>
|
||||
#include <vector>
|
||||
#include <unordered_map>
|
||||
#include <string>
|
||||
#include <columnlynx/common/net/protocol_structs.hpp>
|
||||
#include <columnlynx/common/net/virtual_interface.hpp>
|
||||
#include <columnlynx/client/client_session.hpp>
|
||||
|
||||
using asio::ip::tcp;
|
||||
|
||||
@@ -26,22 +24,28 @@ namespace ColumnLynx::Net::TCP {
|
||||
public:
|
||||
TCPClient(asio::io_context& ioContext,
|
||||
const std::string& host,
|
||||
const std::string& port)
|
||||
const std::string& port,
|
||||
std::shared_ptr<Utils::LibSodiumWrapper> sodiumWrapper,
|
||||
std::shared_ptr<std::array<uint8_t, 32>> aesKey,
|
||||
std::shared_ptr<uint64_t> sessionIDRef,
|
||||
bool insecureMode,
|
||||
std::shared_ptr<VirtualInterface> tun = nullptr)
|
||||
:
|
||||
mResolver(ioContext),
|
||||
mSocket(ioContext),
|
||||
mHost(host),
|
||||
mPort(port),
|
||||
mLibSodiumWrapper(sodiumWrapper),
|
||||
mGlobalKeyRef(aesKey),
|
||||
mSessionIDRef(sessionIDRef),
|
||||
mInsecureMode(insecureMode),
|
||||
mHeartbeatTimer(mSocket.get_executor()),
|
||||
mLastHeartbeatReceived(std::chrono::steady_clock::now()),
|
||||
mLastHeartbeatSent(std::chrono::steady_clock::now())
|
||||
mLastHeartbeatSent(std::chrono::steady_clock::now()),
|
||||
mTun(tun)
|
||||
{
|
||||
// Get initial client config
|
||||
std::string configPath = ClientSession::getInstance().getConfigPath();
|
||||
std::shared_ptr<Utils::LibSodiumWrapper> mLibSodiumWrapper = ClientSession::getInstance().getSodiumWrapper();
|
||||
|
||||
// Preload the config map
|
||||
mRawClientConfig = Utils::getConfigMap(configPath + "client_config");
|
||||
mRawClientConfig = Utils::getConfigMap("client_config");
|
||||
|
||||
auto itPubkey = mRawClientConfig.find("CLIENT_PUBLIC_KEY");
|
||||
auto itPrivkey = mRawClientConfig.find("CLIENT_PRIVATE_KEY");
|
||||
@@ -50,22 +54,16 @@ namespace ColumnLynx::Net::TCP {
|
||||
Utils::log("Loading keypair from config file.");
|
||||
|
||||
PublicKey pk;
|
||||
PrivateSeed seed;
|
||||
PrivateKey sk;
|
||||
|
||||
std::copy_n(Utils::hexStringToBytes(itPrivkey->second).begin(), seed.size(), seed.begin()); // This is extremely stupid, but the C++ compiler has forced my hand (I would've just used to_array, but fucking asio decls)
|
||||
std::copy_n(Utils::hexStringToBytes(itPrivkey->second).begin(), sk.size(), sk.begin()); // This is extremely stupid, but the C++ compiler has forced my hand (I would've just used to_array, but fucking asio decls)
|
||||
std::copy_n(Utils::hexStringToBytes(itPubkey->second).begin(), pk.size(), pk.begin());
|
||||
|
||||
if (!mLibSodiumWrapper->recomputeKeys(seed, pk)) {
|
||||
throw std::runtime_error("Failed to recompute keypair from config file values!");
|
||||
}
|
||||
mLibSodiumWrapper->setKeys(pk, sk);
|
||||
|
||||
Utils::debug("Newly-Loaded Public Key: " + Utils::bytesToHexString(mLibSodiumWrapper->getPublicKey(), 32));
|
||||
} else {
|
||||
#if defined(DEBUG)
|
||||
Utils::warn("No keypair found in config file! Using random key.");
|
||||
#else
|
||||
throw std::runtime_error("No keypair found in config file! Cannot start client without keys.");
|
||||
#endif
|
||||
}
|
||||
}
|
||||
|
||||
@@ -97,14 +95,19 @@ namespace ColumnLynx::Net::TCP {
|
||||
std::string mHost, mPort;
|
||||
uint8_t mServerPublicKey[32]; // Assuming 256-bit public key
|
||||
std::array<uint8_t, 32> mSubmittedChallenge{};
|
||||
uint32_t mConnectionSessionID;
|
||||
std::shared_ptr<Utils::LibSodiumWrapper> mLibSodiumWrapper;
|
||||
uint64_t mConnectionSessionID;
|
||||
SymmetricKey mConnectionAESKey;
|
||||
std::shared_ptr<std::array<uint8_t, 32>> mGlobalKeyRef; // Reference to global AES key
|
||||
std::shared_ptr<uint64_t> mSessionIDRef; // Reference to global Session ID
|
||||
bool mInsecureMode; // Insecure mode flag
|
||||
asio::steady_timer mHeartbeatTimer;
|
||||
std::chrono::steady_clock::time_point mLastHeartbeatReceived;
|
||||
std::chrono::steady_clock::time_point mLastHeartbeatSent;
|
||||
int mMissedHeartbeats = 0;
|
||||
bool mIsHostDomain;
|
||||
Protocol::TunConfig mTunConfig;
|
||||
std::shared_ptr<VirtualInterface> mTun = nullptr;
|
||||
std::unordered_map<std::string, std::string> mRawClientConfig;
|
||||
};
|
||||
}
|
||||
@@ -1,5 +1,5 @@
|
||||
// udp_client.hpp - UDP Client for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
@@ -10,15 +10,17 @@
|
||||
#include <columnlynx/common/libsodium_wrapper.hpp>
|
||||
#include <array>
|
||||
#include <columnlynx/common/net/virtual_interface.hpp>
|
||||
#include <columnlynx/client/client_session.hpp>
|
||||
|
||||
namespace ColumnLynx::Net::UDP {
|
||||
class UDPClient {
|
||||
public:
|
||||
UDPClient(asio::io_context& ioContext,
|
||||
const std::string& host,
|
||||
const std::string& port)
|
||||
: mSocket(ioContext), mResolver(ioContext), mHost(host), mPort(port)
|
||||
const std::string& port,
|
||||
std::shared_ptr<std::array<uint8_t, 32>> aesKeyRef,
|
||||
std::shared_ptr<uint64_t> sessionIDRef,
|
||||
std::shared_ptr<VirtualInterface> tunRef = nullptr)
|
||||
: mSocket(ioContext), mResolver(ioContext), mHost(host), mPort(port), mAesKeyRef(aesKeyRef), mSessionIDRef(sessionIDRef), mTunRef(tunRef)
|
||||
{
|
||||
mStartReceive();
|
||||
}
|
||||
@@ -41,6 +43,9 @@ namespace ColumnLynx::Net::UDP {
|
||||
asio::ip::udp::endpoint mRemoteEndpoint;
|
||||
std::string mHost;
|
||||
std::string mPort;
|
||||
std::shared_ptr<std::array<uint8_t, 32>> mAesKeyRef;
|
||||
std::shared_ptr<uint64_t> mSessionIDRef;
|
||||
std::shared_ptr<VirtualInterface> mTunRef = nullptr;
|
||||
std::array<uint8_t, 2048> mRecvBuffer; // Adjust size as needed
|
||||
};
|
||||
}
|
||||
@@ -1,5 +1,5 @@
|
||||
// libsodium_wrapper.hpp - Libsodium Wrapper for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
@@ -11,13 +11,16 @@
|
||||
#include <columnlynx/common/utils.hpp>
|
||||
#include <array>
|
||||
#include <vector>
|
||||
#include <openssl/x509.h>
|
||||
#include <openssl/x509_vfy.h>
|
||||
#include <openssl/pem.h>
|
||||
#include <openssl/x509v3.h>
|
||||
#include <memory>
|
||||
#include <cstring>
|
||||
|
||||
namespace ColumnLynx {
|
||||
using PublicKey = std::array<uint8_t, crypto_sign_PUBLICKEYBYTES>; // Ed25519
|
||||
using PrivateKey = std::array<uint8_t, crypto_sign_SECRETKEYBYTES>; // Ed25519
|
||||
using PrivateSeed = std::array<uint8_t, crypto_sign_SEEDBYTES>; // 32 bytes
|
||||
using Signature = std::array<uint8_t, crypto_sign_BYTES>; // 64 bytes
|
||||
using SymmetricKey = std::array<uint8_t, crypto_aead_chacha20poly1305_ietf_KEYBYTES>; // 32 bytes
|
||||
using Nonce = std::array<uint8_t, crypto_aead_chacha20poly1305_ietf_NPUBBYTES>; // 12 bytes
|
||||
@@ -54,9 +57,6 @@ namespace ColumnLynx::Utils {
|
||||
}
|
||||
}
|
||||
|
||||
// Recompute the keypair from a given private seed; Will return false on failure
|
||||
bool recomputeKeys(PrivateSeed privateSeed, PublicKey storedPubKey);
|
||||
|
||||
// Helper section
|
||||
|
||||
// Generates a random 256-bit (32-byte) array
|
||||
@@ -211,4 +211,4 @@ namespace ColumnLynx::Utils {
|
||||
std::array<uint8_t, crypto_scalarmult_curve25519_BYTES> mXPublicKey;
|
||||
std::array<uint8_t, crypto_scalarmult_curve25519_BYTES> mXPrivateKey;
|
||||
};
|
||||
}
|
||||
}
|
||||
@@ -1,5 +1,5 @@
|
||||
// protocol_structs.hpp - Network Protocol Structures
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
// session_registry.hpp - Session Registry for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
@@ -27,9 +27,8 @@ namespace ColumnLynx::Net {
|
||||
std::chrono::steady_clock::time_point expires{}; // Time of expiry
|
||||
uint32_t clientTunIP; // Assigned IP
|
||||
uint32_t serverTunIP; // Server IP
|
||||
uint32_t sessionID; // Session ID
|
||||
uint64_t sessionID; // Session ID
|
||||
Nonce base_nonce{};
|
||||
uint32_t noncePrefix;
|
||||
|
||||
~SessionState() { sodium_memzero(aesKey.data(), aesKey.size()); }
|
||||
SessionState(const SessionState&) = delete;
|
||||
@@ -37,7 +36,7 @@ namespace ColumnLynx::Net {
|
||||
SessionState(SessionState&&) = default;
|
||||
SessionState& operator=(SessionState&&) = default;
|
||||
|
||||
explicit SessionState(const SymmetricKey& k, std::chrono::seconds ttl = std::chrono::hours(24), uint32_t clientIP = 0, uint32_t serverIP = 0, uint32_t id = 0) : aesKey(k), clientTunIP(clientIP), serverTunIP(serverIP), sessionID(id) {
|
||||
explicit SessionState(const SymmetricKey& k, std::chrono::seconds ttl = std::chrono::hours(24), uint32_t clientIP = 0, uint32_t serverIP = 0, uint64_t id = 0) : aesKey(k), clientTunIP(clientIP), serverTunIP(serverIP), sessionID(id) {
|
||||
expires = created + ttl;
|
||||
}
|
||||
|
||||
@@ -45,11 +44,6 @@ namespace ColumnLynx::Net {
|
||||
void setUDPEndpoint(const asio::ip::udp::endpoint& ep) {
|
||||
udpEndpoint = ep;
|
||||
}
|
||||
|
||||
void setBaseNonce() {
|
||||
Utils::debug("Generating random base nonce for session " + std::to_string(sessionID));
|
||||
randombytes_buf(base_nonce.data(), base_nonce.size());
|
||||
}
|
||||
};
|
||||
|
||||
class SessionRegistry {
|
||||
@@ -58,19 +52,19 @@ namespace ColumnLynx::Net {
|
||||
static SessionRegistry& getInstance() { static SessionRegistry instance; return instance; }
|
||||
|
||||
// Insert or replace a session entry
|
||||
void put(uint32_t sessionID, std::shared_ptr<SessionState> state);
|
||||
void put(uint64_t sessionID, std::shared_ptr<SessionState> state);
|
||||
|
||||
// Lookup a session entry by session ID
|
||||
std::shared_ptr<const SessionState> get(uint32_t sessionID) const;
|
||||
std::shared_ptr<const SessionState> get(uint64_t sessionID) const;
|
||||
|
||||
// Lookup a session entry by IPv4
|
||||
std::shared_ptr<const SessionState> getByIP(uint32_t ip) const;
|
||||
|
||||
// Get a snapshot of the Session Registry
|
||||
std::unordered_map<uint32_t, std::shared_ptr<SessionState>> snapshot() const;
|
||||
std::unordered_map<uint64_t, std::shared_ptr<SessionState>> snapshot() const;
|
||||
|
||||
// Remove a session by ID
|
||||
void erase(uint32_t sessionID);
|
||||
void erase(uint64_t sessionID);
|
||||
|
||||
// Cleanup expired sessions
|
||||
void cleanupExpired();
|
||||
@@ -78,23 +72,21 @@ namespace ColumnLynx::Net {
|
||||
// Get the number of registered sessions
|
||||
int size() const;
|
||||
|
||||
bool exists(uint32_t sessionID) const;
|
||||
|
||||
// IP management
|
||||
|
||||
// Get the lowest available IPv4 address; Returns 0 if none available
|
||||
uint32_t getFirstAvailableIP(uint32_t baseIP, uint8_t mask) const;
|
||||
|
||||
// Lock IP to session ID; Do NOT call before put() - You will segfault!
|
||||
void lockIP(uint32_t sessionID, uint32_t ip);
|
||||
void lockIP(uint64_t sessionID, uint32_t ip);
|
||||
|
||||
// Unlock IP from session ID
|
||||
void deallocIP(uint32_t sessionID);
|
||||
void deallocIP(uint64_t sessionID);
|
||||
|
||||
private:
|
||||
mutable std::shared_mutex mMutex;
|
||||
std::unordered_map<uint32_t, std::shared_ptr<SessionState>> mSessions;
|
||||
std::unordered_map<uint32_t, uint32_t> mSessionIPs;
|
||||
std::unordered_map<uint64_t, std::shared_ptr<SessionState>> mSessions;
|
||||
std::unordered_map<uint64_t, uint32_t> mSessionIPs;
|
||||
std::unordered_map<uint32_t, std::shared_ptr<SessionState>> mIPSessions;
|
||||
};
|
||||
}
|
||||
}
|
||||
@@ -1,5 +1,5 @@
|
||||
// net_helper.hpp - Network Helper Functions for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
// tcp_message_handler.hpp - TCP Message Handler for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
// tcp_message_type.hpp - TCP Message Types for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
// udp_message_type.hpp - UDP Message Types for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
// virtual_interface.hpp - Virtual Interface for Network Communication
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
@@ -28,20 +28,11 @@
|
||||
#include <arpa/inet.h>
|
||||
#include <sys/poll.h>
|
||||
#elif defined(_WIN32)
|
||||
#define WIN32_LEAN_AND_MEAN
|
||||
#define WINTUN_STATIC
|
||||
#include <windows.h>
|
||||
#include <winsock2.h>
|
||||
#include <ws2tcpip.h>
|
||||
#include <locale>
|
||||
#include <codecvt>
|
||||
#include <winsock2.h>
|
||||
#include <wintun/wintun.h>
|
||||
|
||||
#include <iphlpapi.h>
|
||||
#include <netioapi.h>
|
||||
|
||||
#pragma comment(lib, "iphlpapi.lib")
|
||||
#pragma comment(lib, "ws2_32.lib")
|
||||
#pragma comment(lib, "advapi32.lib")
|
||||
#endif
|
||||
|
||||
namespace ColumnLynx::Net {
|
||||
@@ -52,8 +43,6 @@ namespace ColumnLynx::Net {
|
||||
|
||||
bool configureIP(uint32_t clientIP, uint32_t serverIP,
|
||||
uint8_t prefixLen, uint16_t mtu);
|
||||
|
||||
void resetIP();
|
||||
|
||||
std::vector<uint8_t> readPacket();
|
||||
void writePacket(const std::vector<uint8_t>& packet);
|
||||
@@ -86,42 +75,6 @@ namespace ColumnLynx::Net {
|
||||
return ntohl(addr.s_addr);
|
||||
}
|
||||
|
||||
static inline std::string ipv6ToString(IPv6Addr &ip,
|
||||
bool flip = false)
|
||||
{
|
||||
struct in6_addr addr;
|
||||
|
||||
if (flip) {
|
||||
IPv6Addr flipped;
|
||||
for (size_t i = 0; i < 16; ++i)
|
||||
flipped[i] = ip[15 - i];
|
||||
memcpy(addr.s6_addr, flipped.data(), 16);
|
||||
} else {
|
||||
memcpy(addr.s6_addr, ip.data(), 16);
|
||||
}
|
||||
|
||||
char buf[INET6_ADDRSTRLEN];
|
||||
if (!inet_ntop(AF_INET6, &addr, buf, sizeof(buf)))
|
||||
return "::"; // Fallback
|
||||
|
||||
return std::string(buf);
|
||||
}
|
||||
|
||||
static inline IPv6Addr stringToIpv6(const std::string &ipStr)
|
||||
{
|
||||
IPv6Addr result{};
|
||||
struct in6_addr addr;
|
||||
|
||||
if (inet_pton(AF_INET6, ipStr.c_str(), &addr) != 1) {
|
||||
// "::"
|
||||
result.fill(0);
|
||||
return result;
|
||||
}
|
||||
|
||||
memcpy(result.data(), addr.s6_addr, 16);
|
||||
return result;
|
||||
}
|
||||
|
||||
static inline uint32_t prefixLengthToNetmask(uint8_t prefixLen) {
|
||||
if (prefixLen == 0) return 0;
|
||||
uint32_t mask = (0xFFFFFFFF << (32 - prefixLen)) & 0xFFFFFFFF;
|
||||
@@ -136,9 +89,7 @@ namespace ColumnLynx::Net {
|
||||
std::string mIfName;
|
||||
int mFd; // POSIX
|
||||
#if defined(_WIN32)
|
||||
WINTUN_ADAPTER_HANDLE mAdapter = nullptr;
|
||||
WINTUN_SESSION_HANDLE mSession = nullptr;
|
||||
HANDLE mHandle = nullptr;
|
||||
HANDLE mHandle; // Windows
|
||||
#endif
|
||||
};
|
||||
}
|
||||
@@ -1,5 +1,5 @@
|
||||
// panic_handler.hpp - Panic Handler for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
@@ -186,7 +186,7 @@ namespace ColumnLynx::Utils {
|
||||
|
||||
// Panic the main thread and instantly halt execution. This produces a stack trace dump. Do not use by itself, throw an error instead.
|
||||
static void panic(const std::string& reason) {
|
||||
std::cerr << "\n***\033[31m MASTER THREAD PANIC! \033[0m***\n";
|
||||
std::cerr << "\n***\033[31m MAIN THREAD PANIC! \033[0m***\n";
|
||||
std::cerr << "Reason: " << reason << "\n";
|
||||
std::cerr << "Dumping panic trace...\n";
|
||||
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
// utils.hpp - Utility functions for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
@@ -15,7 +15,6 @@
|
||||
#include <unordered_map>
|
||||
#include <unordered_set>
|
||||
#include <algorithm>
|
||||
#include <bits/stdc++.h>
|
||||
|
||||
#ifdef _WIN32
|
||||
#include <winsock2.h>
|
||||
@@ -30,9 +29,6 @@ namespace ColumnLynx {
|
||||
}
|
||||
|
||||
namespace ColumnLynx::Utils {
|
||||
// Converts unix milliseconds to a local ISO 8601 formatted string; Defaults to local time; Will use UTC if local is false.
|
||||
std::string unixMillisToISO8601(uint64_t unixMillis, bool local = true);
|
||||
|
||||
// General log function. Use for logging important information.
|
||||
void log(const std::string &msg);
|
||||
// General warning function. Use for logging important warnings.
|
||||
@@ -48,7 +44,7 @@ namespace ColumnLynx::Utils {
|
||||
std::string getVersion();
|
||||
unsigned short serverPort();
|
||||
unsigned char protocolVersion();
|
||||
std::vector<std::string> getWhitelistedKeys(std::string basePath);
|
||||
std::vector<std::string> getWhitelistedKeys();
|
||||
|
||||
// Raw byte to hex string conversion helper
|
||||
std::string bytesToHexString(const uint8_t* bytes, size_t length);
|
||||
@@ -65,6 +61,27 @@ namespace ColumnLynx::Utils {
|
||||
return std::string(reinterpret_cast<const char*>(data), length);
|
||||
}
|
||||
|
||||
inline constexpr uint64_t cbswap64(uint64_t x) {
|
||||
return ((x & 0x00000000000000FFULL) << 56) |
|
||||
((x & 0x000000000000FF00ULL) << 40) |
|
||||
((x & 0x0000000000FF0000ULL) << 24) |
|
||||
((x & 0x00000000FF000000ULL) << 8) |
|
||||
((x & 0x000000FF00000000ULL) >> 8) |
|
||||
((x & 0x0000FF0000000000ULL) >> 24) |
|
||||
((x & 0x00FF000000000000ULL) >> 40) |
|
||||
((x & 0xFF00000000000000ULL) >> 56);
|
||||
}
|
||||
|
||||
// host -> big-endian (for little-endian hosts) - 64 bit
|
||||
inline constexpr uint64_t chtobe64(uint64_t x) {
|
||||
return cbswap64(x);
|
||||
}
|
||||
|
||||
// big-endian -> host (for little-endian hosts) - 64 bit
|
||||
inline constexpr uint64_t cbe64toh(uint64_t x) {
|
||||
return cbswap64(x);
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
T cbswap128(const T& x) {
|
||||
static_assert(sizeof(T) == 16, "cbswap128 requires a 128-bit type");
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
// tcp_connection.hpp - TCP Connection for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
@@ -18,7 +18,6 @@
|
||||
#include <columnlynx/common/net/session_registry.hpp>
|
||||
#include <columnlynx/common/net/protocol_structs.hpp>
|
||||
#include <columnlynx/common/net/virtual_interface.hpp>
|
||||
#include <columnlynx/server/server_session.hpp>
|
||||
|
||||
namespace ColumnLynx::Net::TCP {
|
||||
class TCPConnection : public std::enable_shared_from_this<TCPConnection> {
|
||||
@@ -27,9 +26,11 @@ namespace ColumnLynx::Net::TCP {
|
||||
|
||||
static pointer create(
|
||||
asio::ip::tcp::socket socket,
|
||||
std::shared_ptr<Utils::LibSodiumWrapper> sodiumWrapper,
|
||||
std::unordered_map<std::string, std::string>* serverConfig,
|
||||
std::function<void(pointer)> onDisconnect)
|
||||
{
|
||||
auto conn = pointer(new TCPConnection(std::move(socket)));
|
||||
auto conn = pointer(new TCPConnection(std::move(socket), sodiumWrapper, serverConfig));
|
||||
conn->mOnDisconnect = std::move(onDisconnect);
|
||||
return conn;
|
||||
}
|
||||
@@ -41,17 +42,19 @@ namespace ColumnLynx::Net::TCP {
|
||||
// Set callback for disconnects
|
||||
void setDisconnectCallback(std::function<void(std::shared_ptr<TCPConnection>)> cb);
|
||||
// Disconnect the client
|
||||
void disconnect(bool echo = true);
|
||||
void disconnect();
|
||||
|
||||
// Get the assigned session ID
|
||||
uint32_t getSessionID() const;
|
||||
uint64_t getSessionID() const;
|
||||
// Get the assigned AES key; You should probably access this via the Session Registry instead
|
||||
std::array<uint8_t, 32> getAESKey() const;
|
||||
|
||||
private:
|
||||
TCPConnection(asio::ip::tcp::socket socket)
|
||||
TCPConnection(asio::ip::tcp::socket socket, std::shared_ptr<Utils::LibSodiumWrapper> sodiumWrapper, std::unordered_map<std::string, std::string>* serverConfig)
|
||||
:
|
||||
mHandler(std::make_shared<MessageHandler>(std::move(socket))),
|
||||
mLibSodiumWrapper(sodiumWrapper),
|
||||
mRawServerConfig(serverConfig),
|
||||
mHeartbeatTimer(mHandler->socket().get_executor()),
|
||||
mLastHeartbeatReceived(std::chrono::steady_clock::now()),
|
||||
mLastHeartbeatSent(std::chrono::steady_clock::now())
|
||||
@@ -64,13 +67,14 @@ namespace ColumnLynx::Net::TCP {
|
||||
|
||||
std::shared_ptr<MessageHandler> mHandler;
|
||||
std::function<void(std::shared_ptr<TCPConnection>)> mOnDisconnect;
|
||||
std::shared_ptr<Utils::LibSodiumWrapper> mLibSodiumWrapper;
|
||||
std::unordered_map<std::string, std::string>* mRawServerConfig;
|
||||
std::array<uint8_t, 32> mConnectionAESKey;
|
||||
uint32_t mConnectionSessionID;
|
||||
uint64_t mConnectionSessionID;
|
||||
AsymPublicKey mConnectionPublicKey;
|
||||
asio::steady_timer mHeartbeatTimer;
|
||||
std::chrono::steady_clock::time_point mLastHeartbeatReceived;
|
||||
std::chrono::steady_clock::time_point mLastHeartbeatSent;
|
||||
int mMissedHeartbeats = 0;
|
||||
std::string mRemoteIP; // Cached remote IP to avoid calling remote_endpoint() on closed sockets
|
||||
};
|
||||
}
|
||||
@@ -1,5 +1,5 @@
|
||||
// tcp_server.hpp - TCP Server for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
@@ -17,47 +17,41 @@
|
||||
#include <columnlynx/server/net/tcp/tcp_connection.hpp>
|
||||
#include <columnlynx/common/libsodium_wrapper.hpp>
|
||||
#include <columnlynx/common/net/protocol_structs.hpp>
|
||||
#include <columnlynx/server/server_session.hpp>
|
||||
|
||||
namespace ColumnLynx::Net::TCP {
|
||||
|
||||
class TCPServer {
|
||||
public:
|
||||
TCPServer(asio::io_context& ioContext,
|
||||
uint16_t port)
|
||||
uint16_t port,
|
||||
std::shared_ptr<Utils::LibSodiumWrapper> sodiumWrapper,
|
||||
std::shared_ptr<bool> hostRunning, bool ipv4Only = false)
|
||||
: mIoContext(ioContext),
|
||||
mAcceptor(ioContext)
|
||||
mAcceptor(ioContext),
|
||||
mSodiumWrapper(sodiumWrapper),
|
||||
mHostRunning(hostRunning)
|
||||
{
|
||||
// Preload the config map
|
||||
asio::error_code ec_open, ec_v6only, ec_bind;
|
||||
mRawServerConfig = Utils::getConfigMap("server_config", {"NETWORK", "SUBNET_MASK"});
|
||||
|
||||
bool isIPv4Only = ServerSession::getInstance().isIPv4Only();
|
||||
|
||||
if (!isIPv4Only) {
|
||||
// Try IPv6 (dual-stack if supported)
|
||||
asio::error_code ec;
|
||||
|
||||
if (!ipv4Only) {
|
||||
// Try IPv6 first (dual-stack check)
|
||||
asio::ip::tcp::endpoint endpoint_v6(asio::ip::tcp::v6(), port);
|
||||
|
||||
mAcceptor.open(endpoint_v6.protocol(), ec_open);
|
||||
|
||||
if (!ec_open) {
|
||||
// Try enabling dual-stack, but DO NOT treat failure as fatal
|
||||
mAcceptor.set_option(asio::ip::v6_only(false), ec_v6only);
|
||||
|
||||
// Try binding IPv6
|
||||
mAcceptor.bind(endpoint_v6, ec_bind);
|
||||
mAcceptor.open(endpoint_v6.protocol(), ec);
|
||||
if (!ec) {
|
||||
mAcceptor.set_option(asio::ip::v6_only(false), ec); // Allow dual-stack if possible
|
||||
mAcceptor.bind(endpoint_v6, ec);
|
||||
}
|
||||
}
|
||||
|
||||
// If IPv6 bind failed OR IPv6 open failed OR forced IPv4-only
|
||||
if (isIPv4Only || ec_open || ec_bind) {
|
||||
if (!isIPv4Only)
|
||||
Utils::warn("TCP: IPv6 unavailable (open=" + ec_open.message() +
|
||||
", bind=" + ec_bind.message() +
|
||||
"), falling back to IPv4 only");
|
||||
|
||||
// Fallback to IPv4 if anything failed
|
||||
if (ec || ipv4Only) {
|
||||
Utils::warn("TCP: IPv6 unavailable (" + ec.message() + "), falling back to IPv4 only");
|
||||
|
||||
asio::ip::tcp::endpoint endpoint_v4(asio::ip::tcp::v4(), port);
|
||||
|
||||
mAcceptor.close(); // guarantee clean state
|
||||
mAcceptor.close(); // ensure clean state
|
||||
mAcceptor.open(endpoint_v4.protocol());
|
||||
mAcceptor.bind(endpoint_v4);
|
||||
}
|
||||
@@ -78,6 +72,9 @@ namespace ColumnLynx::Net::TCP {
|
||||
asio::io_context &mIoContext;
|
||||
asio::ip::tcp::acceptor mAcceptor;
|
||||
std::unordered_set<TCPConnection::pointer> mClients;
|
||||
std::shared_ptr<Utils::LibSodiumWrapper> mSodiumWrapper;
|
||||
std::shared_ptr<bool> mHostRunning;
|
||||
std::unordered_map<std::string, std::string> mRawServerConfig;
|
||||
};
|
||||
|
||||
}
|
||||
@@ -1,5 +1,5 @@
|
||||
// udp_server.hpp - UDP Server for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
@@ -9,46 +9,31 @@
|
||||
#include <columnlynx/common/utils.hpp>
|
||||
#include <array>
|
||||
#include <columnlynx/common/net/virtual_interface.hpp>
|
||||
#include <columnlynx/common/libsodium_wrapper.hpp>
|
||||
#include <columnlynx/server/server_session.hpp>
|
||||
|
||||
namespace ColumnLynx::Net::UDP {
|
||||
class UDPServer {
|
||||
public:
|
||||
UDPServer(asio::io_context& ioContext, uint16_t port)
|
||||
: mSocket(ioContext)
|
||||
UDPServer(asio::io_context& ioContext, uint16_t port, std::shared_ptr<bool> hostRunning, bool ipv4Only = false, std::shared_ptr<VirtualInterface> tun = nullptr)
|
||||
: mSocket(ioContext), mHostRunning(hostRunning), mTun(tun)
|
||||
{
|
||||
asio::error_code ec_open, ec_v6only, ec_bind;
|
||||
asio::error_code ec;
|
||||
|
||||
if (!mIpv4Only) {
|
||||
if (!ipv4Only) {
|
||||
// Try IPv6 first (dual-stack check)
|
||||
asio::ip::udp::endpoint endpoint_v6(asio::ip::udp::v6(), port);
|
||||
|
||||
// Try opening IPv6 socket
|
||||
mSocket.open(endpoint_v6.protocol(), ec_open);
|
||||
|
||||
if (!ec_open) {
|
||||
// Try enabling dual-stack (non fatal if it fails)
|
||||
mSocket.set_option(asio::ip::v6_only(false), ec_v6only);
|
||||
|
||||
// Attempt bind
|
||||
mSocket.bind(endpoint_v6, ec_bind);
|
||||
mSocket.open(endpoint_v6.protocol(), ec);
|
||||
if (!ec) {
|
||||
mSocket.set_option(asio::ip::v6_only(false), ec); // Allow dual-stack if possible
|
||||
mSocket.bind(endpoint_v6, ec);
|
||||
}
|
||||
}
|
||||
|
||||
// Fallback to IPv4 if IPv6 is unusable
|
||||
if (mIpv4Only || ec_open || ec_bind) {
|
||||
if (!mIpv4Only) {
|
||||
Utils::warn(
|
||||
"UDP: IPv6 unavailable (open=" + ec_open.message() +
|
||||
", bind=" + ec_bind.message() +
|
||||
"), falling back to IPv4 only"
|
||||
);
|
||||
}
|
||||
|
||||
// Fallback to IPv4 if anything failed
|
||||
if (ec || ipv4Only) {
|
||||
Utils::warn("UDP: IPv6 unavailable (" + ec.message() + "), falling back to IPv4 only");
|
||||
|
||||
asio::ip::udp::endpoint endpoint_v4(asio::ip::udp::v4(), port);
|
||||
|
||||
mSocket.close();
|
||||
mSocket = asio::ip::udp::socket(ioContext); // fully reset internal state
|
||||
mSocket.close(); // ensure clean state
|
||||
mSocket.open(endpoint_v4.protocol());
|
||||
mSocket.bind(endpoint_v4);
|
||||
}
|
||||
@@ -61,7 +46,7 @@ namespace ColumnLynx::Net::UDP {
|
||||
void stop();
|
||||
|
||||
// Send UDP data to an endpoint; Fetched via the Session Registry
|
||||
void sendData(uint32_t sessionID, const std::string& data);
|
||||
void sendData(const uint64_t sessionID, const std::string& data);
|
||||
|
||||
private:
|
||||
// Start receiving UDP data
|
||||
@@ -72,7 +57,7 @@ namespace ColumnLynx::Net::UDP {
|
||||
asio::ip::udp::socket mSocket;
|
||||
asio::ip::udp::endpoint mRemoteEndpoint;
|
||||
std::array<uint8_t, 2048> mRecvBuffer; // 2048 seems stable
|
||||
bool mIpv4Only = ServerSession::getInstance().isIPv4Only();
|
||||
const std::shared_ptr<VirtualInterface> mTun = ServerSession::getInstance().getVirtualInterface();
|
||||
std::shared_ptr<bool> mHostRunning;
|
||||
std::shared_ptr<VirtualInterface> mTun;
|
||||
};
|
||||
}
|
||||
@@ -1,62 +0,0 @@
|
||||
// server_session.hpp - Client Session data for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <memory>
|
||||
#include <columnlynx/common/libsodium_wrapper.hpp>
|
||||
#include <array>
|
||||
#include <columnlynx/common/net/virtual_interface.hpp>
|
||||
#include <shared_mutex>
|
||||
|
||||
namespace ColumnLynx {
|
||||
struct ServerState {
|
||||
std::shared_ptr<Utils::LibSodiumWrapper> sodiumWrapper;
|
||||
std::shared_ptr<Net::VirtualInterface> virtualInterface;
|
||||
std::string configPath;
|
||||
std::unordered_map<std::string, std::string> serverConfig;
|
||||
bool ipv4Only;
|
||||
bool hostRunning;
|
||||
|
||||
~ServerState() = default;
|
||||
ServerState(const ServerState&) = delete;
|
||||
ServerState& operator=(const ServerState&) = delete;
|
||||
ServerState(ServerState&&) = default;
|
||||
ServerState& operator=(ServerState&&) = default;
|
||||
|
||||
explicit ServerState() = default;
|
||||
};
|
||||
|
||||
class ServerSession {
|
||||
public:
|
||||
// Return a reference to the Server Session instance
|
||||
static ServerSession& getInstance() { static ServerSession instance; return instance; }
|
||||
|
||||
// Return the current server state
|
||||
std::shared_ptr<ServerState> getServerState() const;
|
||||
|
||||
// Set the server state
|
||||
void setServerState(std::shared_ptr<ServerState> state);
|
||||
|
||||
// Getters
|
||||
std::shared_ptr<Utils::LibSodiumWrapper> getSodiumWrapper() const;
|
||||
const std::string& getConfigPath() const;
|
||||
const std::unordered_map<std::string, std::string>& getRawServerConfig() const;
|
||||
const std::shared_ptr<Net::VirtualInterface>& getVirtualInterface() const;
|
||||
bool isIPv4Only() const;
|
||||
bool isHostRunning() const;
|
||||
|
||||
// Setters
|
||||
void setSodiumWrapper(std::shared_ptr<Utils::LibSodiumWrapper> sodiumWrapper);
|
||||
void setConfigPath(const std::string& configPath);
|
||||
void setRawServerConfig(const std::unordered_map<std::string, std::string>& config);
|
||||
void setVirtualInterface(std::shared_ptr<Net::VirtualInterface> tun);
|
||||
void setIPv4Only(bool ipv4Only);
|
||||
void setHostRunning(bool hostRunning);
|
||||
|
||||
private:
|
||||
mutable std::shared_mutex mMutex;
|
||||
std::shared_ptr<struct ServerState> mServerState{nullptr};
|
||||
};
|
||||
}
|
||||
@@ -1,71 +0,0 @@
|
||||
// client_session.cpp - Client Session data for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#include <columnlynx/client/client_session.hpp>
|
||||
|
||||
namespace ColumnLynx {
|
||||
std::shared_ptr<ClientState> ClientSession::getClientState() const {
|
||||
std::shared_lock lock(mMutex);
|
||||
return mClientState;
|
||||
}
|
||||
|
||||
void ClientSession::setClientState(std::shared_ptr<ClientState> state) {
|
||||
std::unique_lock lock(mMutex);
|
||||
mClientState = state;
|
||||
}
|
||||
|
||||
const std::shared_ptr<Utils::LibSodiumWrapper>& ClientSession::getSodiumWrapper() const {
|
||||
return getClientState()->sodiumWrapper;
|
||||
}
|
||||
|
||||
const SymmetricKey& ClientSession::getAESKey() const {
|
||||
return getClientState()->aesKey;
|
||||
}
|
||||
|
||||
bool ClientSession::isInsecureMode() const {
|
||||
return getClientState()->insecureMode;
|
||||
}
|
||||
|
||||
const std::string& ClientSession::getConfigPath() const {
|
||||
return getClientState()->configPath;
|
||||
}
|
||||
|
||||
const std::shared_ptr<Net::VirtualInterface>& ClientSession::getVirtualInterface() const {
|
||||
return getClientState()->virtualInterface;
|
||||
}
|
||||
|
||||
uint32_t ClientSession::getSessionID() const {
|
||||
return getClientState()->sessionID;
|
||||
}
|
||||
|
||||
void ClientSession::setSodiumWrapper(std::shared_ptr<Utils::LibSodiumWrapper> sodiumWrapper) {
|
||||
std::unique_lock lock(mMutex);
|
||||
mClientState->sodiumWrapper = sodiumWrapper;
|
||||
}
|
||||
|
||||
void ClientSession::setAESKey(const SymmetricKey& aesKey) {
|
||||
std::unique_lock lock(mMutex);
|
||||
mClientState->aesKey = aesKey;
|
||||
}
|
||||
|
||||
void ClientSession::setInsecureMode(bool insecureMode) {
|
||||
std::unique_lock lock(mMutex);
|
||||
mClientState->insecureMode = insecureMode;
|
||||
}
|
||||
|
||||
void ClientSession::setConfigPath(const std::string& configPath) {
|
||||
std::unique_lock lock(mMutex);
|
||||
mClientState->configPath = configPath;
|
||||
}
|
||||
|
||||
void ClientSession::setVirtualInterface(std::shared_ptr<Net::VirtualInterface> virtualInterface) {
|
||||
std::unique_lock lock(mMutex);
|
||||
mClientState->virtualInterface = virtualInterface;
|
||||
}
|
||||
|
||||
void ClientSession::setSessionID(uint32_t sessionID) {
|
||||
std::unique_lock lock(mMutex);
|
||||
mClientState->sessionID = sessionID;
|
||||
}
|
||||
}
|
||||
@@ -1,5 +1,5 @@
|
||||
// main.cpp - Client entry point for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#include <asio.hpp>
|
||||
@@ -11,12 +11,6 @@
|
||||
#include <columnlynx/client/net/udp/udp_client.hpp>
|
||||
#include <cxxopts.hpp>
|
||||
#include <columnlynx/common/net/virtual_interface.hpp>
|
||||
#include <columnlynx/client/client_session.hpp>
|
||||
#include <thread>
|
||||
|
||||
#if defined(__WIN32__)
|
||||
#include <windows.h>
|
||||
#endif
|
||||
|
||||
using asio::ip::tcp;
|
||||
using namespace ColumnLynx::Utils;
|
||||
@@ -27,19 +21,18 @@ volatile sig_atomic_t done = 0;
|
||||
|
||||
void signalHandler(int signum) {
|
||||
if (signum == SIGINT || signum == SIGTERM) {
|
||||
//log("Received termination signal. Shutting down client.");
|
||||
done = 1;
|
||||
}
|
||||
}
|
||||
|
||||
int main(int argc, char** argv) {
|
||||
// Capture SIGINT and SIGTERM for graceful shutdown
|
||||
#if !defined(_WIN32)
|
||||
struct sigaction action;
|
||||
memset(&action, 0, sizeof(struct sigaction));
|
||||
action.sa_handler = signalHandler;
|
||||
sigaction(SIGINT, &action, nullptr);
|
||||
sigaction(SIGTERM, &action, nullptr);
|
||||
#endif
|
||||
|
||||
PanicHandler::init();
|
||||
|
||||
@@ -54,23 +47,15 @@ int main(int argc, char** argv) {
|
||||
#else
|
||||
("i,interface", "Override used interface", cxxopts::value<std::string>()->default_value("lynx0"))
|
||||
#endif
|
||||
("ignore-whitelist", "Ignore if server is not in whitelisted_keys", cxxopts::value<bool>()->default_value("false"))
|
||||
#if defined(__WIN32__)
|
||||
/* Get config dir in LOCALAPPDATA\ColumnLynx\ */
|
||||
("config-dir", "Override config dir path", cxxopts::value<std::string>()->default_value(std::string((std::getenv("LOCALAPPDATA") ? std::getenv("LOCALAPPDATA") : "C:\\ProgramData")) + "\\ColumnLynx\\"));
|
||||
#elif defined(__APPLE__)
|
||||
("config-dir", "Override config dir path", cxxopts::value<std::string>()->default_value(std::string((std::getenv("HOME") ? std::getenv("HOME") : "")) + "/Library/Application Support/ColumnLynx/"));
|
||||
#else
|
||||
("config-dir", "Override config dir path", cxxopts::value<std::string>()->default_value(std::string((std::getenv("SUDO_USER") ? "/home/" + std::string(std::getenv("SUDO_USER")) : (std::getenv("HOME") ? std::getenv("HOME") : ""))) + "/.config/columnlynx/"));
|
||||
#endif
|
||||
("allow-selfsigned", "Allow self-signed certificates", cxxopts::value<bool>()->default_value("false"));
|
||||
|
||||
bool insecureMode = options.parse(argc, argv).count("ignore-whitelist") > 0;
|
||||
bool insecureMode = options.parse(argc, argv).count("allow-selfsigned") > 0;
|
||||
|
||||
auto optionsObj = options.parse(argc, argv);
|
||||
if (optionsObj.count("help")) {
|
||||
std::cout << options.help() << std::endl;
|
||||
std::cout << "This software is licensed under the GPLv2-only license OR the GPLv3 license.\n";
|
||||
std::cout << "Copyright (C) 2026, The ColumnLynx Contributors.\n";
|
||||
std::cout << "Copyright (C) 2025, The ColumnLynx Contributors.\n";
|
||||
std::cout << "This software is provided under ABSOLUTELY NO WARRANTY, to the extent permitted by law.\n";
|
||||
return 0;
|
||||
}
|
||||
@@ -83,55 +68,23 @@ int main(int argc, char** argv) {
|
||||
log("This software is licensed under the GPLv2 only OR the GPLv3. See LICENSES/ for details.");
|
||||
|
||||
#if defined(__WIN32__)
|
||||
//WintunInitialize();
|
||||
WintunInitialize();
|
||||
#endif
|
||||
|
||||
// Get the config path, ENV > CLI > /etc/columnlynx
|
||||
std::string configPath = optionsObj["config-dir"].as<std::string>();
|
||||
const char* envConfigPath = std::getenv("COLUMNLYNX_CONFIG_DIR");
|
||||
if (envConfigPath != nullptr) {
|
||||
configPath = std::string(envConfigPath);
|
||||
}
|
||||
|
||||
if (configPath.back() != '/' && configPath.back() != '\\') {
|
||||
#if defined(__WIN32__)
|
||||
configPath += "\\";
|
||||
#else
|
||||
configPath += "/";
|
||||
#endif
|
||||
}
|
||||
|
||||
struct ClientState initialState{};
|
||||
initialState.configPath = configPath;
|
||||
initialState.insecureMode = insecureMode;
|
||||
initialState.send_cnt = 0;
|
||||
initialState.recv_cnt = 0;
|
||||
randombytes_buf(&initialState.noncePrefix, sizeof(uint32_t)); // Randomize nonce prefix
|
||||
|
||||
std::shared_ptr<VirtualInterface> tun = std::make_shared<VirtualInterface>(optionsObj["interface"].as<std::string>());
|
||||
log("Using virtual interface: " + tun->getName());
|
||||
initialState.virtualInterface = tun;
|
||||
|
||||
std::shared_ptr<LibSodiumWrapper> sodiumWrapper = std::make_shared<LibSodiumWrapper>();
|
||||
debug("Public Key: " + Utils::bytesToHexString(sodiumWrapper->getPublicKey(), 32));
|
||||
debug("Private Key: " + Utils::bytesToHexString(sodiumWrapper->getPrivateKey(), 64));
|
||||
initialState.sodiumWrapper = sodiumWrapper;
|
||||
|
||||
std::array<uint8_t, 32> aesKey = std::array<uint8_t, 32>();
|
||||
aesKey.fill(0); // Defualt zeroed state until modified by handshake
|
||||
uint32_t sessionID = 0;
|
||||
initialState.aesKey = aesKey;
|
||||
initialState.sessionID = sessionID;
|
||||
|
||||
ColumnLynx::ClientSession::getInstance().setClientState(std::make_shared<ClientState>(std::move(initialState))); // Set initial state
|
||||
|
||||
if (insecureMode) {
|
||||
warn("You have started the client with the --ignore-whitelist. This means that the client will NOT attempt to verify the server's public key. This is INSECURE and SHOULDN'T be used!");
|
||||
}
|
||||
std::shared_ptr<std::array<uint8_t, 32>> aesKey = std::make_shared<std::array<uint8_t, 32>>();
|
||||
aesKey->fill(0); // Defualt zeroed state until modified by handshake
|
||||
std::shared_ptr<uint64_t> sessionID = std::make_shared<uint64_t>(0);
|
||||
|
||||
asio::io_context io;
|
||||
auto client = std::make_shared<ColumnLynx::Net::TCP::TCPClient>(io, host, port); // TODO: Move to ClientSession state
|
||||
auto udpClient = std::make_shared<ColumnLynx::Net::UDP::UDPClient>(io, host, port);
|
||||
auto client = std::make_shared<ColumnLynx::Net::TCP::TCPClient>(io, host, port, sodiumWrapper, aesKey, sessionID, insecureMode, tun);
|
||||
auto udpClient = std::make_shared<ColumnLynx::Net::UDP::UDPClient>(io, host, port, aesKey, sessionID, tun);
|
||||
|
||||
client->start();
|
||||
udpClient->start();
|
||||
@@ -142,18 +95,14 @@ int main(int argc, char** argv) {
|
||||
});
|
||||
//ioThread.join();
|
||||
|
||||
log("Attempting connection to " + host + ":" + port);
|
||||
debug("Client connection flag: " + std::to_string(client->isConnected()));
|
||||
debug("Client handshake flag: " + std::to_string(client->isHandshakeComplete()));
|
||||
debug("isDone flag: " + std::to_string(done));
|
||||
log("Client connected to " + host + ":" + port);
|
||||
|
||||
// Client is running
|
||||
while ((client->isConnected() || !client->isHandshakeComplete()) && !done) {
|
||||
//debug("Client connection flag: " + std::to_string(client->isConnected()));
|
||||
auto packet = tun->readPacket();
|
||||
/*if (!client->isConnected() || done) {
|
||||
if (!client->isConnected() || done) {
|
||||
break; // Bail out if connection died or signal set while blocked
|
||||
}*/
|
||||
}
|
||||
|
||||
if (packet.empty()) {
|
||||
continue;
|
||||
|
||||
@@ -1,9 +1,9 @@
|
||||
// tcp_client.cpp - TCP Client for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#include <columnlynx/client/net/tcp/tcp_client.hpp>
|
||||
//#include <arpa/inet.h>
|
||||
#include <arpa/inet.h>
|
||||
|
||||
namespace ColumnLynx::Net::TCP {
|
||||
void TCPClient::start() {
|
||||
@@ -13,46 +13,30 @@ namespace ColumnLynx::Net::TCP {
|
||||
if (!ec) {
|
||||
asio::async_connect(mSocket, endpoints,
|
||||
[this, self](asio::error_code ec, const tcp::endpoint&) {
|
||||
if (!ec) {
|
||||
if (!NetHelper::isExpectedDisconnect(ec)) {
|
||||
mConnected = true;
|
||||
Utils::log("Client connected.");
|
||||
mHandler = std::make_shared<MessageHandler>(std::move(mSocket));
|
||||
mHandler->onMessage([this](AnyMessageType type, const std::string& data) {
|
||||
mHandleMessage(static_cast<ServerMessageType>(MessageHandler::toUint8(type)), data);
|
||||
});
|
||||
// Close only after peer FIN to avoid RSTs
|
||||
mHandler->onDisconnect([this](const asio::error_code& ec) {
|
||||
asio::error_code ec2;
|
||||
if (mHandler) {
|
||||
mHandler->socket().close(ec2);
|
||||
}
|
||||
mConnected = false;
|
||||
Utils::log(std::string("Server disconnected: ") + ec.message());
|
||||
});
|
||||
mHandler->start();
|
||||
|
||||
// Init connection handshake
|
||||
Utils::log("Sending handshake init to server.");
|
||||
|
||||
// Check if hostname or IPv4/IPv6
|
||||
try {
|
||||
asio::ip::make_address(mHost);
|
||||
self->mIsHostDomain = false; // IPv4 or IPv6 literal
|
||||
} catch (const asio::system_error&) {
|
||||
self->mIsHostDomain = true; // hostname / domain
|
||||
}
|
||||
|
||||
sockaddr_in addr4{};
|
||||
sockaddr_in6 addr6{};
|
||||
self->mIsHostDomain = inet_pton(AF_INET, mHost.c_str(), (void*)(&addr4)) != 1 && inet_pton(AF_INET6, mHost.c_str(), (void*)(&addr6)) != 1; // Voodoo black magic
|
||||
|
||||
std::vector<uint8_t> payload;
|
||||
payload.reserve(1 + crypto_box_PUBLICKEYBYTES);
|
||||
payload.push_back(Utils::protocolVersion());
|
||||
Utils::log("Using protocol version: " + std::to_string(Utils::protocolVersion()));
|
||||
/*payload.insert(payload.end(),
|
||||
mLibSodiumWrapper->getXPublicKey(),
|
||||
mLibSodiumWrapper->getXPublicKey() + crypto_box_PUBLICKEYBYTES
|
||||
);*/
|
||||
|
||||
const auto& mLibSodiumWrapper = ClientSession::getInstance().getSodiumWrapper();
|
||||
payload.insert(payload.end(),
|
||||
mLibSodiumWrapper->getPublicKey(),
|
||||
mLibSodiumWrapper->getPublicKey() + crypto_sign_PUBLICKEYBYTES
|
||||
@@ -62,9 +46,7 @@ namespace ColumnLynx::Net::TCP {
|
||||
|
||||
mStartHeartbeat();
|
||||
} else {
|
||||
if (!NetHelper::isExpectedDisconnect(ec)) {
|
||||
Utils::error("Client connect failed: " + ec.message());
|
||||
}
|
||||
Utils::error("Client connect failed: " + ec.message());
|
||||
}
|
||||
});
|
||||
} else {
|
||||
@@ -95,14 +77,18 @@ namespace ColumnLynx::Net::TCP {
|
||||
asio::error_code ec;
|
||||
mHeartbeatTimer.cancel();
|
||||
|
||||
// Half-close: stop sending, keep reading until peer FIN
|
||||
mHandler->socket().shutdown(tcp::socket::shutdown_send, ec);
|
||||
mHandler->socket().shutdown(tcp::socket::shutdown_both, ec);
|
||||
if (ec) {
|
||||
Utils::error("Error during socket shutdown: " + ec.message());
|
||||
}
|
||||
|
||||
// Do not close immediately; rely on onDisconnect to finalize
|
||||
Utils::log("Client initiated graceful disconnect (half-close).");
|
||||
mHandler->socket().close(ec);
|
||||
if (ec) {
|
||||
Utils::error("Error during socket close: " + ec.message());
|
||||
}
|
||||
|
||||
mConnected = false;
|
||||
Utils::log("Client disconnected.");
|
||||
}
|
||||
}
|
||||
|
||||
@@ -133,9 +119,11 @@ namespace ColumnLynx::Net::TCP {
|
||||
mHandler->socket().shutdown(tcp::socket::shutdown_both, ec);
|
||||
mHandler->socket().close(ec);
|
||||
mConnected = false;
|
||||
|
||||
ClientSession::getInstance().setAESKey({}); // Clear AES key with all zeros
|
||||
ClientSession::getInstance().setSessionID(0);
|
||||
|
||||
mGlobalKeyRef = nullptr;
|
||||
if (mSessionIDRef) {
|
||||
*mSessionIDRef = 0;
|
||||
}
|
||||
|
||||
return;
|
||||
}
|
||||
@@ -151,15 +139,13 @@ namespace ColumnLynx::Net::TCP {
|
||||
void TCPClient::mHandleMessage(ServerMessageType type, const std::string& data) {
|
||||
switch (type) {
|
||||
case ServerMessageType::HANDSHAKE_IDENTIFY: {
|
||||
Utils::log("Received server identity: " + data);
|
||||
std::memcpy(mServerPublicKey, data.data(), std::min(data.size(), sizeof(mServerPublicKey)));
|
||||
std::string hexServerPub = Utils::bytesToHexString(mServerPublicKey, 32);
|
||||
Utils::log("Received server identity. Public Key: " + hexServerPub);
|
||||
|
||||
// Verify pubkey against whitelisted_keys
|
||||
const std::string& configPath = ClientSession::getInstance().getConfigPath();
|
||||
std::vector<std::string> whitelistedKeys = Utils::getWhitelistedKeys(configPath);
|
||||
std::vector<std::string> whitelistedKeys = Utils::getWhitelistedKeys();
|
||||
if (std::find(whitelistedKeys.begin(), whitelistedKeys.end(), Utils::bytesToHexString(mServerPublicKey, 32)) == whitelistedKeys.end()) { // Key verification is handled in later steps of the handshake
|
||||
if (!ClientSession::getInstance().isInsecureMode()) {
|
||||
if (!mInsecureMode) {
|
||||
Utils::error("Server public key not in whitelisted_keys. Terminating connection.");
|
||||
disconnect();
|
||||
return;
|
||||
@@ -195,14 +181,16 @@ namespace ColumnLynx::Net::TCP {
|
||||
|
||||
// Generate AES key and send confirmation
|
||||
mConnectionAESKey = Utils::LibSodiumWrapper::generateRandom256Bit();
|
||||
ClientSession::getInstance().setAESKey(mConnectionAESKey);
|
||||
if (mGlobalKeyRef) { // Copy to the global reference
|
||||
std::copy(mConnectionAESKey.begin(), mConnectionAESKey.end(), mGlobalKeyRef->begin());
|
||||
}
|
||||
AsymNonce nonce{};
|
||||
randombytes_buf(nonce.data(), nonce.size());
|
||||
|
||||
// TODO: This is pretty redundant, it should return the required type directly
|
||||
std::array<uint8_t, 32> arrayPrivateKey;
|
||||
std::copy(ClientSession::getInstance().getSodiumWrapper()->getXPrivateKey(),
|
||||
ClientSession::getInstance().getSodiumWrapper()->getXPrivateKey() + 32,
|
||||
std::copy(mLibSodiumWrapper->getXPrivateKey(),
|
||||
mLibSodiumWrapper->getXPrivateKey() + 32,
|
||||
arrayPrivateKey.begin());
|
||||
|
||||
std::vector<uint8_t> encr = Utils::LibSodiumWrapper::encryptAsymmetric(
|
||||
@@ -245,18 +233,19 @@ namespace ColumnLynx::Net::TCP {
|
||||
std::memcpy(&mConnectionSessionID, decrypted.data(), sizeof(mConnectionSessionID));
|
||||
std::memcpy(&mTunConfig, decrypted.data() + sizeof(mConnectionSessionID), sizeof(Protocol::TunConfig));
|
||||
|
||||
mConnectionSessionID = ntohl(mConnectionSessionID);
|
||||
mConnectionSessionID = Utils::cbe64toh(mConnectionSessionID);
|
||||
|
||||
Utils::log("Connection established with Session ID: " + std::to_string(mConnectionSessionID));
|
||||
|
||||
ClientSession::getInstance().setSessionID(mConnectionSessionID);
|
||||
if (mSessionIDRef) { // Copy to the global reference
|
||||
*mSessionIDRef = mConnectionSessionID;
|
||||
}
|
||||
|
||||
uint32_t clientIP = ntohl(mTunConfig.clientIP);
|
||||
uint32_t serverIP = ntohl(mTunConfig.serverIP);
|
||||
uint8_t prefixLen = mTunConfig.prefixLength;
|
||||
uint16_t mtu = mTunConfig.mtu;
|
||||
|
||||
const auto& mTun = ClientSession::getInstance().getVirtualInterface();
|
||||
if (mTun) {
|
||||
mTun->configureIP(clientIP, serverIP, prefixLen, mtu);
|
||||
}
|
||||
@@ -280,12 +269,6 @@ namespace ColumnLynx::Net::TCP {
|
||||
disconnect(false);
|
||||
}
|
||||
break;
|
||||
case ServerMessageType::KILL_CONNECTION:
|
||||
Utils::warn("Server is killing the connection: " + data);
|
||||
if (mConnected) {
|
||||
disconnect(false);
|
||||
}
|
||||
break;
|
||||
default:
|
||||
Utils::log("Received unknown message type from server.");
|
||||
break;
|
||||
|
||||
@@ -1,59 +1,23 @@
|
||||
// udp_client.cpp - UDP Client for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#include <columnlynx/client/net/udp/udp_client.hpp>
|
||||
|
||||
namespace ColumnLynx::Net::UDP {
|
||||
void UDPClient::start() {
|
||||
asio::error_code ec;
|
||||
|
||||
// Resolve using an unspecified protocol (allows both IPv4 and IPv6)
|
||||
auto endpoints = mResolver.resolve(
|
||||
asio::ip::udp::v6(), // Try IPv6 first (dual-stack with v4)
|
||||
mHost,
|
||||
mPort,
|
||||
ec
|
||||
);
|
||||
|
||||
if (ec) {
|
||||
// If IPv6 fails (host has no AAAA), try IPv4
|
||||
endpoints = mResolver.resolve(
|
||||
asio::ip::udp::v4(),
|
||||
mHost,
|
||||
mPort,
|
||||
ec
|
||||
);
|
||||
}
|
||||
|
||||
if (ec) {
|
||||
Utils::error("UDP resolve failed: " + ec.message());
|
||||
return;
|
||||
}
|
||||
|
||||
// Use whichever endpoint resolved
|
||||
// TODO: Add IPv6
|
||||
auto endpoints = mResolver.resolve(asio::ip::udp::v4(), mHost, mPort);
|
||||
mRemoteEndpoint = *endpoints.begin();
|
||||
|
||||
// Open socket using the resolved endpoint's protocol
|
||||
mSocket.open(mRemoteEndpoint.protocol(), ec);
|
||||
if (ec) {
|
||||
Utils::error("UDP socket open failed: " + ec.message());
|
||||
return;
|
||||
}
|
||||
|
||||
mSocket.open(asio::ip::udp::v4());
|
||||
Utils::log("UDP Client ready to send to " + mRemoteEndpoint.address().to_string() + ":" + std::to_string(mRemoteEndpoint.port()));
|
||||
}
|
||||
|
||||
void UDPClient::sendMessage(const std::string& data) {
|
||||
UDPPacketHeader hdr{};
|
||||
uint8_t nonce[12];
|
||||
uint32_t prefix = ClientSession::getInstance().getNoncePrefix();
|
||||
uint64_t sendCount = ClientSession::getInstance().getSendCount();
|
||||
memcpy(nonce, &prefix, sizeof(uint32_t)); // Prefix nonce with client-specific random value
|
||||
memcpy(nonce + sizeof(uint32_t), &sendCount, sizeof(uint64_t)); // Use send count as nonce suffix to ensure uniqueness
|
||||
std::copy_n(nonce, 12, hdr.nonce.data());
|
||||
randombytes_buf(hdr.nonce.data(), hdr.nonce.size());
|
||||
|
||||
if (ClientSession::getInstance().getAESKey().empty() || ClientSession::getInstance().getSessionID() == 0) {
|
||||
if (mAesKeyRef == nullptr || mSessionIDRef == nullptr) {
|
||||
Utils::error("UDP Client AES key or Session ID reference is null!");
|
||||
return;
|
||||
}
|
||||
@@ -62,28 +26,24 @@ namespace ColumnLynx::Net::UDP {
|
||||
|
||||
auto encryptedPayload = Utils::LibSodiumWrapper::encryptMessage(
|
||||
reinterpret_cast<const uint8_t*>(data.data()), data.size(),
|
||||
ClientSession::getInstance().getAESKey(), hdr.nonce, "udp-data"
|
||||
*mAesKeyRef, hdr.nonce, "udp-data"
|
||||
//std::string(reinterpret_cast<const char*>(&mSessionIDRef), sizeof(uint64_t))
|
||||
);
|
||||
|
||||
std::vector<uint8_t> packet;
|
||||
packet.reserve(sizeof(UDPPacketHeader) + encryptedPayload.size());
|
||||
packet.reserve(sizeof(UDPPacketHeader) + sizeof(uint64_t) + encryptedPayload.size());
|
||||
packet.insert(packet.end(),
|
||||
reinterpret_cast<uint8_t*>(&hdr),
|
||||
reinterpret_cast<uint8_t*>(&hdr) + sizeof(UDPPacketHeader)
|
||||
);
|
||||
uint32_t sessionID = static_cast<uint32_t>(ClientSession::getInstance().getSessionID());
|
||||
uint32_t sessionIDNet = sessionID;
|
||||
packet.insert(packet.end(),
|
||||
reinterpret_cast<uint8_t*>(&sessionIDNet),
|
||||
reinterpret_cast<uint8_t*>(&sessionIDNet) + sizeof(uint32_t)
|
||||
reinterpret_cast<uint8_t*>(mSessionIDRef.get()),
|
||||
reinterpret_cast<uint8_t*>(mSessionIDRef.get()) + sizeof(uint64_t)
|
||||
);
|
||||
packet.insert(packet.end(), encryptedPayload.begin(), encryptedPayload.end());
|
||||
|
||||
mSocket.send_to(asio::buffer(packet), mRemoteEndpoint);
|
||||
Utils::debug("Sent UDP packet of size " + std::to_string(packet.size()));
|
||||
|
||||
ClientSession::getInstance().incrementSendCount();
|
||||
}
|
||||
|
||||
void UDPClient::stop() {
|
||||
@@ -116,7 +76,7 @@ namespace ColumnLynx::Net::UDP {
|
||||
}
|
||||
|
||||
void UDPClient::mHandlePacket(std::size_t bytes) {
|
||||
if (bytes < sizeof(UDPPacketHeader) + sizeof(uint32_t)) {
|
||||
if (bytes < sizeof(UDPPacketHeader) + sizeof(uint64_t)) {
|
||||
Utils::warn("UDP Client received packet too small to process.");
|
||||
return;
|
||||
}
|
||||
@@ -126,28 +86,27 @@ namespace ColumnLynx::Net::UDP {
|
||||
std::memcpy(&hdr, mRecvBuffer.data(), sizeof(UDPPacketHeader));
|
||||
|
||||
// Parse session ID
|
||||
uint32_t sessionIDNet;
|
||||
std::memcpy(&sessionIDNet, mRecvBuffer.data() + sizeof(UDPPacketHeader), sizeof(uint32_t));
|
||||
uint32_t sessionID = ntohl(sessionIDNet);
|
||||
uint64_t sessionID;
|
||||
std::memcpy(&sessionID, mRecvBuffer.data() + sizeof(UDPPacketHeader), sizeof(uint64_t));
|
||||
|
||||
if (sessionID != ClientSession::getInstance().getSessionID()) {
|
||||
Utils::warn("This packet that isn't for me! Dropping!");
|
||||
if (sessionID != *mSessionIDRef) {
|
||||
Utils::warn("Got packet that isn't for me! Dropping!");
|
||||
return;
|
||||
}
|
||||
|
||||
// Decrypt payload
|
||||
std::vector<uint8_t> ciphertext(
|
||||
mRecvBuffer.begin() + sizeof(UDPPacketHeader) + sizeof(uint32_t),
|
||||
mRecvBuffer.begin() + sizeof(UDPPacketHeader) + sizeof(uint64_t),
|
||||
mRecvBuffer.begin() + bytes
|
||||
);
|
||||
|
||||
if (ClientSession::getInstance().getAESKey().empty()) {
|
||||
if (mAesKeyRef == nullptr) {
|
||||
Utils::error("UDP Client AES key reference is null!");
|
||||
return;
|
||||
}
|
||||
|
||||
std::vector<uint8_t> plaintext = Utils::LibSodiumWrapper::decryptMessage(
|
||||
ciphertext.data(), ciphertext.size(), ClientSession::getInstance().getAESKey(), hdr.nonce, "udp-data"
|
||||
ciphertext.data(), ciphertext.size(), *mAesKeyRef, hdr.nonce, "udp-data"
|
||||
//std::string(reinterpret_cast<const char*>(&mSessionIDRef), sizeof(uint64_t))
|
||||
);
|
||||
|
||||
@@ -159,7 +118,6 @@ namespace ColumnLynx::Net::UDP {
|
||||
Utils::debug("UDP Client received packet from " + mRemoteEndpoint.address().to_string() + " - Packet size: " + std::to_string(bytes));
|
||||
|
||||
// Write to TUN
|
||||
const auto& mTunRef = ClientSession::getInstance().getVirtualInterface();
|
||||
if (mTunRef) {
|
||||
mTunRef->writePacket(plaintext);
|
||||
}
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
// libsodium_wrapper.cpp - Libsodium Wrapper for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#include <columnlynx/common/libsodium_wrapper.hpp>
|
||||
@@ -41,27 +41,4 @@ namespace ColumnLynx::Utils {
|
||||
randombytes_buf(randbytes.data(), randbytes.size());
|
||||
return randbytes;
|
||||
}
|
||||
|
||||
bool LibSodiumWrapper::recomputeKeys(PrivateSeed privateSeed, PublicKey storedPubKey) {
|
||||
int res = crypto_sign_seed_keypair(mPublicKey.data(), mPrivateKey.data(), privateSeed.data());
|
||||
|
||||
if (res != 0) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// Convert to Curve25519 keys for encryption
|
||||
res = crypto_sign_ed25519_pk_to_curve25519(mXPublicKey.data(), mPublicKey.data());
|
||||
res = crypto_sign_ed25519_sk_to_curve25519(mXPrivateKey.data(), mPrivateKey.data());
|
||||
|
||||
if (res != 0) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// Compare to stored for verification
|
||||
if (sodium_memcmp(mPublicKey.data(), storedPubKey.data(), crypto_sign_PUBLICKEYBYTES) != 0) {
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
}
|
||||
@@ -1,17 +1,17 @@
|
||||
// session_registry.cpp - Session Registry for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#include <columnlynx/common/net/session_registry.hpp>
|
||||
|
||||
namespace ColumnLynx::Net {
|
||||
void SessionRegistry::put(uint32_t sessionID, std::shared_ptr<SessionState> state) {
|
||||
void SessionRegistry::put(uint64_t sessionID, std::shared_ptr<SessionState> state) {
|
||||
std::unique_lock lock(mMutex);
|
||||
mSessions[sessionID] = std::move(state);
|
||||
mIPSessions[mSessions[sessionID]->clientTunIP] = mSessions[sessionID];
|
||||
}
|
||||
|
||||
std::shared_ptr<const SessionState> SessionRegistry::get(uint32_t sessionID) const {
|
||||
std::shared_ptr<const SessionState> SessionRegistry::get(uint64_t sessionID) const {
|
||||
std::shared_lock lock(mMutex);
|
||||
auto it = mSessions.find(sessionID);
|
||||
return (it == mSessions.end()) ? nullptr : it->second;
|
||||
@@ -23,14 +23,14 @@ namespace ColumnLynx::Net {
|
||||
return (it == mIPSessions.end()) ? nullptr : it->second;
|
||||
}
|
||||
|
||||
std::unordered_map<uint32_t, std::shared_ptr<SessionState>> SessionRegistry::snapshot() const {
|
||||
std::unordered_map<uint32_t, std::shared_ptr<SessionState>> snap;
|
||||
std::unordered_map<uint64_t, std::shared_ptr<SessionState>> SessionRegistry::snapshot() const {
|
||||
std::unordered_map<uint64_t, std::shared_ptr<SessionState>> snap;
|
||||
std::shared_lock lock(mMutex);
|
||||
snap = mSessions;
|
||||
return snap;
|
||||
}
|
||||
|
||||
void SessionRegistry::erase(uint32_t sessionID) {
|
||||
void SessionRegistry::erase(uint64_t sessionID) {
|
||||
std::unique_lock lock(mMutex);
|
||||
mSessions.erase(sessionID);
|
||||
}
|
||||
@@ -60,11 +60,6 @@ namespace ColumnLynx::Net {
|
||||
return static_cast<int>(mSessions.size());
|
||||
}
|
||||
|
||||
bool SessionRegistry::exists(uint32_t sessionID) const {
|
||||
std::shared_lock lock(mMutex);
|
||||
return mSessions.find(sessionID) != mSessions.end();
|
||||
}
|
||||
|
||||
uint32_t SessionRegistry::getFirstAvailableIP(uint32_t baseIP, uint8_t mask) const {
|
||||
std::shared_lock lock(mMutex);
|
||||
|
||||
@@ -82,7 +77,7 @@ namespace ColumnLynx::Net {
|
||||
return 0;
|
||||
}
|
||||
|
||||
void SessionRegistry::lockIP(uint32_t sessionID, uint32_t ip) {
|
||||
void SessionRegistry::lockIP(uint64_t sessionID, uint32_t ip) {
|
||||
std::unique_lock lock(mMutex);
|
||||
mSessionIPs[sessionID] = ip;
|
||||
|
||||
@@ -92,7 +87,7 @@ namespace ColumnLynx::Net {
|
||||
mIPSessions[ip] = mSessions.find(sessionID)->second;
|
||||
}
|
||||
|
||||
void SessionRegistry::deallocIP(uint32_t sessionID) {
|
||||
void SessionRegistry::deallocIP(uint64_t sessionID) {
|
||||
std::unique_lock lock(mMutex);
|
||||
|
||||
auto it = mSessionIPs.find(sessionID);
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
// tcp_message_handler.cpp - TCP Message Handler for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#include <columnlynx/common/net/tcp/tcp_message_handler.hpp>
|
||||
@@ -43,19 +43,13 @@ namespace ColumnLynx::Net::TCP {
|
||||
auto self = shared_from_this();
|
||||
asio::async_read(mSocket, asio::buffer(mHeader),
|
||||
[this, self](asio::error_code ec, std::size_t) {
|
||||
if (!ec) {
|
||||
if (!NetHelper::isExpectedDisconnect(ec)) {
|
||||
mCurrentType = decodeMessageType(mHeader[0]);
|
||||
|
||||
uint16_t len = (mHeader[1] << 8) | mHeader[2];
|
||||
mReadBody(len);
|
||||
} else {
|
||||
if (!NetHelper::isExpectedDisconnect(ec)) {
|
||||
Utils::error("Header read failed: " + ec.message());
|
||||
}
|
||||
// Connection closed, trigger disconnect handler
|
||||
if (mOnDisconnect) {
|
||||
mOnDisconnect(ec);
|
||||
}
|
||||
Utils::error("Header read failed: " + ec.message());
|
||||
}
|
||||
}
|
||||
);
|
||||
@@ -67,7 +61,7 @@ namespace ColumnLynx::Net::TCP {
|
||||
|
||||
asio::async_read(mSocket, asio::buffer(mBody),
|
||||
[this, self](asio::error_code ec, std::size_t) {
|
||||
if (!ec) {
|
||||
if (!NetHelper::isExpectedDisconnect(ec)) {
|
||||
std::string payload(mBody.begin(), mBody.end());
|
||||
|
||||
// Dispatch based on message type
|
||||
@@ -77,10 +71,8 @@ namespace ColumnLynx::Net::TCP {
|
||||
|
||||
mReadHeader(); // Keep listening
|
||||
} else {
|
||||
if (!NetHelper::isExpectedDisconnect(ec)) {
|
||||
Utils::error("Body read failed: " + ec.message());
|
||||
}
|
||||
// Connection closed, trigger disconnect handler
|
||||
Utils::error("Body read failed: " + ec.message());
|
||||
|
||||
if (mOnDisconnect) {
|
||||
mOnDisconnect(ec);
|
||||
}
|
||||
|
||||
@@ -1,65 +1,29 @@
|
||||
// utils.cpp - Utility functions for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#include <columnlynx/common/utils.hpp>
|
||||
|
||||
namespace ColumnLynx::Utils {
|
||||
std::string unixMillisToISO8601(uint64_t unixMillis, bool local) {
|
||||
using namespace std::chrono;
|
||||
|
||||
// Convert milliseconds since epoch to system_clock::time_point
|
||||
system_clock::time_point tp = system_clock::time_point(milliseconds(unixMillis));
|
||||
|
||||
// Convert to time_t for localtime conversion
|
||||
std::time_t tt = system_clock::to_time_t(tp);
|
||||
std::tm localTm;
|
||||
|
||||
if (local) {
|
||||
#ifdef _WIN32
|
||||
localtime_s(&localTm, &tt);
|
||||
#else
|
||||
localtime_r(&tt, &localTm);
|
||||
#endif
|
||||
} else {
|
||||
#ifdef _WIN32
|
||||
gmtime_s(&localTm, &tt);
|
||||
#else
|
||||
gmtime_r(&tt, &localTm);
|
||||
#endif
|
||||
}
|
||||
|
||||
// Format the time to ISO 8601
|
||||
char buffer[30];
|
||||
std::strftime(buffer, sizeof(buffer), "%Y-%m-%dT%H:%M:%S", &localTm);
|
||||
|
||||
// Append milliseconds
|
||||
auto ms = duration_cast<milliseconds>(tp.time_since_epoch()) % 1000;
|
||||
char iso8601[34];
|
||||
std::snprintf(iso8601, sizeof(iso8601), "%s.%03lld", buffer, static_cast<long long>(ms.count()));
|
||||
|
||||
return std::string(iso8601);
|
||||
}
|
||||
|
||||
void log(const std::string &msg) {
|
||||
uint64_t now = std::chrono::duration_cast<std::chrono::milliseconds>(std::chrono::system_clock::now().time_since_epoch()).count();
|
||||
std::cout << "\033[0m[" << unixMillisToISO8601(now) << " LOG] " << msg << std::endl;
|
||||
std::cout << "\033[0m[" << std::to_string(now) << " LOG] " << msg << std::endl;
|
||||
}
|
||||
|
||||
void warn(const std::string &msg) {
|
||||
uint64_t now = std::chrono::duration_cast<std::chrono::milliseconds>(std::chrono::system_clock::now().time_since_epoch()).count();
|
||||
std::cerr << "\033[33m[" << unixMillisToISO8601(now) << " WARN] " << msg << "\033[0m" << std::endl;
|
||||
std::cerr << "\033[33m[" << std::to_string(now) << " WARN] " << msg << "\033[0m" << std::endl;
|
||||
}
|
||||
|
||||
void error(const std::string &msg) {
|
||||
uint64_t now = std::chrono::duration_cast<std::chrono::milliseconds>(std::chrono::system_clock::now().time_since_epoch()).count();
|
||||
std::cerr << "\033[31m[" << unixMillisToISO8601(now) << " ERROR] " << msg << "\033[0m" << std::endl;
|
||||
std::cerr << "\033[31m[" << std::to_string(now) << " ERROR] " << msg << "\033[0m" << std::endl;
|
||||
}
|
||||
|
||||
void debug(const std::string &msg) {
|
||||
#if DEBUG || _DEBUG
|
||||
uint64_t now = std::chrono::duration_cast<std::chrono::milliseconds>(std::chrono::system_clock::now().time_since_epoch()).count();
|
||||
std::cerr << "\033[95m[" << unixMillisToISO8601(now) << " DEBUG] " << msg << "\033[0m" << std::endl;
|
||||
std::cerr << "\033[95m[" << std::to_string(now) << " DEBUG] " << msg << "\033[0m" << std::endl;
|
||||
#else
|
||||
return;
|
||||
#endif
|
||||
@@ -85,7 +49,7 @@ namespace ColumnLynx::Utils {
|
||||
}
|
||||
|
||||
std::string getVersion() {
|
||||
return "1.1.0";
|
||||
return "b0.1";
|
||||
}
|
||||
|
||||
unsigned short serverPort() {
|
||||
@@ -93,7 +57,7 @@ namespace ColumnLynx::Utils {
|
||||
}
|
||||
|
||||
unsigned char protocolVersion() {
|
||||
return 2;
|
||||
return 1;
|
||||
}
|
||||
|
||||
std::string bytesToHexString(const uint8_t* bytes, size_t length) {
|
||||
@@ -137,25 +101,17 @@ namespace ColumnLynx::Utils {
|
||||
return bytes;
|
||||
}
|
||||
|
||||
std::vector<std::string> getWhitelistedKeys(std::string basePath) {
|
||||
std::vector<std::string> getWhitelistedKeys() {
|
||||
// Currently re-reads the file every time, should be fine.
|
||||
// Advantage of it is that you don't need to reload the server binary after adding/removing keys. Disadvantage is re-reading the file every time.
|
||||
// I might redo this part.
|
||||
|
||||
std::vector<std::string> out;
|
||||
|
||||
std::ifstream file(basePath + "whitelisted_keys");
|
||||
if (!file.is_open()) {
|
||||
warn("Failed to open whitelisted_keys file at path: " + basePath + "whitelisted_keys");
|
||||
return out;
|
||||
}
|
||||
|
||||
std::ifstream file("whitelisted_keys"); // TODO: This is hardcoded for now, make dynamic
|
||||
std::string line;
|
||||
|
||||
while (std::getline(file, line)) {
|
||||
// Convert to upper case to align with the bytesToHexString() output
|
||||
for (int i = 0; i < line.length(); i++) {
|
||||
line[i] = toupper(line[i]);
|
||||
}
|
||||
out.push_back(line);
|
||||
}
|
||||
|
||||
@@ -167,10 +123,6 @@ namespace ColumnLynx::Utils {
|
||||
std::vector<std::string> readLines;
|
||||
|
||||
std::ifstream file(path);
|
||||
if (!file.is_open()) {
|
||||
throw std::runtime_error("Failed to open config file at path: " + path);
|
||||
}
|
||||
|
||||
std::string line;
|
||||
|
||||
while (std::getline(file, line)) {
|
||||
|
||||
@@ -1,60 +1,11 @@
|
||||
// virtual_interface.cpp - Virtual Interface for Network Communication
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#include <columnlynx/common/net/virtual_interface.hpp>
|
||||
|
||||
// This is all fucking voodoo dark magic.
|
||||
|
||||
#if defined(_WIN32)
|
||||
|
||||
static HMODULE gWintun = nullptr;
|
||||
|
||||
static WINTUN_OPEN_ADAPTER_FUNC* pWintunOpenAdapter;
|
||||
static WINTUN_START_SESSION_FUNC* pWintunStartSession;
|
||||
static WINTUN_END_SESSION_FUNC* pWintunEndSession;
|
||||
static WINTUN_GET_READ_WAIT_EVENT_FUNC* pWintunGetReadWaitEvent;
|
||||
static WINTUN_RECEIVE_PACKET_FUNC* pWintunReceivePacket;
|
||||
static WINTUN_RELEASE_RECEIVE_PACKET_FUNC* pWintunReleaseReceivePacket;
|
||||
static WINTUN_ALLOCATE_SEND_PACKET_FUNC* pWintunAllocateSendPacket;
|
||||
static WINTUN_SEND_PACKET_FUNC* pWintunSendPacket;
|
||||
static WINTUN_CREATE_ADAPTER_FUNC* pWintunCreateAdapter;
|
||||
|
||||
static void InitializeWintun()
|
||||
{
|
||||
if (gWintun)
|
||||
return;
|
||||
|
||||
gWintun = LoadLibraryExW(
|
||||
L"wintun.dll",
|
||||
nullptr,
|
||||
LOAD_LIBRARY_SEARCH_APPLICATION_DIR
|
||||
);
|
||||
|
||||
if (!gWintun)
|
||||
throw std::runtime_error("Failed to load wintun.dll");
|
||||
|
||||
#define RESOLVE(name, type) \
|
||||
p##name = reinterpret_cast<type*>( \
|
||||
GetProcAddress(gWintun, #name)); \
|
||||
if (!p##name) \
|
||||
throw std::runtime_error("Missing Wintun symbol: " #name);
|
||||
|
||||
RESOLVE(WintunOpenAdapter, WINTUN_OPEN_ADAPTER_FUNC)
|
||||
RESOLVE(WintunStartSession, WINTUN_START_SESSION_FUNC)
|
||||
RESOLVE(WintunEndSession, WINTUN_END_SESSION_FUNC)
|
||||
RESOLVE(WintunGetReadWaitEvent, WINTUN_GET_READ_WAIT_EVENT_FUNC)
|
||||
RESOLVE(WintunReceivePacket, WINTUN_RECEIVE_PACKET_FUNC)
|
||||
RESOLVE(WintunReleaseReceivePacket, WINTUN_RELEASE_RECEIVE_PACKET_FUNC)
|
||||
RESOLVE(WintunAllocateSendPacket, WINTUN_ALLOCATE_SEND_PACKET_FUNC)
|
||||
RESOLVE(WintunSendPacket, WINTUN_SEND_PACKET_FUNC)
|
||||
RESOLVE(WintunCreateAdapter, WINTUN_CREATE_ADAPTER_FUNC)
|
||||
|
||||
#undef RESOLVE
|
||||
}
|
||||
|
||||
#endif // _WIN32
|
||||
|
||||
namespace ColumnLynx::Net {
|
||||
// ------------------------------ Constructor ------------------------------
|
||||
VirtualInterface::VirtualInterface(const std::string& ifName)
|
||||
@@ -72,7 +23,7 @@ namespace ColumnLynx::Net {
|
||||
|
||||
if (ioctl(mFd, TUNSETIFF, &ifr) < 0) {
|
||||
close(mFd);
|
||||
throw std::runtime_error("TUNSETIFF failed (try running with sudo): " + std::string(strerror(errno)));
|
||||
throw std::runtime_error("TUNSETIFF failed: " + std::string(strerror(errno)));
|
||||
}
|
||||
|
||||
#elif defined(__APPLE__)
|
||||
@@ -96,7 +47,7 @@ namespace ColumnLynx::Net {
|
||||
|
||||
if (connect(mFd, (struct sockaddr*)&sc, sizeof(sc)) < 0) {
|
||||
if (errno == EPERM)
|
||||
throw std::runtime_error("connect(AF_SYS_CONTROL) failed: Insufficient permissions (try running with sudo)");
|
||||
throw std::runtime_error("connect(AF_SYS_CONTROL) failed: Insufficient permissions (try running as root)");
|
||||
throw std::runtime_error("connect(AF_SYS_CONTROL) failed: " + std::string(strerror(errno)));
|
||||
}
|
||||
|
||||
@@ -112,33 +63,20 @@ namespace ColumnLynx::Net {
|
||||
Utils::log("VirtualInterface: opened macOS UTUN: " + mIfName);
|
||||
|
||||
#elif defined(_WIN32)
|
||||
// ---- Windows: Wintun (WireGuard virtual adapter) ----
|
||||
WINTUN_ADAPTER_HANDLE adapter =
|
||||
WintunOpenAdapter(L"ColumnLynx", std::wstring(ifName.begin(), ifName.end()).c_str());
|
||||
if (!adapter)
|
||||
throw std::runtime_error("Wintun adapter not found or not installed");
|
||||
|
||||
// Convert to Windows' wchar_t* thingy
|
||||
std::wstring_convert<std::codecvt_utf8<wchar_t>> converter;
|
||||
std::wstring wide_string = converter.from_bytes(mIfName);
|
||||
const wchar_t* wide_c_str = wide_string.c_str();
|
||||
|
||||
InitializeWintun();
|
||||
|
||||
mAdapter = pWintunOpenAdapter(wide_c_str);
|
||||
|
||||
if (!mAdapter) {
|
||||
mAdapter = pWintunCreateAdapter(
|
||||
wide_c_str,
|
||||
L"ColumnLynx",
|
||||
nullptr
|
||||
);
|
||||
}
|
||||
|
||||
if (!mAdapter)
|
||||
throw std::runtime_error("Failed to open or create Wintun adapter (run running as admin)");
|
||||
|
||||
mSession = pWintunStartSession(mAdapter, 0x200000);
|
||||
if (!mSession)
|
||||
WINTUN_SESSION_HANDLE session =
|
||||
WintunStartSession(adapter, 0x200000); // ring buffer size
|
||||
if (!session)
|
||||
throw std::runtime_error("Failed to start Wintun session");
|
||||
|
||||
mHandle = pWintunGetReadWaitEvent(mSession);
|
||||
mFd = -1;
|
||||
mHandle = WintunGetReadWaitEvent(session);
|
||||
mFd = -1; // not used on Windows
|
||||
mIfName = ifName;
|
||||
|
||||
#else
|
||||
throw std::runtime_error("Unsupported platform");
|
||||
@@ -151,8 +89,9 @@ namespace ColumnLynx::Net {
|
||||
if (mFd >= 0)
|
||||
close(mFd);
|
||||
#elif defined(_WIN32)
|
||||
if (mSession)
|
||||
pWintunEndSession(mSession);
|
||||
// Wintun sessions need explicit stop
|
||||
// (assuming you stored the session handle as member)
|
||||
// WintunEndSession(mSession);
|
||||
#endif
|
||||
}
|
||||
|
||||
@@ -218,13 +157,11 @@ namespace ColumnLynx::Net {
|
||||
|
||||
#elif defined(_WIN32)
|
||||
|
||||
DWORD size = 0;
|
||||
BYTE* packet = pWintunReceivePacket(mSession, &size);
|
||||
if (!packet)
|
||||
return {};
|
||||
WINTUN_PACKET* packet = WintunReceivePacket(mSession, nullptr);
|
||||
if (!packet) return {};
|
||||
|
||||
std::vector<uint8_t> buf(packet, packet + size);
|
||||
pWintunReleaseReceivePacket(mSession, packet);
|
||||
std::vector<uint8_t> buf(packet->Data, packet->Data + packet->Length);
|
||||
WintunReleaseReceivePacket(mSession, packet);
|
||||
return buf;
|
||||
|
||||
#else
|
||||
@@ -269,16 +206,12 @@ namespace ColumnLynx::Net {
|
||||
|
||||
#elif defined(_WIN32)
|
||||
|
||||
BYTE* tx = pWintunAllocateSendPacket(
|
||||
mSession,
|
||||
static_cast<DWORD>(packet.size())
|
||||
);
|
||||
|
||||
WINTUN_PACKET* tx = WintunAllocateSendPacket(mSession, (DWORD)packet.size());
|
||||
if (!tx)
|
||||
throw std::runtime_error("WintunAllocateSendPacket failed");
|
||||
|
||||
memcpy(tx, packet.data(), packet.size());
|
||||
pWintunSendPacket(mSession, tx);
|
||||
memcpy(tx->Data, packet.data(), packet.size());
|
||||
WintunSendPacket(mSession, tx);
|
||||
|
||||
#endif
|
||||
}
|
||||
@@ -304,53 +237,7 @@ namespace ColumnLynx::Net {
|
||||
return false;
|
||||
#endif
|
||||
}
|
||||
|
||||
void VirtualInterface::resetIP() {
|
||||
#if defined(__linux__)
|
||||
char cmd[512];
|
||||
snprintf(cmd, sizeof(cmd),
|
||||
"ip addr flush dev %s",
|
||||
mIfName.c_str()
|
||||
);
|
||||
system(cmd);
|
||||
#elif defined(__APPLE__)
|
||||
char cmd[512];
|
||||
snprintf(cmd, sizeof(cmd),
|
||||
"ifconfig %s inet 0.0.0.0 delete",
|
||||
mIfName.c_str()
|
||||
);
|
||||
system(cmd);
|
||||
|
||||
snprintf(cmd, sizeof(cmd),
|
||||
"ifconfig %s inet6 :: delete",
|
||||
mIfName.c_str()
|
||||
);
|
||||
system(cmd);
|
||||
|
||||
// Wipe old routes
|
||||
//snprintf(cmd, sizeof(cmd),
|
||||
// "route -n delete -net %s",
|
||||
// mIfName.c_str()
|
||||
//);
|
||||
//system(cmd);
|
||||
#elif defined(_WIN32)
|
||||
char cmd[512];
|
||||
// Remove any persistent routes associated with this interface
|
||||
snprintf(cmd, sizeof(cmd),
|
||||
"netsh routing ip delete persistentroute all name=\"%s\"",
|
||||
mIfName.c_str()
|
||||
);
|
||||
system(cmd);
|
||||
|
||||
// Reset to DHCP
|
||||
snprintf(cmd, sizeof(cmd),
|
||||
"netsh interface ip set address name=\"%s\" dhcp",
|
||||
mIfName.c_str()
|
||||
);
|
||||
system(cmd);
|
||||
#endif
|
||||
}
|
||||
|
||||
|
||||
// ------------------------------------------------------------
|
||||
// Linux
|
||||
// ------------------------------------------------------------
|
||||
@@ -413,12 +300,6 @@ namespace ColumnLynx::Net {
|
||||
mIfName.c_str(), ipStr.c_str(), peerStr.c_str(), mtu, prefixStr.c_str());
|
||||
system(cmd);
|
||||
|
||||
// Host bits are auto-normalized by the kernel on macOS, so we don't need to worry about them not being zeroed out.
|
||||
snprintf(cmd, sizeof(cmd),
|
||||
"route -n add -net %s/%d -interface %s",
|
||||
ipStr.c_str(), prefixLen, mIfName.c_str());
|
||||
system(cmd);
|
||||
|
||||
Utils::log("Executed command: " + std::string(cmd));
|
||||
|
||||
return true;
|
||||
@@ -433,66 +314,29 @@ namespace ColumnLynx::Net {
|
||||
uint16_t mtu)
|
||||
{
|
||||
#ifdef _WIN32
|
||||
// Interface alias → LUID → Index
|
||||
std::wstring ifAlias(mIfName.begin(), mIfName.end());
|
||||
std::string ip = ipv4ToString(clientIP);
|
||||
std::string gw = ipv4ToString(serverIP);
|
||||
std::string mask;
|
||||
|
||||
NET_LUID luid;
|
||||
if (ConvertInterfaceAliasToLuid(ifAlias.c_str(), &luid) != NO_ERROR)
|
||||
return false;
|
||||
// Convert prefixLen → subnet mask
|
||||
uint32_t maskInt = (prefixLen == 0) ? 0 : (0xFFFFFFFF << (32 - prefixLen));
|
||||
mask = ipv4ToString(maskInt);
|
||||
|
||||
NET_IFINDEX ifIndex;
|
||||
if (ConvertInterfaceLuidToIndex(&luid, &ifIndex) != NO_ERROR)
|
||||
return false;
|
||||
char cmd[256];
|
||||
|
||||
// ssign IPv4 address + prefix
|
||||
MIB_UNICASTIPADDRESS_ROW addr;
|
||||
InitializeUnicastIpAddressEntry(&addr);
|
||||
// 1. Set the static IP + mask + gateway
|
||||
snprintf(cmd, sizeof(cmd),
|
||||
"netsh interface ip set address name=\"%s\" static %s %s %s",
|
||||
mIfName.c_str(), ip.c_str(), mask.c_str(), gw.c_str()
|
||||
);
|
||||
system(cmd);
|
||||
|
||||
addr.InterfaceIndex = ifIndex;
|
||||
addr.Address.si_family = AF_INET;
|
||||
addr.Address.Ipv4.sin_addr.s_addr = htonl(clientIP);
|
||||
addr.OnLinkPrefixLength = prefixLen;
|
||||
addr.DadState = IpDadStatePreferred;
|
||||
|
||||
if (CreateUnicastIpAddressEntry(&addr) != NO_ERROR)
|
||||
return false;
|
||||
|
||||
// Set MTU
|
||||
MIB_IFROW ifRow;
|
||||
ifRow.dwIndex = ifIndex;
|
||||
|
||||
if (GetIfEntry(&ifRow) != NO_ERROR)
|
||||
return false;
|
||||
|
||||
ifRow.dwMtu = mtu;
|
||||
|
||||
if (SetIfEntry(&ifRow) != NO_ERROR)
|
||||
return false;
|
||||
|
||||
// Add persistent route for VPN network via this interface
|
||||
uint32_t mask =
|
||||
(prefixLen == 0) ? 0 : (0xFFFFFFFFu << (32 - prefixLen));
|
||||
uint32_t network = clientIP & mask;
|
||||
|
||||
MIB_IPFORWARD_ROW2 route;
|
||||
InitializeIpForwardEntry(&route);
|
||||
|
||||
route.InterfaceIndex = ifIndex;
|
||||
route.DestinationPrefix.Prefix.si_family = AF_INET;
|
||||
route.DestinationPrefix.Prefix.Ipv4.sin_addr.s_addr = htonl(network);
|
||||
route.DestinationPrefix.PrefixLength = prefixLen;
|
||||
|
||||
route.NextHop.si_family = AF_INET;
|
||||
route.NextHop.Ipv4.sin_addr.s_addr = 0;
|
||||
|
||||
route.Metric = 1;
|
||||
route.Protocol = static_cast<NL_ROUTE_PROTOCOL>(MIB_IPPROTO_NETMGMT);
|
||||
route.ValidLifetime = 0xFFFFFFFF;
|
||||
route.PreferredLifetime = 0xFFFFFFFF;
|
||||
|
||||
DWORD r = CreateIpForwardEntry2(&route);
|
||||
if (r != NO_ERROR && r != ERROR_OBJECT_ALREADY_EXISTS)
|
||||
return false;
|
||||
// 2. Set MTU (separate command)
|
||||
snprintf(cmd, sizeof(cmd),
|
||||
"netsh interface ipv4 set subinterface \"%s\" mtu=%u store=persistent",
|
||||
mIfName.c_str(), mtu
|
||||
);
|
||||
system(cmd);
|
||||
|
||||
return true;
|
||||
#else
|
||||
|
||||
@@ -1,11 +1,9 @@
|
||||
// main.cpp - Server entry point for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#include <asio.hpp>
|
||||
#include <iostream>
|
||||
#include <thread>
|
||||
#include <chrono>
|
||||
#include <columnlynx/common/utils.hpp>
|
||||
#include <columnlynx/common/panic_handler.hpp>
|
||||
#include <columnlynx/server/net/tcp/tcp_server.hpp>
|
||||
@@ -15,11 +13,6 @@
|
||||
#include <unordered_map>
|
||||
#include <cxxopts.hpp>
|
||||
#include <columnlynx/common/net/virtual_interface.hpp>
|
||||
#include <columnlynx/server/server_session.hpp>
|
||||
|
||||
#if defined(__WIN32__)
|
||||
#include <windows.h>
|
||||
#endif
|
||||
|
||||
using asio::ip::tcp;
|
||||
using namespace ColumnLynx::Utils;
|
||||
@@ -30,7 +23,20 @@ using namespace ColumnLynx;
|
||||
|
||||
volatile sig_atomic_t done = 0;
|
||||
|
||||
void signalHandler(int signum) {
|
||||
if (signum == SIGINT || signum == SIGTERM) {
|
||||
log("Received termination signal. Shutting down server gracefully.");
|
||||
done = 1;
|
||||
}
|
||||
}
|
||||
|
||||
int main(int argc, char** argv) {
|
||||
// Capture SIGINT and SIGTERM for graceful shutdown
|
||||
struct sigaction action;
|
||||
memset(&action, 0, sizeof(struct sigaction));
|
||||
action.sa_handler = signalHandler;
|
||||
sigaction(SIGINT, &action, nullptr);
|
||||
sigaction(SIGTERM, &action, nullptr);
|
||||
|
||||
cxxopts::Options options("columnlynx_server", "ColumnLynx Server Application");
|
||||
|
||||
@@ -42,12 +48,7 @@ int main(int argc, char** argv) {
|
||||
#else
|
||||
("i,interface", "Override used interface", cxxopts::value<std::string>()->default_value("lynx0"))
|
||||
#endif
|
||||
#if defined(__WIN32__)
|
||||
/* Get config dir in LOCALAPPDATA\ColumnLynx\ */
|
||||
("config-dir", "Override config dir path", cxxopts::value<std::string>()->default_value("C:\\ProgramData\\ColumnLynx\\"));
|
||||
#else
|
||||
("config-dir", "Override config dir path", cxxopts::value<std::string>()->default_value("/etc/columnlynx"));
|
||||
#endif
|
||||
("config", "Override config file path", cxxopts::value<std::string>()->default_value("./server_config"));
|
||||
|
||||
PanicHandler::init();
|
||||
|
||||
@@ -56,7 +57,7 @@ int main(int argc, char** argv) {
|
||||
if (optionsObj.count("help")) {
|
||||
std::cout << options.help() << std::endl;
|
||||
std::cout << "This software is licensed under the GPLv2-only license OR the GPLv3 license.\n";
|
||||
std::cout << "Copyright (C) 2026, The ColumnLynx Contributors.\n";
|
||||
std::cout << "Copyright (C) 2025, The ColumnLynx Contributors.\n";
|
||||
std::cout << "This software is provided under ABSOLUTELY NO WARRANTY, to the extent permitted by law.\n";
|
||||
return 0;
|
||||
}
|
||||
@@ -67,43 +68,14 @@ int main(int argc, char** argv) {
|
||||
log("This software is licensed under the GPLv2 only OR the GPLv3. See LICENSES/ for details.");
|
||||
|
||||
#if defined(__WIN32__)
|
||||
//WintunInitialize();
|
||||
WintunInitialize();
|
||||
#endif
|
||||
|
||||
struct ServerState serverState{};
|
||||
|
||||
// Get the config path, ENV > CLI > /etc/columnlynx
|
||||
std::string configPath = optionsObj["config-dir"].as<std::string>();
|
||||
const char* envConfigPath = std::getenv("COLUMNLYNX_CONFIG_DIR");
|
||||
if (envConfigPath != nullptr) {
|
||||
configPath = std::string(envConfigPath);
|
||||
}
|
||||
|
||||
if (configPath.back() != '/' && configPath.back() != '\\') {
|
||||
#if defined(__WIN32__)
|
||||
configPath += "\\";
|
||||
#else
|
||||
configPath += "/";
|
||||
#endif
|
||||
}
|
||||
|
||||
serverState.configPath = configPath;
|
||||
|
||||
#if defined(DEBUG)
|
||||
std::unordered_map<std::string, std::string> config = Utils::getConfigMap(configPath + "server_config", { "NETWORK", "SUBNET_MASK" });
|
||||
#else
|
||||
// A production server should never use random keys. If the config file cannot be read or does not contain keys, the server will fail to start.
|
||||
std::unordered_map<std::string, std::string> config = Utils::getConfigMap(configPath + "server_config", { "NETWORK", "SUBNET_MASK", "SERVER_PUBLIC_KEY", "SERVER_PRIVATE_KEY" });
|
||||
#endif
|
||||
|
||||
serverState.serverConfig = config;
|
||||
std::unordered_map<std::string, std::string> config = Utils::getConfigMap(optionsObj["config"].as<std::string>());
|
||||
|
||||
std::shared_ptr<VirtualInterface> tun = std::make_shared<VirtualInterface>(optionsObj["interface"].as<std::string>());
|
||||
log("Using virtual interface: " + tun->getName());
|
||||
|
||||
// Store a reference to the tun in the serverState, it will increment and keep a safe reference (we love shared_ptrs)
|
||||
serverState.virtualInterface = tun;
|
||||
|
||||
// Generate a temporary keypair, replace with actual CA signed keys later (Note, these are stored in memory)
|
||||
std::shared_ptr<LibSodiumWrapper> sodiumWrapper = std::make_shared<LibSodiumWrapper>();
|
||||
|
||||
@@ -114,42 +86,31 @@ int main(int argc, char** argv) {
|
||||
log("Loading keypair from config file.");
|
||||
|
||||
PublicKey pk;
|
||||
PrivateSeed seed;
|
||||
PrivateKey sk;
|
||||
|
||||
std::copy_n(Utils::hexStringToBytes(itPrivkey->second).begin(), seed.size(), seed.begin());
|
||||
std::copy_n(Utils::hexStringToBytes(itPrivkey->second).begin(), sk.size(), sk.begin());
|
||||
std::copy_n(Utils::hexStringToBytes(itPubkey->second).begin(), pk.size(), pk.begin());
|
||||
|
||||
if (!sodiumWrapper->recomputeKeys(seed, pk)) {
|
||||
throw std::runtime_error("Failed to recompute keypair from config file values!");
|
||||
}
|
||||
sodiumWrapper->setKeys(pk, sk);
|
||||
} else {
|
||||
#if defined(DEBUG)
|
||||
warn("No keypair found in config file! Using random key.");
|
||||
#else
|
||||
throw std::runtime_error("No keypair found in config file! Cannot start server without keys.");
|
||||
#endif
|
||||
}
|
||||
|
||||
log("Server public key: " + bytesToHexString(sodiumWrapper->getPublicKey(), crypto_sign_PUBLICKEYBYTES));
|
||||
|
||||
serverState.sodiumWrapper = sodiumWrapper;
|
||||
serverState.ipv4Only = ipv4Only;
|
||||
serverState.hostRunning = true;
|
||||
|
||||
// Store the global state; from now on, it should only be accessed through the ServerSession singleton, which will ensure thread safety with its internal mutex
|
||||
ServerSession::getInstance().setServerState(std::make_shared<ServerState>(std::move(serverState)));
|
||||
std::shared_ptr<bool> hostRunning = std::make_shared<bool>(true);
|
||||
|
||||
asio::io_context io;
|
||||
|
||||
auto server = std::make_shared<TCPServer>(io, serverPort());
|
||||
auto udpServer = std::make_shared<UDPServer>(io, serverPort());
|
||||
auto server = std::make_shared<TCPServer>(io, serverPort(), sodiumWrapper, hostRunning, ipv4Only);
|
||||
auto udpServer = std::make_shared<UDPServer>(io, serverPort(), hostRunning, ipv4Only, tun);
|
||||
|
||||
asio::signal_set signals(io, SIGINT, SIGTERM);
|
||||
signals.async_wait([&](const std::error_code&, int) {
|
||||
log("Received termination signal. Shutting down server gracefully.");
|
||||
done = 1;
|
||||
asio::post(io, [&]() {
|
||||
ServerSession::getInstance().setHostRunning(false);
|
||||
*hostRunning = false;
|
||||
server->stop();
|
||||
udpServer->stop();
|
||||
});
|
||||
@@ -167,34 +128,19 @@ int main(int argc, char** argv) {
|
||||
while (!done) {
|
||||
auto packet = tun->readPacket();
|
||||
if (packet.empty()) {
|
||||
// Small sleep to avoid busy-waiting and to allow signal processing
|
||||
std::this_thread::sleep_for(std::chrono::milliseconds(10));
|
||||
continue;
|
||||
}
|
||||
|
||||
const uint8_t* ip = packet.data();
|
||||
uint32_t srcIP = ntohl(*(uint32_t*)(ip + 12)); // IPv4 source address offset
|
||||
uint32_t dstIP = ntohl(*(uint32_t*)(ip + 16)); // IPv4 destination address offset
|
||||
uint32_t dstIP = ntohl(*(uint32_t*)(ip + 16)); // IPv4 destination address offset in IPv6-mapped header
|
||||
|
||||
// First, check if destination IP is a registered client (e.g., server responding to client or client-to-client)
|
||||
auto dstSession = SessionRegistry::getInstance().getByIP(dstIP);
|
||||
if (dstSession) {
|
||||
// Destination is a registered client, forward to that client's session
|
||||
udpServer->sendData(dstSession->sessionID, std::string(packet.begin(), packet.end()));
|
||||
auto session = SessionRegistry::getInstance().getByIP(dstIP);
|
||||
if (!session) {
|
||||
Utils::warn("TUN: No session found for destination IP " + VirtualInterface::ipv4ToString(dstIP));
|
||||
continue;
|
||||
}
|
||||
|
||||
// Destination is not a registered client, check if source is (for external routing)
|
||||
auto srcSession = SessionRegistry::getInstance().getByIP(srcIP);
|
||||
if (srcSession) {
|
||||
// Source is a registered client, write to TUN interface to forward to external destination
|
||||
tun->writePacket(packet);
|
||||
continue;
|
||||
}
|
||||
|
||||
// Neither source nor destination is registered, drop the packet
|
||||
Utils::warn("TUN: No session found for source IP " + VirtualInterface::ipv4ToString(srcIP) +
|
||||
" or destination IP " + VirtualInterface::ipv4ToString(dstIP));
|
||||
udpServer->sendData(session->sessionID, std::string(packet.begin(), packet.end()));
|
||||
}
|
||||
|
||||
log("Shutting down server...");
|
||||
|
||||
@@ -1,44 +1,25 @@
|
||||
// tcp_connection.cpp - TCP Connection for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#include <columnlynx/server/net/tcp/tcp_connection.hpp>
|
||||
|
||||
namespace ColumnLynx::Net::TCP {
|
||||
void TCPConnection::start() {
|
||||
try {
|
||||
// Cache the remote IP early to avoid calling remote_endpoint() on closed sockets later
|
||||
mRemoteIP = mHandler->socket().remote_endpoint().address().to_string();
|
||||
} catch (const std::exception& e) {
|
||||
mRemoteIP = "unknown";
|
||||
Utils::warn("Failed to get remote endpoint: " + std::string(e.what()));
|
||||
}
|
||||
|
||||
mHandler->onMessage([this](AnyMessageType type, const std::string& data) {
|
||||
mHandleMessage(static_cast<ClientMessageType>(MessageHandler::toUint8(type)), data);
|
||||
});
|
||||
|
||||
mHandler->onDisconnect([this](const asio::error_code& ec) {
|
||||
// Peer has closed; finalize locally without sending RST
|
||||
Utils::log("Client disconnected: " + mRemoteIP + " - " + ec.message());
|
||||
asio::error_code ec2;
|
||||
mHandler->socket().close(ec2);
|
||||
|
||||
SessionRegistry::getInstance().erase(mConnectionSessionID);
|
||||
SessionRegistry::getInstance().deallocIP(mConnectionSessionID);
|
||||
|
||||
Utils::log("Closed connection to " + mRemoteIP);
|
||||
|
||||
if (mOnDisconnect) {
|
||||
mOnDisconnect(shared_from_this());
|
||||
}
|
||||
Utils::log("Client disconnected: " + mHandler->socket().remote_endpoint().address().to_string() + " - " + ec.message());
|
||||
disconnect();
|
||||
});
|
||||
|
||||
mHandler->start();
|
||||
mStartHeartbeat();
|
||||
|
||||
// Placeholder for message handling setup
|
||||
Utils::log("Client connected: " + mRemoteIP);
|
||||
Utils::log("Client connected: " + mHandler->socket().remote_endpoint().address().to_string());
|
||||
}
|
||||
|
||||
void TCPConnection::sendMessage(ServerMessageType type, const std::string& data) {
|
||||
@@ -51,22 +32,26 @@ namespace ColumnLynx::Net::TCP {
|
||||
mOnDisconnect = std::move(cb);
|
||||
}
|
||||
|
||||
void TCPConnection::disconnect(bool echo) {
|
||||
if (echo) {
|
||||
mHandler->sendMessage(ServerMessageType::GRACEFUL_DISCONNECT, "Server initiated disconnect.");
|
||||
}
|
||||
void TCPConnection::disconnect() {
|
||||
std::string ip = mHandler->socket().remote_endpoint().address().to_string();
|
||||
|
||||
mHandler->sendMessage(ServerMessageType::GRACEFUL_DISCONNECT, "Server initiated disconnect.");
|
||||
mHeartbeatTimer.cancel();
|
||||
asio::error_code ec;
|
||||
// Half-close: stop sending, keep reading until peer FIN
|
||||
mHandler->socket().shutdown(asio::ip::tcp::socket::shutdown_send, ec);
|
||||
if (ec) {
|
||||
Utils::error("Error during socket shutdown: " + ec.message());
|
||||
mHandler->socket().shutdown(asio::ip::tcp::socket::shutdown_both, ec);
|
||||
mHandler->socket().close(ec);
|
||||
|
||||
SessionRegistry::getInstance().erase(mConnectionSessionID);
|
||||
SessionRegistry::getInstance().deallocIP(mConnectionSessionID);
|
||||
|
||||
Utils::log("Closed connection to " + ip);
|
||||
|
||||
if (mOnDisconnect) {
|
||||
mOnDisconnect(shared_from_this());
|
||||
}
|
||||
// Do not close immediately; final cleanup happens in onDisconnect
|
||||
Utils::log("Initiated graceful disconnect (half-close) to " + mRemoteIP);
|
||||
}
|
||||
|
||||
uint32_t TCPConnection::getSessionID() const {
|
||||
uint64_t TCPConnection::getSessionID() const {
|
||||
return mConnectionSessionID;
|
||||
}
|
||||
|
||||
@@ -107,7 +92,7 @@ namespace ColumnLynx::Net::TCP {
|
||||
}
|
||||
|
||||
void TCPConnection::mHandleMessage(ClientMessageType type, const std::string& data) {
|
||||
std::string& reqAddr = mRemoteIP;
|
||||
std::string reqAddr = mHandler->socket().remote_endpoint().address().to_string();
|
||||
|
||||
switch (type) {
|
||||
case ClientMessageType::HANDSHAKE_INIT: {
|
||||
@@ -145,7 +130,7 @@ namespace ColumnLynx::Net::TCP {
|
||||
|
||||
Utils::debug("Key attempted connect: " + Utils::bytesToHexString(signPk.data(), signPk.size()));
|
||||
|
||||
std::vector<std::string> whitelistedKeys = Utils::getWhitelistedKeys(ServerSession::getInstance().getConfigPath());
|
||||
std::vector<std::string> whitelistedKeys = Utils::getWhitelistedKeys();
|
||||
|
||||
if (std::find(whitelistedKeys.begin(), whitelistedKeys.end(), Utils::bytesToHexString(signPk.data(), signPk.size())) == whitelistedKeys.end()) {
|
||||
Utils::warn("Non-whitelisted client attempted to connect, terminating. Client IP: " + reqAddr);
|
||||
@@ -156,7 +141,7 @@ namespace ColumnLynx::Net::TCP {
|
||||
|
||||
Utils::debug("Client " + reqAddr + " passed authorized_keys");
|
||||
|
||||
mHandler->sendMessage(ServerMessageType::HANDSHAKE_IDENTIFY, Utils::uint8ArrayToString(ServerSession::getInstance().getSodiumWrapper()->getPublicKey(), crypto_sign_PUBLICKEYBYTES)); // This public key should always exist
|
||||
mHandler->sendMessage(ServerMessageType::HANDSHAKE_IDENTIFY, Utils::uint8ArrayToString(mLibSodiumWrapper->getPublicKey(), crypto_sign_PUBLICKEYBYTES)); // This public key should always exist
|
||||
break;
|
||||
}
|
||||
case ClientMessageType::HANDSHAKE_CHALLENGE: {
|
||||
@@ -169,7 +154,7 @@ namespace ColumnLynx::Net::TCP {
|
||||
// Sign the challenge
|
||||
Signature sig = Utils::LibSodiumWrapper::signMessage(
|
||||
challengeData, sizeof(challengeData),
|
||||
ServerSession::getInstance().getSodiumWrapper()->getPrivateKey()
|
||||
mLibSodiumWrapper->getPrivateKey()
|
||||
);
|
||||
|
||||
mHandler->sendMessage(ServerMessageType::HANDSHAKE_CHALLENGE_RESPONSE, Utils::uint8ArrayToString(sig.data(), sig.size())); // Placeholder response
|
||||
@@ -191,8 +176,8 @@ namespace ColumnLynx::Net::TCP {
|
||||
std::memcpy(ciphertext.data(), data.data() + nonce.size(), ciphertext.size());
|
||||
try {
|
||||
std::array<uint8_t, 32> arrayPrivateKey;
|
||||
std::copy(ServerSession::getInstance().getSodiumWrapper()->getXPrivateKey(),
|
||||
ServerSession::getInstance().getSodiumWrapper()->getXPrivateKey() + 32,
|
||||
std::copy(mLibSodiumWrapper->getXPrivateKey(),
|
||||
mLibSodiumWrapper->getXPrivateKey() + 32,
|
||||
arrayPrivateKey.begin());
|
||||
|
||||
// Decrypt the AES key using the client's public key and server's private key
|
||||
@@ -211,18 +196,14 @@ namespace ColumnLynx::Net::TCP {
|
||||
|
||||
std::memcpy(mConnectionAESKey.data(), decrypted.data(), decrypted.size());
|
||||
|
||||
// Make a Session ID - unique and not zero (zero is reserved for invalid sessions)
|
||||
do {
|
||||
randombytes_buf(&mConnectionSessionID, sizeof(mConnectionSessionID));
|
||||
} while (SessionRegistry::getInstance().exists(mConnectionSessionID) || mConnectionSessionID == 0); // Regenerate if it already exists or is zero (zero is reserved for invalid sessions)
|
||||
// Make a Session ID
|
||||
randombytes_buf(&mConnectionSessionID, sizeof(mConnectionSessionID));
|
||||
|
||||
// Encrypt the Session ID with the established AES key (using symmetric encryption, nonce can be all zeros for this purpose)
|
||||
Nonce symNonce{}; // All zeros
|
||||
|
||||
const auto& serverConfig = ServerSession::getInstance().getRawServerConfig();
|
||||
|
||||
std::string networkString = serverConfig.find("NETWORK")->second; // The load check guarantees that this value exists
|
||||
uint8_t configMask = std::stoi(serverConfig.find("SUBNET_MASK")->second); // Same deal here
|
||||
std::string networkString = mRawServerConfig->find("NETWORK")->second; // The load check guarantees that this value exists
|
||||
uint8_t configMask = std::stoi(mRawServerConfig->find("SUBNET_MASK")->second); // Same deal here
|
||||
|
||||
uint32_t baseIP = Net::VirtualInterface::stringToIpv4(networkString);
|
||||
|
||||
@@ -249,11 +230,11 @@ namespace ColumnLynx::Net::TCP {
|
||||
tunConfig.dns1 = htonl(0x08080808); // 8.8.8.8
|
||||
tunConfig.dns2 = 0;
|
||||
|
||||
uint32_t sessionIDNet = htonl(mConnectionSessionID);
|
||||
uint64_t sessionIDNet = Utils::chtobe64(mConnectionSessionID);
|
||||
|
||||
std::vector<uint8_t> payload(sizeof(uint32_t) + sizeof(tunConfig));
|
||||
std::memcpy(payload.data(), &sessionIDNet, sizeof(uint32_t));
|
||||
std::memcpy(payload.data() + sizeof(uint32_t), &tunConfig, sizeof(tunConfig));
|
||||
std::vector<uint8_t> payload(sizeof(uint64_t) + sizeof(tunConfig));
|
||||
std::memcpy(payload.data(), &sessionIDNet, sizeof(uint64_t));
|
||||
std::memcpy(payload.data() + sizeof(uint64_t), &tunConfig, sizeof(tunConfig));
|
||||
|
||||
std::vector<uint8_t> encryptedPayload = Utils::LibSodiumWrapper::encryptMessage(
|
||||
payload.data(), payload.size(),
|
||||
@@ -265,7 +246,6 @@ namespace ColumnLynx::Net::TCP {
|
||||
// Add to session registry
|
||||
Utils::log("Handshake with " + reqAddr + " completed successfully. Session ID assigned (" + std::to_string(mConnectionSessionID) + ").");
|
||||
auto session = std::make_shared<SessionState>(mConnectionAESKey, std::chrono::hours(12), clientIP, htonl(0x0A0A0001), mConnectionSessionID);
|
||||
session->setBaseNonce(); // Set it
|
||||
SessionRegistry::getInstance().put(mConnectionSessionID, std::move(session));
|
||||
SessionRegistry::getInstance().lockIP(mConnectionSessionID, clientIP);
|
||||
|
||||
@@ -292,11 +272,6 @@ namespace ColumnLynx::Net::TCP {
|
||||
disconnect();
|
||||
break;
|
||||
}
|
||||
case ClientMessageType::KILL_CONNECTION: {
|
||||
Utils::warn("Received KILL_CONNECTION from " + reqAddr + ": " + data);
|
||||
disconnect();
|
||||
break;
|
||||
}
|
||||
default:
|
||||
Utils::warn("Unhandled message type from " + reqAddr);
|
||||
break;
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
// tcp_server.cpp - TCP Server for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#include <columnlynx/server/net/tcp/tcp_server.hpp>
|
||||
@@ -27,13 +27,15 @@ namespace ColumnLynx::Net::TCP {
|
||||
}
|
||||
Utils::error("Accept failed: " + ec.message());
|
||||
// Try again only if still running
|
||||
if (ServerSession::getInstance().isHostRunning() && mAcceptor.is_open())
|
||||
if (mHostRunning && *mHostRunning && mAcceptor.is_open())
|
||||
mStartAccept();
|
||||
return;
|
||||
}
|
||||
|
||||
auto client = TCPConnection::create(
|
||||
std::move(socket),
|
||||
mSodiumWrapper,
|
||||
&mRawServerConfig,
|
||||
[this](std::shared_ptr<TCPConnection> c) {
|
||||
mClients.erase(c);
|
||||
Utils::log("Client removed.");
|
||||
@@ -43,7 +45,7 @@ namespace ColumnLynx::Net::TCP {
|
||||
client->start();
|
||||
Utils::log("Accepted new client connection.");
|
||||
|
||||
if (ServerSession::getInstance().isHostRunning() && mAcceptor.is_open())
|
||||
if (mHostRunning && *mHostRunning && mAcceptor.is_open())
|
||||
mStartAccept();
|
||||
}
|
||||
);
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
// udp_server.cpp - UDP Server for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Copyright (C) 2025 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#include <columnlynx/server/net/udp/udp_server.hpp>
|
||||
@@ -16,27 +16,26 @@ namespace ColumnLynx::Net::UDP {
|
||||
if (ec) {
|
||||
if (ec == asio::error::operation_aborted) return; // Socket closed
|
||||
// Other recv error
|
||||
if (ServerSession::getInstance().isHostRunning()) mStartReceive();
|
||||
if (mHostRunning && *mHostRunning) mStartReceive();
|
||||
return;
|
||||
}
|
||||
if (bytes > 0) mHandlePacket(bytes);
|
||||
if (ServerSession::getInstance().isHostRunning()) mStartReceive();
|
||||
if (mHostRunning && *mHostRunning) mStartReceive();
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
void UDPServer::mHandlePacket(std::size_t bytes) {
|
||||
if (bytes < sizeof(UDPPacketHeader) + sizeof(uint32_t))
|
||||
if (bytes < sizeof(UDPPacketHeader))
|
||||
return;
|
||||
|
||||
const auto* hdr = reinterpret_cast<UDPPacketHeader*>(mRecvBuffer.data());
|
||||
|
||||
// Get plaintext session ID (first 4 bytes after header, in network byte order)
|
||||
uint32_t sessionIDNet = 0;
|
||||
std::memcpy(&sessionIDNet, mRecvBuffer.data() + sizeof(UDPPacketHeader), sizeof(uint32_t));
|
||||
uint32_t sessionID = sessionIDNet; // ntohl(sessionIDNet); --- IGNORE ---
|
||||
// Get plaintext session ID (assuming first 8 bytes after nonce (header))
|
||||
uint64_t sessionID = 0;
|
||||
std::memcpy(&sessionID, mRecvBuffer.data() + sizeof(UDPPacketHeader), sizeof(uint64_t));
|
||||
|
||||
auto it = mRecvBuffer.begin() + sizeof(UDPPacketHeader) + sizeof(uint32_t);
|
||||
auto it = mRecvBuffer.begin() + sizeof(UDPPacketHeader) + sizeof(uint64_t);
|
||||
std::vector<uint8_t> encryptedPayload(it, mRecvBuffer.begin() + bytes);
|
||||
|
||||
// Get associated session state
|
||||
@@ -55,7 +54,7 @@ namespace ColumnLynx::Net::UDP {
|
||||
encryptedPayload.data(), encryptedPayload.size(),
|
||||
session->aesKey,
|
||||
hdr->nonce, "udp-data"
|
||||
//std::string(reinterpret_cast<const char*>(&sessionID), sizeof(uint32_t))
|
||||
//std::string(reinterpret_cast<const char*>(&sessionID), sizeof(uint64_t))
|
||||
);
|
||||
|
||||
Utils::debug("Passed decryption");
|
||||
@@ -77,7 +76,7 @@ namespace ColumnLynx::Net::UDP {
|
||||
}
|
||||
}
|
||||
|
||||
void UDPServer::sendData(uint32_t sessionID, const std::string& data) {
|
||||
void UDPServer::sendData(const uint64_t sessionID, const std::string& data) {
|
||||
// Find the IPv4/IPv6 endpoint for the session
|
||||
std::shared_ptr<const SessionState> session = SessionRegistry::getInstance().get(sessionID);
|
||||
if (!session) {
|
||||
@@ -93,29 +92,23 @@ namespace ColumnLynx::Net::UDP {
|
||||
|
||||
// Prepare packet
|
||||
UDPPacketHeader hdr{};
|
||||
uint8_t nonce[12];
|
||||
uint32_t prefix = session->noncePrefix;
|
||||
uint64_t sendCount = const_cast<SessionState*>(session.get())->send_ctr.fetch_add(1, std::memory_order_relaxed);
|
||||
memcpy(nonce, &prefix, sizeof(uint32_t)); // Prefix nonce
|
||||
memcpy(nonce + sizeof(uint32_t), &sendCount, sizeof(uint64_t)); // Use send count as nonce suffix to ensure uniqueness
|
||||
std::copy_n(nonce, 12, hdr.nonce.data());
|
||||
randombytes_buf(hdr.nonce.data(), hdr.nonce.size());
|
||||
|
||||
auto encryptedPayload = Utils::LibSodiumWrapper::encryptMessage(
|
||||
reinterpret_cast<const uint8_t*>(data.data()), data.size(),
|
||||
session->aesKey, hdr.nonce, "udp-data"
|
||||
//std::string(reinterpret_cast<const char*>(&sessionID), sizeof(uint32_t))
|
||||
//std::string(reinterpret_cast<const char*>(&sessionID), sizeof(uint64_t))
|
||||
);
|
||||
|
||||
std::vector<uint8_t> packet;
|
||||
packet.reserve(sizeof(UDPPacketHeader) + sizeof(uint32_t) + encryptedPayload.size());
|
||||
packet.reserve(sizeof(UDPPacketHeader) + sizeof(uint64_t) + encryptedPayload.size());
|
||||
packet.insert(packet.end(),
|
||||
reinterpret_cast<uint8_t*>(&hdr),
|
||||
reinterpret_cast<uint8_t*>(&hdr) + sizeof(UDPPacketHeader)
|
||||
);
|
||||
uint32_t sessionIDNet = htonl(sessionID);
|
||||
packet.insert(packet.end(),
|
||||
reinterpret_cast<const uint8_t*>(&sessionIDNet),
|
||||
reinterpret_cast<const uint8_t*>(&sessionIDNet) + sizeof(sessionIDNet)
|
||||
reinterpret_cast<const uint8_t*>(&sessionID),
|
||||
reinterpret_cast<const uint8_t*>(&sessionID) + sizeof(sessionID)
|
||||
);
|
||||
packet.insert(packet.end(), encryptedPayload.begin(), encryptedPayload.end());
|
||||
|
||||
|
||||
@@ -1,92 +0,0 @@
|
||||
// server_session.cpp - Client Session data for ColumnLynx
|
||||
// Copyright (C) 2026 DcruBro
|
||||
// Distributed under the terms of the GNU General Public License, either version 2 only or version 3. See LICENSES/ for details.
|
||||
|
||||
#include <columnlynx/server/server_session.hpp>
|
||||
|
||||
namespace ColumnLynx {
|
||||
std::shared_ptr<ServerState> ServerSession::getServerState() const {
|
||||
std::shared_lock lock(mMutex);
|
||||
return mServerState;
|
||||
}
|
||||
|
||||
void ServerSession::setServerState(std::shared_ptr<ServerState> state) {
|
||||
std::unique_lock lock(mMutex);
|
||||
mServerState = std::move(state);
|
||||
}
|
||||
|
||||
std::shared_ptr<Utils::LibSodiumWrapper> ServerSession::getSodiumWrapper() const {
|
||||
std::shared_lock lock(mMutex);
|
||||
return mServerState ? mServerState->sodiumWrapper : nullptr;
|
||||
}
|
||||
|
||||
const std::string& ServerSession::getConfigPath() const {
|
||||
static const std::string emptyString;
|
||||
std::shared_ptr<ServerState> state = getServerState();
|
||||
return state ? state->configPath : emptyString;
|
||||
}
|
||||
|
||||
const std::unordered_map<std::string, std::string>& ServerSession::getRawServerConfig() const {
|
||||
static const std::unordered_map<std::string, std::string> emptyMap;
|
||||
std::shared_ptr<ServerState> state = getServerState();
|
||||
return state ? state->serverConfig : emptyMap;
|
||||
}
|
||||
|
||||
const std::shared_ptr<Net::VirtualInterface>& ServerSession::getVirtualInterface() const {
|
||||
static const std::shared_ptr<Net::VirtualInterface> nullTun = nullptr;
|
||||
std::shared_ptr<ServerState> state = getServerState();
|
||||
return state ? state->virtualInterface : nullTun;
|
||||
}
|
||||
|
||||
bool ServerSession::isIPv4Only() const {
|
||||
std::shared_ptr<ServerState> state = getServerState();
|
||||
return state ? state->ipv4Only : false;
|
||||
}
|
||||
|
||||
bool ServerSession::isHostRunning() const {
|
||||
std::shared_ptr<ServerState> state = getServerState();
|
||||
return state ? state->hostRunning : false;
|
||||
}
|
||||
|
||||
void ServerSession::setSodiumWrapper(std::shared_ptr<Utils::LibSodiumWrapper> sodiumWrapper) {
|
||||
std::unique_lock lock(mMutex);
|
||||
if (!mServerState)
|
||||
mServerState = std::make_shared<ServerState>();
|
||||
mServerState->sodiumWrapper = std::move(sodiumWrapper);
|
||||
}
|
||||
|
||||
void ServerSession::setConfigPath(const std::string& configPath) {
|
||||
std::unique_lock lock(mMutex);
|
||||
if (!mServerState)
|
||||
mServerState = std::make_shared<ServerState>();
|
||||
mServerState->configPath = configPath;
|
||||
}
|
||||
|
||||
void ServerSession::setRawServerConfig(const std::unordered_map<std::string, std::string>& config) {
|
||||
std::unique_lock lock(mMutex);
|
||||
if (!mServerState)
|
||||
mServerState = std::make_shared<ServerState>();
|
||||
mServerState->serverConfig = config;
|
||||
}
|
||||
|
||||
void ServerSession::setVirtualInterface(std::shared_ptr<Net::VirtualInterface> tun) {
|
||||
std::unique_lock lock(mMutex);
|
||||
if (!mServerState)
|
||||
mServerState = std::make_shared<ServerState>();
|
||||
mServerState->virtualInterface = std::move(tun);
|
||||
}
|
||||
|
||||
void ServerSession::setIPv4Only(bool ipv4Only) {
|
||||
std::unique_lock lock(mMutex);
|
||||
if (!mServerState)
|
||||
mServerState = std::make_shared<ServerState>();
|
||||
mServerState->ipv4Only = ipv4Only;
|
||||
}
|
||||
|
||||
void ServerSession::setHostRunning(bool hostRunning) {
|
||||
std::unique_lock lock(mMutex);
|
||||
if (!mServerState)
|
||||
mServerState = std::make_shared<ServerState>();
|
||||
mServerState->hostRunning = hostRunning;
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user