162 lines
6.6 KiB
C++
162 lines
6.6 KiB
C++
// main.cpp - Client entry point for ColumnLynx
|
|
// 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 <csignal>
|
|
#include <iostream>
|
|
#include <columnlynx/common/utils.hpp>
|
|
#include <columnlynx/common/panic_handler.hpp>
|
|
#include <columnlynx/client/net/tcp/tcp_client.hpp>
|
|
#include <columnlynx/client/net/udp/udp_client.hpp>
|
|
#include <cxxopts.hpp>
|
|
#include <columnlynx/common/net/virtual_interface.hpp>
|
|
|
|
#if defined(__WIN32__)
|
|
#include <windows.h>
|
|
#endif
|
|
|
|
using asio::ip::tcp;
|
|
using namespace ColumnLynx::Utils;
|
|
using namespace ColumnLynx::Net;
|
|
using namespace ColumnLynx;
|
|
|
|
volatile sig_atomic_t done = 0;
|
|
|
|
void signalHandler(int signum) {
|
|
if (signum == SIGINT || signum == SIGTERM) {
|
|
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();
|
|
|
|
cxxopts::Options options("columnlynx_client", "ColumnLynx Client Application");
|
|
|
|
options.add_options()
|
|
("h,help", "Print help")
|
|
("s,server", "Server address", cxxopts::value<std::string>()->default_value("127.0.0.1"))
|
|
("p,port", "Server port", cxxopts::value<uint16_t>()->default_value(std::to_string(serverPort())))
|
|
#if defined(__APPLE__)
|
|
("i,interface", "Override used interface", cxxopts::value<std::string>()->default_value("utun0"))
|
|
#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
|
|
|
|
bool insecureMode = options.parse(argc, argv).count("ignore-whitelist") > 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) 2025, The ColumnLynx Contributors.\n";
|
|
std::cout << "This software is provided under ABSOLUTELY NO WARRANTY, to the extent permitted by law.\n";
|
|
return 0;
|
|
}
|
|
|
|
auto host = optionsObj["server"].as<std::string>();
|
|
auto port = std::to_string(optionsObj["port"].as<uint16_t>());
|
|
|
|
try {
|
|
log("ColumnLynx Client, Version " + getVersion());
|
|
log("This software is licensed under the GPLv2 only OR the GPLv3. See LICENSES/ for details.");
|
|
|
|
#if defined(__WIN32__)
|
|
//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
|
|
}
|
|
|
|
std::shared_ptr<VirtualInterface> tun = std::make_shared<VirtualInterface>(optionsObj["interface"].as<std::string>());
|
|
log("Using virtual interface: " + tun->getName());
|
|
|
|
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));
|
|
|
|
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);
|
|
|
|
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!");
|
|
}
|
|
|
|
asio::io_context io;
|
|
auto client = std::make_shared<ColumnLynx::Net::TCP::TCPClient>(io, host, port, sodiumWrapper, aesKey, sessionID, insecureMode, configPath, tun);
|
|
auto udpClient = std::make_shared<ColumnLynx::Net::UDP::UDPClient>(io, host, port, aesKey, sessionID, tun);
|
|
|
|
client->start();
|
|
udpClient->start();
|
|
|
|
// Run the IO context in a separate thread
|
|
std::thread ioThread([&io]() {
|
|
io.run();
|
|
});
|
|
//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));
|
|
|
|
// 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) {
|
|
break; // Bail out if connection died or signal set while blocked
|
|
}*/
|
|
|
|
if (packet.empty()) {
|
|
continue;
|
|
}
|
|
|
|
udpClient->sendMessage(std::string(packet.begin(), packet.end()));
|
|
}
|
|
|
|
log("Client shutting down.");
|
|
udpClient->stop();
|
|
client->disconnect();
|
|
io.stop();
|
|
|
|
if (ioThread.joinable())
|
|
ioThread.join();
|
|
|
|
} catch (const std::exception& e) {
|
|
error("Client error: " + std::string(e.what()));
|
|
}
|
|
}
|