mirror of
https://github.com/amnezia-vpn/amnezia-client.git
synced 2026-05-08 14:33:23 +00:00
* refactor: move business logic from servers model * refactor: move containersModel initialization * refactor: added protocol ui controller and removed settings class from protocols model * refactor: moved cli management to separate controller * refactor: moved app split to separate controller * refactor: moved site split to separate controller * refactor: moved allowed dns to separate controller * refactor: moved language logic to separate ui controller * refactor: removed Settings from devices model * refactor: moved configs and services api logit to separate core controller * refactor: added a layer with a repository between the storage and controllers * refactor: use child parent system instead of smart pointers for controllers and models initialization * refactor: moved install functions from server controller to install controller * refactor: install controller refactoring * chore: renamed exportController to exportUiController * refactor: separate export controller * refactor: removed VpnConfigurationsController * chore: renamed ServerController to SshSession * refactor: replaced ServerController to SshSession * chore: moved qml controllers to separate folder * chore: include fixes * chore: moved utils from core root to core/utils * chore: include fixes * chore: rename core/utils files to camelCase foramt * chore: include fixes * chore: moved some utils to api and selfhosted folders * chore: include fixes * chore: remove unused file * chore: moved serialization folder to core/utils * chore: include fixes * chore: moved some files from client root to core/utils * chore: include fixes * chore: moved ui utils to ui/utils folder * chore: include fixes * chore: move utils from root to ui/utils * chore: include fixes * chore: moved configurators to core/configurators * chore: include fixes * refactor: moved iap logic from ui controller to core * refactor: moved remaining core logic from ApiConfigsController to SubscriptionController * chore: rename apiNewsController to apiNewsUiController * refactor: moved core logic from news ui controller to core * chore: renamed apiConfigsController to subscriptionUiController * chore: include fixes * refactor: merge ApiSettingsController with SubscriptionUiController * chore: moved ui selfhosted controllers to separate folder * chore: include fixes * chore: rename connectionController to connectiomUiController * refactor: moved core logic from connectionUiController * chore: rename settingsController to settingsUiController * refactor: move core logic from settingsUiController * refactor: moved core controller signal/slot connections to separate class * fix: newsController fixes after refactoring * chore: rename model to camelCase * chore: include fixes * chore: remove unused code * chore: move selfhosted core to separate folder * chore: include fixes * chore: rename importController to importUiController * refactor: move core logic from importUiController * chore: minor fixes * chore: remove prem v1 migration * refactor: remove openvpn over cloak and openvpn over shadowsocks * refactor: removed protocolsForContainer function * refactor: add core models * refactor: replace json with c++ structs for server config * refactor: move getDnsPair to ServerConfigUtils * feat: add admin selfhosted config export test * feat: add multi import test * refactor: use coreController for tests * feat: add few simple tests * chore: qrepos in all core controllers * feat: add test for settings * refactor: remove repo dependency from configurators * chore: moved protocols to core folder * chore: include fixes * refactor: moved containersDefs, defs, apiDefs, protocolsDefs to different places * chore: include fixes * chore: build fixes * chore: build fixes * refactor: remove q repo and interface repo * feat: add test for ui servers model and controller * chore: renamed to camelCase * chore: include fixes * refactor: moved core logic from sites ui controller * fix: fixed api config processing * fix: fixed processed server index processing * refactor: protocol models now use c++ structs instead of json configs * refactor: servers model now use c++ struct instead of json config * fix: fixed default server index processing * fix: fix logs init * fix: fix secure settings load keys * chore: build fixes * fix: fixed clear settings * fix: fixed restore backup * fix: sshSession usage * fix: fixed export functions signatures * fix: return missing part from buildContainerWorker * fix: fixed server description on page home * refactor: add container config helpers functions * refactor: c++ structs instead of json * chore: add dns protocol config struct * refactor: move config utils functions to config structs * feat: add test for selfhosted server setup * refactor: separate resources.qrc * fix: fixed server rename * chore: return nameOverriddenByUser * fix: build fixes * fix: fixed models init * refactor: cleanup models usage * fix: fixed models init * chore: cleanup connections and functions signatures * chore: cleanup updateModel calls * feat: added cache to servers repo * chore: cleanup unused functions * chore: ssxray processing * chore: remove transportProtoWithDefault and portWithDefault functions * chore: removed proto types any and l2tp * refactor: moved some constants * fix: fixed native configs export * refactor: remove json from processConfigWith functions * fix: fixed processed server index usage * fix: qml warning fixes * chore: merge fixes * chore: update tests * fix: fixed xray config processing * fix: fixed split tunneling processing * chore: rename sites controllers and model * chore: rename fixes * chore: minor fixes * chore: remove ability to load backup from "file with connection settings" button * fix: fixed api device revoke * fix: remove full model update when renaming a user * fix: fixed premium/free server rename * fix: fixed selfhosted new server install * fix: fixed updateContainer function * fix: fixed revoke for external premium configs * feat: add native configs qr processing * chore: codestyle fixes * fix: fixed admin config create * chore: again remove ability to load backup from "file with connection settings" button * chore: minor fixes * fix: fixed variables initialization * fix: fixed qml imports * fix: minor fixes * fix: fix vpnConnection function calls * feat: add buckup error handling * fix: fixed admin config revok * fix: fixed selfhosted awg installation * fix: ad visability * feat: add empty check for primary dns * chore: minor fixes
389 lines
13 KiB
C++
389 lines
13 KiB
C++
#include <QCoreApplication>
|
|
#include <QFileInfo>
|
|
#include <QProcess>
|
|
#include <QRandomGenerator>
|
|
#include <QTcpServer>
|
|
#include <QTcpSocket>
|
|
#include <QNetworkInterface>
|
|
|
|
#include "core/utils/networkUtilities.h"
|
|
#include "ipc.h"
|
|
#include "openVpnProtocol.h"
|
|
#include "core/utils/utilities.h"
|
|
#include "core/protocols/protocolUtils.h"
|
|
#include "version.h"
|
|
|
|
OpenVpnProtocol::OpenVpnProtocol(const QJsonObject &configuration, QObject *parent) : VpnProtocol(configuration, parent)
|
|
{
|
|
readOpenVpnConfiguration(configuration);
|
|
connect(&m_managementServer, &ManagementServer::readyRead, this,
|
|
&OpenVpnProtocol::onReadyReadDataFromManagementServer);
|
|
}
|
|
|
|
OpenVpnProtocol::~OpenVpnProtocol()
|
|
{
|
|
OpenVpnProtocol::stop();
|
|
QThread::msleep(200);
|
|
}
|
|
|
|
QString OpenVpnProtocol::defaultConfigFileName()
|
|
{
|
|
return defaultConfigPath() + QString("/%1.ovpn").arg(APPLICATION_NAME);
|
|
}
|
|
|
|
QString OpenVpnProtocol::defaultConfigPath()
|
|
{
|
|
QString p = QStandardPaths::writableLocation(QStandardPaths::AppDataLocation) + "/config";
|
|
Utils::initializePath(p);
|
|
|
|
return p;
|
|
}
|
|
|
|
void OpenVpnProtocol::stop()
|
|
{
|
|
qDebug() << "OpenVpnProtocol::stop()";
|
|
setConnectionState(Vpn::ConnectionState::Disconnecting);
|
|
|
|
// TODO: need refactoring
|
|
// sendTermSignal() will even return true while server connected ???
|
|
if ((m_connectionState == Vpn::ConnectionState::Preparing) || (m_connectionState == Vpn::ConnectionState::Connecting)
|
|
|| (m_connectionState == Vpn::ConnectionState::Connected)
|
|
|| (m_connectionState == Vpn::ConnectionState::Reconnecting)) {
|
|
if (!sendTermSignal()) {
|
|
killOpenVpnProcess();
|
|
}
|
|
QThread::msleep(10);
|
|
m_managementServer.stop();
|
|
}
|
|
|
|
#if defined(Q_OS_WIN) || defined(Q_OS_LINUX) || defined(Q_OS_MACOS)
|
|
IpcClient::withInterface([](QSharedPointer<IpcInterfaceReplica> iface) {
|
|
QRemoteObjectPendingReply<bool> reply = iface->disableKillSwitch();
|
|
if (!reply.waitForFinished(1000) && !reply.returnValue()) {
|
|
qWarning() << "OpenVpnProtocol::stop(): Failed to disable killswitch";
|
|
}
|
|
});
|
|
#endif
|
|
|
|
setConnectionState(Vpn::ConnectionState::Disconnected);
|
|
}
|
|
|
|
ErrorCode OpenVpnProtocol::prepare()
|
|
{
|
|
return IpcClient::withInterface([](QSharedPointer<IpcInterfaceReplica> iface) {
|
|
QRemoteObjectPendingReply<QStringList> listReply = iface->getTapList();
|
|
if (!listReply.waitForFinished(1000)) {
|
|
return ErrorCode::InternalError;
|
|
}
|
|
|
|
QStringList list = listReply.returnValue();
|
|
if (list.empty()) {
|
|
QRemoteObjectPendingReply<bool> installReply = iface->checkAndInstallDriver();
|
|
if (!installReply.waitForFinished() || !installReply.returnValue()) {
|
|
return ErrorCode::OpenVpnTapAdapterError;
|
|
}
|
|
}
|
|
|
|
return ErrorCode::NoError;
|
|
}, [] () {
|
|
return ErrorCode::AmneziaServiceConnectionFailed;
|
|
});
|
|
}
|
|
|
|
void OpenVpnProtocol::killOpenVpnProcess()
|
|
{
|
|
if (m_openVpnProcess) {
|
|
m_openVpnProcess->close();
|
|
}
|
|
}
|
|
|
|
void OpenVpnProtocol::readOpenVpnConfiguration(const QJsonObject &configuration)
|
|
{
|
|
if (configuration.contains(ProtocolUtils::key_proto_config_data(Proto::OpenVpn))) {
|
|
m_configData = configuration;
|
|
QJsonObject jConfig = configuration.value(ProtocolUtils::key_proto_config_data(Proto::OpenVpn)).toObject();
|
|
|
|
m_configFile.open();
|
|
m_configFile.write(jConfig.value(configKey::config).toString().toUtf8());
|
|
m_configFile.close();
|
|
m_configFileName = m_configFile.fileName();
|
|
qDebug().noquote() << QString("Set config data") << m_configFileName;
|
|
}
|
|
}
|
|
|
|
bool OpenVpnProtocol::openVpnProcessIsRunning() const
|
|
{
|
|
return Utils::processIsRunning("openvpn");
|
|
}
|
|
|
|
void OpenVpnProtocol::disconnectFromManagementServer()
|
|
{
|
|
m_managementServer.stop();
|
|
}
|
|
|
|
QString OpenVpnProtocol::configPath() const
|
|
{
|
|
return m_configFileName;
|
|
}
|
|
|
|
void OpenVpnProtocol::sendManagementCommand(const QString &command)
|
|
{
|
|
QIODevice *device = dynamic_cast<QIODevice *>(m_managementServer.socket().data());
|
|
if (device) {
|
|
QTextStream stream(device);
|
|
stream << command << Qt::endl;
|
|
}
|
|
}
|
|
|
|
uint OpenVpnProtocol::selectMgmtPort()
|
|
{
|
|
for (int i = 0; i < 100; ++i) {
|
|
quint32 port = QRandomGenerator::global()->generate();
|
|
port = (double)(65000 - 15001) * port / UINT32_MAX + 15001;
|
|
|
|
QTcpServer s;
|
|
bool ok = s.listen(QHostAddress::LocalHost, port);
|
|
if (ok)
|
|
return port;
|
|
}
|
|
return m_managementPort;
|
|
}
|
|
|
|
void OpenVpnProtocol::updateRouteGateway(QString line)
|
|
{
|
|
if (line.contains("net_route_v4_best_gw")) {
|
|
QStringList params = line.split(" ");
|
|
if (params.size() == 6) {
|
|
m_routeGateway = params.at(3);
|
|
}
|
|
} else {
|
|
line = line.split("ROUTE_GATEWAY", Qt::SkipEmptyParts).at(1);
|
|
if (!line.contains("/"))
|
|
return;
|
|
m_routeGateway = line.split("/", Qt::SkipEmptyParts).first();
|
|
m_routeGateway.replace(" ", "");
|
|
}
|
|
qDebug() << "Set VPN route gateway" << m_routeGateway;
|
|
}
|
|
|
|
ErrorCode OpenVpnProtocol::start()
|
|
{
|
|
OpenVpnProtocol::stop();
|
|
|
|
if (!QFileInfo::exists(Utils::openVpnExecPath())) {
|
|
setLastError(ErrorCode::OpenVpnExecutableMissing);
|
|
return lastError();
|
|
}
|
|
|
|
if (!QFileInfo::exists(configPath())) {
|
|
setLastError(ErrorCode::OpenVpnConfigMissing);
|
|
return lastError();
|
|
}
|
|
|
|
#ifdef AMNEZIA_DESKTOP
|
|
const ErrorCode res = IpcClient::withInterface([&](QSharedPointer<IpcInterfaceReplica> iface) {
|
|
QString ip = NetworkUtilities::getIPAddress(m_configData.value(amnezia::configKey::hostName).toString());
|
|
QRemoteObjectPendingReply<bool> reply = iface->addKillSwitchAllowedRange(QStringList(ip));
|
|
if (!reply.waitForFinished(1000) || !reply.returnValue()) {
|
|
return ErrorCode::AmneziaServiceConnectionFailed;
|
|
}
|
|
return ErrorCode::NoError;
|
|
});
|
|
if (res != ErrorCode::NoError) {
|
|
return res;
|
|
}
|
|
#endif
|
|
|
|
// Detect default gateway
|
|
#ifdef Q_OS_MAC
|
|
QProcess p;
|
|
p.setProcessChannelMode(QProcess::MergedChannels);
|
|
|
|
p.start("route",
|
|
QStringList() << "-n"
|
|
<< "get"
|
|
<< "default");
|
|
p.waitForFinished();
|
|
QString s = p.readAll();
|
|
|
|
QRegularExpression rx(R"(gateway:\s*(\d+\.\d+\.\d+\.\d+))");
|
|
QRegularExpressionMatch match = rx.match(s);
|
|
if (match.hasMatch()) {
|
|
m_routeGateway = match.captured(1);
|
|
qDebug() << "Set VPN route gateway" << m_routeGateway;
|
|
} else {
|
|
qWarning() << "Unable to set VPN route gateway, output:\n" << s;
|
|
}
|
|
#endif
|
|
|
|
uint mgmtPort = selectMgmtPort();
|
|
qDebug() << "OpenVpnProtocol::start mgmt port selected:" << mgmtPort;
|
|
|
|
if (!m_managementServer.start(m_managementHost, mgmtPort)) {
|
|
setLastError(ErrorCode::OpenVpnManagementServerError);
|
|
return lastError();
|
|
}
|
|
|
|
setConnectionState(Vpn::ConnectionState::Connecting);
|
|
|
|
m_openVpnProcess = IpcClient::CreatePrivilegedProcess();
|
|
|
|
if (!m_openVpnProcess) {
|
|
setLastError(ErrorCode::AmneziaServiceConnectionFailed);
|
|
return ErrorCode::AmneziaServiceConnectionFailed;
|
|
}
|
|
|
|
m_openVpnProcess->setProgram(PermittedProcess::OpenVPN);
|
|
QStringList arguments({
|
|
"--config", configPath(), "--management", m_managementHost, QString::number(mgmtPort),
|
|
"--management-client" /*, "--log", vpnLogFileNamePath */
|
|
});
|
|
m_openVpnProcess->setArguments(arguments);
|
|
|
|
qDebug() << arguments.join(" ");
|
|
connect(m_openVpnProcess.data(), &IpcProcessInterfaceReplica::errorOccurred,
|
|
[&](QProcess::ProcessError error) { qDebug() << "PrivilegedProcess errorOccurred" << error; });
|
|
|
|
connect(m_openVpnProcess.data(), &IpcProcessInterfaceReplica::stateChanged,
|
|
[&](QProcess::ProcessState newState) { qDebug() << "PrivilegedProcess stateChanged" << newState; });
|
|
|
|
connect(m_openVpnProcess.data(), &IpcProcessInterfaceReplica::finished, this,
|
|
[&]() { setConnectionState(Vpn::ConnectionState::Disconnected); });
|
|
|
|
m_openVpnProcess->start();
|
|
|
|
return ErrorCode::NoError;
|
|
}
|
|
|
|
bool OpenVpnProtocol::sendTermSignal()
|
|
{
|
|
return m_managementServer.writeCommand("signal SIGTERM");
|
|
}
|
|
|
|
void OpenVpnProtocol::sendByteCount()
|
|
{
|
|
m_managementServer.writeCommand("bytecount 1");
|
|
}
|
|
|
|
void OpenVpnProtocol::sendInitialData()
|
|
{
|
|
m_managementServer.writeCommand("state on");
|
|
m_managementServer.writeCommand("log on");
|
|
}
|
|
|
|
void OpenVpnProtocol::onReadyReadDataFromManagementServer()
|
|
{
|
|
for (;;) {
|
|
QString line = m_managementServer.readLine().simplified();
|
|
|
|
if (line.isEmpty()) {
|
|
return;
|
|
}
|
|
|
|
if (!line.contains(">BYTECOUNT")) {
|
|
qDebug().noquote() << line;
|
|
}
|
|
|
|
if (line.contains(">INFO:OpenVPN Management Interface")) {
|
|
sendInitialData();
|
|
} else if (line.startsWith(">STATE")) {
|
|
if (line.contains("CONNECTED,SUCCESS")) {
|
|
sendByteCount();
|
|
stopTimeoutTimer();
|
|
setConnectionState(Vpn::ConnectionState::Connected);
|
|
continue;
|
|
} else if (line.contains("EXITING,SIGTER")) {
|
|
// openVpnStateSigTermHandler();
|
|
setConnectionState(Vpn::ConnectionState::Disconnecting);
|
|
continue;
|
|
} else if (line.contains("RECONNECTING")) {
|
|
setConnectionState(Vpn::ConnectionState::Reconnecting);
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (line.contains("ROUTE_GATEWAY") || line.contains("net_route_v4_best_gw")) {
|
|
updateRouteGateway(line);
|
|
}
|
|
|
|
if (line.contains("PUSH: Received control message")) {
|
|
updateVpnGateway(line);
|
|
}
|
|
|
|
if (line.contains("FATAL")) {
|
|
if (line.contains("tap-windows6 adapters on this system are currently in use or disabled")) {
|
|
emit protocolError(ErrorCode::OpenVpnAdaptersInUseError);
|
|
} else {
|
|
emit protocolError(ErrorCode::OpenVpnUnknownError);
|
|
}
|
|
return;
|
|
}
|
|
|
|
QByteArray data(line.toStdString().c_str());
|
|
if (data.contains(">BYTECOUNT:")) {
|
|
int beg = data.lastIndexOf(">BYTECOUNT:");
|
|
int end = data.indexOf("\n", beg);
|
|
|
|
beg += sizeof(">BYTECOUNT:") - 1;
|
|
QList<QByteArray> count = data.mid(beg, end - beg + 1).split(',');
|
|
|
|
quint64 r = static_cast<quint64>(count.at(0).trimmed().toULongLong());
|
|
quint64 s = static_cast<quint64>(count.at(1).trimmed().toULongLong());
|
|
|
|
setBytesChanged(r, s);
|
|
}
|
|
}
|
|
}
|
|
|
|
void OpenVpnProtocol::updateVpnGateway(const QString &line)
|
|
{
|
|
// line looks like
|
|
// PUSH: Received control message: 'PUSH_REPLY,route 10.8.0.1,topology net30,ping 10,ping-restart
|
|
// 120,ifconfig 10.8.0.6 10.8.0.5,peer-id 0,cipher AES-256-GCM'
|
|
QStringList params = line.split(",");
|
|
for (const QString &l : params) {
|
|
if (l.contains("ifconfig")) {
|
|
if (l.split(" ").size() == 3) {
|
|
m_vpnLocalAddress = l.split(" ").at(1);
|
|
m_vpnGateway = l.split(" ").at(2);
|
|
#ifdef Q_OS_WIN
|
|
QThread::msleep(300);
|
|
IpcClient::withInterface([&](QSharedPointer<IpcInterfaceReplica> iface) {
|
|
QList<QNetworkInterface> netInterfaces = QNetworkInterface::allInterfaces();
|
|
for (int i = 0; i < netInterfaces.size(); i++) {
|
|
for (int j=0; j < netInterfaces.at(i).addressEntries().size(); j++)
|
|
{
|
|
// killSwitch toggle
|
|
if (m_vpnLocalAddress == netInterfaces.at(i).addressEntries().at(j).ip().toString()) {
|
|
if (QVariant(m_configData.value(configKey::killSwitchOption).toString()).toBool()) {
|
|
iface->enableKillSwitch(m_configData, netInterfaces.at(i).index());
|
|
}
|
|
m_configData.insert("vpnAdapterIndex", netInterfaces.at(i).index());
|
|
m_configData.insert("vpnGateway", m_vpnGateway);
|
|
m_configData.insert("vpnServer",
|
|
NetworkUtilities::getIPAddress(m_configData.value(amnezia::configKey::hostName).toString()));
|
|
iface->enablePeerTraffic(m_configData);
|
|
}
|
|
}
|
|
}
|
|
});
|
|
#endif
|
|
#if defined(Q_OS_LINUX) || defined(Q_OS_MACOS)
|
|
// killSwitch toggle
|
|
if (QVariant(m_configData.value(configKey::killSwitchOption).toString()).toBool()) {
|
|
m_configData.insert("vpnServer",
|
|
NetworkUtilities::getIPAddress(m_configData.value(amnezia::configKey::hostName).toString()));
|
|
IpcClient::withInterface([&](QSharedPointer<IpcInterfaceReplica> iface) {
|
|
QRemoteObjectPendingReply<bool> reply = iface->enableKillSwitch(m_configData, 0);
|
|
if (!reply.waitForFinished(1000) || !reply.returnValue()) {
|
|
qWarning() << "OpenVpnProtocol::updateVpnGateway(): Failed to enable killswitch";
|
|
}
|
|
});
|
|
}
|
|
#endif
|
|
qDebug() << QString("Set vpn local address %1, gw %2").arg(m_vpnLocalAddress).arg(vpnGateway());
|
|
}
|
|
}
|
|
}
|
|
}
|