Files
amnezia-client/client/vpnConnection.cpp
T

674 lines
24 KiB
C++
Raw Normal View History

#include "vpnConnection.h"
2023-08-30 15:10:44 +05:00
2020-12-26 15:03:51 +03:00
#include <QDebug>
#include <QEventLoop>
2021-01-06 17:12:24 +03:00
#include <QFile>
2022-01-22 20:00:06 +03:00
#include <QHostInfo>
2021-02-18 15:00:41 +03:00
#include <QJsonObject>
#include <QObject>
#include <QSharedPointer>
#include <QString>
#include <QStringList>
#include <QTimer>
2020-12-26 15:03:51 +03:00
#include <core/configurators/openVpnConfigurator.h>
#include <core/configurators/wireguardConfigurator.h>
2022-02-22 02:08:57 +03:00
#ifdef AMNEZIA_DESKTOP
#include "core/utils/ipcClient.h"
#include <core/protocols/wireGuardProtocol.h>
2022-02-22 02:08:57 +03:00
#endif
2021-01-06 17:12:24 +03:00
2021-09-30 18:16:41 +03:00
#ifdef Q_OS_ANDROID
2023-12-04 18:23:08 +03:00
#include "platforms/android/android_controller.h"
#include <QThread>
2021-09-30 18:16:41 +03:00
#endif
#if defined(Q_OS_IOS) || defined(MACOS_NE)
2023-08-28 22:03:28 +03:00
#include "platforms/ios/ios_controller.h"
#endif
#include "core/utils/constants/protocolConstants.h"
#include "core/utils/networkUtilities.h"
2020-12-26 15:03:51 +03:00
using namespace ProtocolUtils;
VpnConnection::VpnConnection(SecureServersRepository* serversRepository, SecureAppSettingsRepository* appSettingsRepository, QObject *parent)
: QObject(parent), m_serversRepository(serversRepository), m_appSettingsRepository(appSettingsRepository), m_checkTimer(this), m_trafficGuard(new VpnTrafficGuard(appSettingsRepository, this))
2021-12-20 15:43:36 +03:00
{
#if defined(Q_OS_IOS) || defined(MACOS_NE)
2025-11-30 18:49:16 -08:00
m_checkTimer.setInterval(1000);
2026-02-23 11:00:13 +01:00
connect(IosController::Instance(), &IosController::connectionStateChanged, this, &VpnConnection::setConnectionState);
2023-08-27 10:46:41 -07:00
connect(IosController::Instance(), &IosController::bytesChanged, this, &VpnConnection::onBytesChanged);
#endif
2021-02-18 15:00:41 +03:00
}
2021-02-03 15:42:36 +03:00
2021-02-18 15:00:41 +03:00
VpnConnection::~VpnConnection()
{
2020-12-26 15:03:51 +03:00
}
void VpnConnection::onBytesChanged(quint64 receivedBytes, quint64 sentBytes)
{
emit bytesChanged(receivedBytes, sentBytes);
2020-12-26 15:03:51 +03:00
}
2025-05-02 23:54:36 -07:00
void VpnConnection::onKillSwitchModeChanged(bool enabled)
{
#ifdef AMNEZIA_DESKTOP
2025-12-19 04:09:50 +01:00
IpcClient::withInterface([enabled](QSharedPointer<IpcInterfaceReplica> iface){
QRemoteObjectPendingReply<bool> reply = iface->refreshKillSwitch(enabled);
2026-02-19 13:21:49 +01:00
if (reply.waitForFinished() && reply.returnValue())
2025-12-19 04:09:50 +01:00
qDebug() << "VpnConnection::onKillSwitchModeChanged: Killswitch refreshed";
else
qWarning() << "VpnConnection::onKillSwitchModeChanged: Failed to execute remote refreshKillSwitch call";
});
2025-05-02 23:54:36 -07:00
#endif
}
2023-05-14 21:11:19 +08:00
void VpnConnection::onConnectionStateChanged(Vpn::ConnectionState state)
2020-12-26 15:03:51 +03:00
{
2022-02-22 02:08:57 +03:00
#ifdef AMNEZIA_DESKTOP
switch (state) {
case Vpn::ConnectionState::Connected: {
m_trafficGuard->setupRoutes(m_vpnConfiguration, vpnProtocol(), m_remoteAddress);
} break;
default:
break;
}
2022-02-22 02:08:57 +03:00
#endif
2022-12-12 16:16:12 +04:00
#if defined(Q_OS_IOS) || defined(MACOS_NE)
2025-12-30 04:45:32 +02:00
if (state == Vpn::ConnectionState::Connected ||
state == Vpn::ConnectionState::Connecting ||
state == Vpn::ConnectionState::Reconnecting) {
2023-08-08 19:02:41 -07:00
m_checkTimer.start();
2023-08-28 22:03:28 +03:00
} else {
2023-08-08 19:02:41 -07:00
m_checkTimer.stop();
2022-12-12 16:16:12 +04:00
}
#endif
2020-12-26 15:03:51 +03:00
}
void VpnConnection::setRepositories(SecureServersRepository* serversRepository, SecureAppSettingsRepository* appSettingsRepository)
{
m_serversRepository = serversRepository;
m_appSettingsRepository = appSettingsRepository;
m_trafficGuard.reset(new VpnTrafficGuard(appSettingsRepository, this));
}
2021-02-18 15:00:41 +03:00
QSharedPointer<VpnProtocol> VpnConnection::vpnProtocol() const
{
return m_active ? m_active->protocol() : m_vpnProtocol;
2021-02-18 15:00:41 +03:00
}
2025-11-30 18:49:16 -08:00
void VpnConnection::disconnectSlots()
{
if (auto proto = vpnProtocol()) {
proto->disconnect();
2025-11-30 18:49:16 -08:00
}
}
2021-01-06 17:12:24 +03:00
ErrorCode VpnConnection::lastError() const
2020-12-26 23:17:20 +03:00
{
2023-12-22 15:35:24 +03:00
#ifdef Q_OS_ANDROID
return ErrorCode::AndroidError;
#endif
auto proto = vpnProtocol();
return proto.isNull() ? ErrorCode::InternalError : proto->lastError();
2020-12-26 23:17:20 +03:00
}
Vpn::ConnectionState VpnConnection::connectionState() const
{
return m_connectionState;
}
QString VpnConnection::allocateIfname()
{
#ifdef Q_OS_MACOS
QString kernelAssigned;
IpcClient::withInterface([&kernelAssigned](QSharedPointer<IpcInterfaceReplica> iface) {
auto reply = iface->reserveUtunName();
if (reply.waitForFinished(2000)) {
kernelAssigned = reply.returnValue();
}
});
if (kernelAssigned.isEmpty() || m_ifnamesInUse.contains(kernelAssigned)) {
qCritical() << "allocateIfname: kernel utun reservation failed";
return QString();
}
m_ifnamesInUse.insert(kernelAssigned);
return kernelAssigned;
#else
for (int i = 0; ; ++i) {
const QString name = QStringLiteral("amn%1").arg(i);
if (!m_ifnamesInUse.contains(name)) {
m_ifnamesInUse.insert(name);
return name;
}
}
#endif
}
void VpnConnection::releaseIfname(const QString& ifname)
{
m_ifnamesInUse.remove(ifname);
}
void VpnConnection::wireTunnelSignals(Tunnel* tunnel, bool isActive)
{
connect(tunnel, &Tunnel::prepared, this, &VpnConnection::onTunnelPrepared);
connect(tunnel, &Tunnel::activated, this, &VpnConnection::onTunnelActivated);
connect(tunnel, &Tunnel::failed, this, &VpnConnection::onTunnelFailed);
if (isActive) {
connect(tunnel, &Tunnel::bytesChanged, this, &VpnConnection::onBytesChanged);
}
}
void VpnConnection::connectToVpn(const QString &serverId, DockerContainer container, const QJsonObject &vpnConfiguration)
2021-01-06 17:12:24 +03:00
{
if (!m_appSettingsRepository || !m_serversRepository) {
qCritical() << "VpnConnection::connectToVpn: repositories not initialized";
setConnectionState(Vpn::ConnectionState::Error);
return;
}
qDebug() << QString("Trying to connect to VPN, server id is %1, container is %2, route mode is")
.arg(serverId)
.arg(ContainerUtils::containerToString(container))
<< m_appSettingsRepository->routeMode();
2021-10-26 12:59:20 +03:00
const QString resolvedRemote =
NetworkUtilities::getIPAddress(vpnConfiguration.value(configKey::hostName).toString());
#ifdef AMNEZIA_DESKTOP
const bool isWg = VpnProtocol::isWireGuardBased(container);
const bool isXray = VpnProtocol::isXrayBased(container);
const bool useTunnelPath = isWg || isXray;
const QString preAllocatedIfname = useTunnelPath ? allocateIfname() : QString();
if (useTunnelPath && preAllocatedIfname.isEmpty()) {
setConnectionState(Vpn::ConnectionState::Error);
emit vpnProtocolError(ErrorCode::AmneziaServiceConnectionFailed);
return;
}
if (m_active
&& m_connectionState == Vpn::ConnectionState::Connected
&& useTunnelPath) {
if (!m_trafficGuard->allowEndpoint(resolvedRemote, preAllocatedIfname)) {
releaseIfname(preAllocatedIfname);
setConnectionState(Vpn::ConnectionState::Error);
emit vpnProtocolError(ErrorCode::AmneziaServiceConnectionFailed);
return;
}
startTunnelSwitch(container, vpnConfiguration, resolvedRemote, preAllocatedIfname);
return;
}
if (!m_trafficGuard->allowEndpoint(resolvedRemote, preAllocatedIfname)) {
if (useTunnelPath) releaseIfname(preAllocatedIfname);
setConnectionState(Vpn::ConnectionState::Error);
emit vpnProtocolError(ErrorCode::AmneziaServiceConnectionFailed);
return;
}
#endif
2026-02-19 13:21:49 +01:00
setConnectionState(Vpn::ConnectionState::Connecting);
2021-02-18 15:00:41 +03:00
QJsonObject config = vpnConfiguration;
2024-04-25 20:01:00 +07:00
2023-08-27 10:46:41 -07:00
#ifdef AMNEZIA_DESKTOP
if (m_active) {
const QString oldIfname = m_active->ifname();
m_trafficGuard->tearDown(m_active);
delete m_active;
m_active = nullptr;
releaseIfname(oldIfname);
}
2021-02-18 15:00:41 +03:00
if (m_vpnProtocol) {
disconnect(m_vpnProtocol.data(), &VpnProtocol::protocolError, this, &VpnConnection::vpnProtocolError);
m_trafficGuard->flushAll();
2021-02-18 15:00:41 +03:00
m_vpnProtocol->stop();
2021-02-25 18:05:42 +03:00
m_vpnProtocol.reset();
2021-02-18 15:00:41 +03:00
}
2023-08-27 10:46:41 -07:00
#endif
2021-02-18 15:00:41 +03:00
appendKillSwitchConfig(config);
appendSplitTunnelingConfig(config);
m_vpnConfiguration = config;
m_remoteAddress = resolvedRemote;
#ifdef AMNEZIA_DESKTOP
if (useTunnelPath) {
config.insert("ifname", preAllocatedIfname);
if (isXray) {
config.insert("tunName", preAllocatedIfname);
config.insert("deviceIpv4Address", amnezia::protocols::xray::defaultLocalAddr);
} else if (isWg) {
const QString protoName = config.value("protocol").toString();
const QJsonObject wgConfig = config.value(protoName + "_config_data").toObject();
const QString clientIp = wgConfig.value(amnezia::configKey::clientIp).toString();
if (!clientIp.isEmpty()) {
config.insert("deviceIpv4Address", clientIp);
}
}
m_vpnConfiguration = config;
m_active = new Tunnel(preAllocatedIfname, container, config, resolvedRemote, this);
wireTunnelSignals(m_active, /*isActive=*/true);
wireDaemonReconnectSignals();
m_trafficGuard->setConfig(config);
m_trafficGuard->bringUp(m_active);
return;
}
2023-10-13 15:45:06 +05:00
2021-10-04 21:13:07 +03:00
m_vpnProtocol.reset(VpnProtocol::factory(container, m_vpnConfiguration));
2021-10-04 19:07:49 +03:00
if (!m_vpnProtocol) {
2026-02-19 13:21:49 +01:00
setConnectionState(Vpn::ConnectionState::Error);
2021-10-26 12:59:20 +03:00
return;
2021-04-04 23:12:36 +03:00
}
2021-10-04 19:07:49 +03:00
m_vpnProtocol->prepare();
m_trafficGuard->setConfig(m_vpnConfiguration);
#elif defined Q_OS_ANDROID
2023-12-04 18:23:08 +03:00
androidVpnProtocol = createDefaultAndroidVpnProtocol();
createAndroidConnections();
2021-11-26 17:43:02 +03:00
2021-10-07 22:52:13 +03:00
m_vpnProtocol.reset(androidVpnProtocol);
#elif defined Q_OS_IOS || defined(MACOS_NE)
Proto proto = ContainerUtils::defaultProtocol(container);
2023-08-27 10:46:41 -07:00
IosController::Instance()->connectVpn(proto, m_vpnConfiguration);
connect(&m_checkTimer, &QTimer::timeout, IosController::Instance(), &IosController::checkStatus);
return;
2021-09-30 18:16:41 +03:00
#endif
2020-12-26 15:03:51 +03:00
createProtocolConnections();
2020-12-26 15:03:51 +03:00
2026-02-19 13:21:49 +01:00
if (ErrorCode err = m_vpnProtocol->start(); err != ErrorCode::NoError) {
setConnectionState(Vpn::ConnectionState::Error);
emit vpnProtocolError(err);
}
2020-12-26 15:03:51 +03:00
}
2023-08-28 22:03:28 +03:00
void VpnConnection::createProtocolConnections()
{
2023-03-15 17:46:22 +03:00
connect(m_vpnProtocol.data(), &VpnProtocol::protocolError, this, &VpnConnection::vpnProtocolError);
2026-02-19 13:21:49 +01:00
connect(m_vpnProtocol.data(), &VpnProtocol::connectionStateChanged, this, &VpnConnection::setConnectionState);
2023-03-15 17:46:22 +03:00
connect(m_vpnProtocol.data(), SIGNAL(bytesChanged(quint64, quint64)), this, SLOT(onBytesChanged(quint64, quint64)));
connect(m_vpnProtocol.data(), &VpnProtocol::tunnelAddressesUpdated, this,
[this](const QString& gateway, const QString& localAddress) {
m_trafficGuard->applyKillSwitch(nullptr, gateway, localAddress);
});
2026-02-19 13:21:49 +01:00
wireDaemonReconnectSignals();
}
void VpnConnection::wireDaemonReconnectSignals()
{
2026-02-19 13:21:49 +01:00
#ifdef AMNEZIA_DESKTOP
IpcClient::withInterface([this](QSharedPointer<IpcInterfaceReplica> rep) {
connect(rep.data(), &IpcInterfaceReplica::networkChanged, this, &VpnConnection::reconnectToVpn,
static_cast<Qt::ConnectionType>(Qt::QueuedConnection | Qt::UniqueConnection));
connect(rep.data(), &IpcInterfaceReplica::wakeup, this, &VpnConnection::reconnectToVpn,
static_cast<Qt::ConnectionType>(Qt::QueuedConnection | Qt::UniqueConnection));
2026-02-19 13:21:49 +01:00
});
#endif
2023-03-15 17:46:22 +03:00
}
void VpnConnection::appendKillSwitchConfig(QJsonObject &config)
2024-04-25 20:01:00 +07:00
{
#ifdef AMNEZIA_DESKTOP
if (!m_appSettingsRepository) {
qCritical() << "VpnConnection::appendKillSwitchConfig: repositories not initialized";
return;
}
config.insert(configKey::killSwitchOption, QVariant(m_appSettingsRepository->isKillSwitchEnabled()).toString());
config.insert(configKey::allowedDnsServers, QVariant(m_appSettingsRepository->getAllowedDnsServers()).toJsonValue());
#else
Q_UNUSED(config)
#endif
2024-04-25 20:01:00 +07:00
}
void VpnConnection::appendSplitTunnelingConfig(QJsonObject &config)
2023-10-13 15:45:06 +05:00
{
if (!m_appSettingsRepository) {
qCritical() << "VpnConnection::appendSplitTunnelingConfig: repositories not initialized";
return;
}
bool allowSiteBasedSplitTunneling = true;
// this block is for old native configs and for old self-hosted configs
auto protocolName = config.value(configKey::vpnProto).toString();
if (protocolName == ProtocolUtils::protoToString(Proto::Awg) || protocolName == ProtocolUtils::protoToString(Proto::WireGuard)) {
allowSiteBasedSplitTunneling = false;
auto configData = config.value(protocolName + "_config_data").toObject();
if (configData.value(configKey::allowedIps).isString()) {
QJsonArray allowedIpsJsonArray = QJsonArray::fromStringList(configData.value(configKey::allowedIps).toString().split(", "));
configData.insert(configKey::allowedIps, allowedIpsJsonArray);
config.insert(protocolName + "_config_data", configData);
} else if (configData.value(configKey::allowedIps).isUndefined()) {
auto nativeConfig = configData.value(configKey::config).toString();
auto nativeConfigLines = nativeConfig.split("\n");
for (auto &line : nativeConfigLines) {
if (line.contains("AllowedIPs")) {
auto allowedIpsString = line.split(" = ");
if (allowedIpsString.size() < 1) {
break;
}
QJsonArray allowedIpsJsonArray = QJsonArray::fromStringList(allowedIpsString.at(1).split(", "));
configData.insert(configKey::allowedIps, allowedIpsJsonArray);
config.insert(protocolName + "_config_data", configData);
break;
}
}
}
2024-04-26 18:44:27 +05:00
if (configData.value(configKey::persistentKeepAlive).isUndefined()) {
auto nativeConfig = configData.value(configKey::config).toString();
auto nativeConfigLines = nativeConfig.split("\n");
for (auto &line : nativeConfigLines) {
if (line.contains("PersistentKeepalive")) {
auto persistentKeepaliveString = line.split(" = ");
if (persistentKeepaliveString.size() < 1) {
break;
}
configData.insert(configKey::persistentKeepAlive, persistentKeepaliveString.at(1));
config.insert(protocolName + "_config_data", configData);
break;
}
}
}
QJsonArray allowedIpsJsonArray = configData.value(configKey::allowedIps).toArray();
if (allowedIpsJsonArray.contains("0.0.0.0/0") && allowedIpsJsonArray.contains("::/0")) {
allowSiteBasedSplitTunneling = true;
}
}
amnezia::RouteMode routeMode = amnezia::RouteMode::VpnAllSites;
QJsonArray sitesJsonArray;
if (m_appSettingsRepository->isSitesSplitTunnelingEnabled()) {
routeMode = m_appSettingsRepository->routeMode();
if (allowSiteBasedSplitTunneling) {
QStringList sites;
const QVariantMap &m = m_appSettingsRepository->vpnSites(routeMode);
for (auto i = m.constBegin(); i != m.constEnd(); ++i) {
if (NetworkUtilities::checkIpSubnetFormat(i.key())) {
sites.append(i.key());
} else if (NetworkUtilities::checkIpSubnetFormat(i.value().toString())) {
sites.append(i.value().toString());
}
}
sites.removeDuplicates();
2024-04-26 18:44:27 +05:00
for (const auto &site : sites) {
sitesJsonArray.append(site);
}
2023-10-13 15:45:06 +05:00
2025-04-10 11:24:33 +04:00
if (sitesJsonArray.isEmpty()) {
routeMode = amnezia::RouteMode::VpnAllSites;
} else if (routeMode == amnezia::RouteMode::VpnOnlyForwardSites) {
2025-04-10 11:24:33 +04:00
// Allow traffic to Amnezia DNS
sitesJsonArray.append(config.value(configKey::dns1).toString());
sitesJsonArray.append(config.value(configKey::dns2).toString());
2024-04-26 18:44:27 +05:00
}
}
2023-10-25 22:19:07 +03:00
}
config.insert(configKey::splitTunnelType, routeMode);
config.insert(configKey::splitTunnelSites, sitesJsonArray);
amnezia::AppsRouteMode appsRouteMode = amnezia::AppsRouteMode::VpnAllApps;
QJsonArray appsJsonArray;
if (m_appSettingsRepository->isAppsSplitTunnelingEnabled()) {
appsRouteMode = m_appSettingsRepository->appsRouteMode();
2024-04-01 18:45:00 +07:00
auto apps = m_appSettingsRepository->vpnApps(appsRouteMode);
for (const auto &app : apps) {
appsJsonArray.append(app.appPath.isEmpty() ? app.packageName : app.appPath);
}
2025-04-10 11:24:33 +04:00
if (appsJsonArray.isEmpty()) {
appsRouteMode = amnezia::AppsRouteMode::VpnAllApps;
2025-04-10 11:24:33 +04:00
}
2024-04-01 18:45:00 +07:00
}
config.insert(configKey::appSplitTunnelType, appsRouteMode);
config.insert(configKey::splitTunnelApps, appsJsonArray);
2026-02-11 17:44:11 +02:00
qDebug() << QString("Site split tunneling is %1, route mode is %2")
.arg(m_appSettingsRepository->isSitesSplitTunnelingEnabled() ? "enabled" : "disabled")
2026-02-11 17:44:11 +02:00
.arg(routeMode);
qDebug() << QString("App split tunneling is %1, route mode is %2")
.arg(m_appSettingsRepository->isAppsSplitTunnelingEnabled() ? "enabled" : "disabled")
2026-02-11 17:44:11 +02:00
.arg(appsRouteMode);
2023-10-13 15:45:06 +05:00
}
2023-03-15 17:46:22 +03:00
#ifdef Q_OS_ANDROID
2023-08-28 22:03:28 +03:00
void VpnConnection::restoreConnection()
{
createAndroidConnections();
m_vpnProtocol.reset(androidVpnProtocol);
createProtocolConnections();
}
void VpnConnection::createAndroidConnections()
{
2023-12-04 18:23:08 +03:00
androidVpnProtocol = createDefaultAndroidVpnProtocol();
2023-08-28 22:03:28 +03:00
connect(AndroidController::instance(), &AndroidController::connectionStateChanged, androidVpnProtocol,
&AndroidVpnProtocol::setConnectionState);
connect(AndroidController::instance(), &AndroidController::statisticsUpdated, androidVpnProtocol, &AndroidVpnProtocol::setBytesChanged);
}
2023-12-04 18:23:08 +03:00
AndroidVpnProtocol *VpnConnection::createDefaultAndroidVpnProtocol()
{
2023-12-04 18:23:08 +03:00
return new AndroidVpnProtocol(m_vpnConfiguration);
}
2023-03-15 17:46:22 +03:00
#endif
2021-01-09 19:55:16 +03:00
QString VpnConnection::bytesPerSecToText(quint64 bytes)
2020-12-26 15:03:51 +03:00
{
2021-01-09 19:55:16 +03:00
double mbps = bytes * 8 / 1e6;
return QString("%1 %2").arg(QString::number(mbps, 'f', 2)).arg(tr("Mbps")); // Mbit/s
2020-12-26 15:03:51 +03:00
}
2026-02-19 13:21:49 +01:00
void VpnConnection::reconnectToVpn() {
if (m_connectionState != Vpn::ConnectionState::Connected) {
qWarning() << QString("Reconnect triggered on %1 during inappropriate state: %2; ignoring slot")
.arg(QMetaEnum::fromType<Vpn::ConnectionState>().valueToKey(m_connectionState));
return;
}
qDebug() << "Reconnect triggered. Reconnecting to the server";
setConnectionState(Vpn::ConnectionState::Reconnecting);
#ifdef AMNEZIA_DESKTOP
if (m_active) {
m_active->restart();
return;
}
#endif
if (m_vpnProtocol.isNull())
return;
2026-02-19 13:21:49 +01:00
m_vpnProtocol->stop();
if (ErrorCode err = m_vpnProtocol->start(); err != ErrorCode::NoError) {
setConnectionState(Vpn::ConnectionState::Error);
emit vpnProtocolError(err);
}
}
2020-12-26 15:03:51 +03:00
void VpnConnection::disconnectFromVpn()
{
#if defined(Q_OS_IOS) || defined(MACOS_NE)
// iOS/macOS NE use IosController directly; m_vpnProtocol is not set there.
IosController::Instance()->disconnectVpn();
disconnect(&m_checkTimer, &QTimer::timeout, IosController::Instance(), &IosController::checkStatus);
#endif
#ifdef AMNEZIA_DESKTOP
if (m_staging) {
m_trafficGuard->tearDown(m_staging);
releaseIfname(m_staging->ifname());
delete m_staging;
m_staging = nullptr;
}
if (m_active) {
setConnectionState(Vpn::ConnectionState::Disconnecting);
m_trafficGuard->tearDown(m_active);
m_trafficGuard->flushAll();
releaseIfname(m_active->ifname());
delete m_active;
m_active = nullptr;
setConnectionState(Vpn::ConnectionState::Disconnected);
return;
}
#endif
2025-12-19 04:09:50 +01:00
if (m_vpnProtocol.isNull()) {
2026-02-19 13:21:49 +01:00
setConnectionState(Vpn::ConnectionState::Disconnected);
2025-12-19 04:09:50 +01:00
return;
}
2025-11-30 18:49:16 -08:00
2026-02-19 13:21:49 +01:00
setConnectionState(Vpn::ConnectionState::Disconnecting);
2022-02-22 02:08:57 +03:00
#ifdef Q_OS_ANDROID
2025-12-19 04:09:50 +01:00
auto *const connection = new QMetaObject::Connection;
*connection = connect(AndroidController::instance(), &AndroidController::vpnStateChanged, this,
[this, connection](AndroidController::ConnectionState state) {
if (state == AndroidController::ConnectionState::DISCONNECTED) {
setConnectionState(Vpn::ConnectionState::Disconnected);
2025-12-19 04:09:50 +01:00
disconnect(*connection);
delete connection;
}
});
2022-02-22 02:08:57 +03:00
#endif
#ifdef AMNEZIA_DESKTOP
m_trafficGuard->flushAll();
#endif
2026-02-19 13:21:49 +01:00
m_vpnProtocol->stop();
2025-11-30 18:49:16 -08:00
#if !defined(Q_OS_ANDROID) && !defined(AMNEZIA_DESKTOP)
2025-12-19 04:09:50 +01:00
m_vpnProtocol->deleteLater();
#endif
2025-11-30 18:49:16 -08:00
m_vpnProtocol = nullptr;
2020-12-26 15:03:51 +03:00
}
2026-02-19 13:21:49 +01:00
void VpnConnection::setConnectionState(Vpn::ConnectionState state) {
onConnectionStateChanged(state);
2021-01-15 23:36:35 +03:00
2026-02-19 13:21:49 +01:00
if (state == Vpn::Disconnected && m_connectionState == Vpn::Reconnecting)
return;
2020-12-26 15:03:51 +03:00
2026-02-19 13:21:49 +01:00
m_connectionState = state;
emit connectionStateChanged(state);
2020-12-26 15:03:51 +03:00
}
void VpnConnection::startTunnelSwitch(DockerContainer container,
const QJsonObject &vpnConfiguration,
const QString &resolvedRemote,
const QString &stagingIfname)
{
QJsonObject config = vpnConfiguration;
config.insert("ifname", stagingIfname);
if (VpnProtocol::isXrayBased(container)) {
config.insert("tunName", stagingIfname);
config.insert("deviceIpv4Address", amnezia::protocols::xray::defaultLocalAddr);
} else if (VpnProtocol::isWireGuardBased(container)) {
const QString protoName = config.value("protocol").toString();
const QJsonObject wgConfig = config.value(protoName + "_config_data").toObject();
const QString clientIp = wgConfig.value(amnezia::configKey::clientIp).toString();
if (!clientIp.isEmpty()) {
config.insert("deviceIpv4Address", clientIp);
}
}
appendKillSwitchConfig(config);
appendSplitTunnelingConfig(config);
m_staging = new Tunnel(stagingIfname, container, config, resolvedRemote, this);
if (m_active) {
m_staging->setHandoverIfname(m_active->ifname());
}
wireTunnelSignals(m_staging, /*isActive=*/false);
setConnectionState(Vpn::ConnectionState::Switching);
m_trafficGuard->bringUp(m_staging);
}
void VpnConnection::onTunnelPrepared()
{
Tunnel* tunnel = qobject_cast<Tunnel*>(sender());
if (!tunnel) return;
if (tunnel == m_staging && m_active) {
Tunnel* oldTunnel = m_active;
const QString oldIfname = oldTunnel->ifname();
m_active = m_staging;
m_staging = nullptr;
connect(m_active, &Tunnel::bytesChanged, this, &VpnConnection::onBytesChanged);
m_vpnConfiguration = m_active->config();
m_remoteAddress = m_active->remoteAddress();
m_trafficGuard->setConfig(m_vpnConfiguration);
// Run the swap from a clean event-loop tick so the nested QEventLoop inside
// VpnTrafficGuard::swap does not deadlock the LSC.readData stack frame that
// delivered Tunnel::prepared.
QMetaObject::invokeMethod(this, [this, oldTunnel, oldIfname]() {
m_trafficGuard->swap(oldTunnel, m_active);
delete oldTunnel;
releaseIfname(oldIfname);
}, Qt::QueuedConnection);
return;
}
m_trafficGuard->commit(tunnel);
}
void VpnConnection::onTunnelActivated()
{
Tunnel* tunnel = qobject_cast<Tunnel*>(sender());
if (!tunnel) return;
if (tunnel == m_active) {
setConnectionState(Vpn::ConnectionState::Connected);
}
}
void VpnConnection::onTunnelFailed(amnezia::ErrorCode error)
{
Tunnel* tunnel = qobject_cast<Tunnel*>(sender());
if (!tunnel) return;
if (tunnel == m_staging) {
m_trafficGuard->release(m_staging);
m_staging->deactivate();
releaseIfname(m_staging->ifname());
m_staging->deleteLater();
m_staging = nullptr;
setConnectionState(Vpn::ConnectionState::Connected);
emit serverSwitchFailed();
return;
}
if (tunnel == m_active) {
m_trafficGuard->tearDown(m_active);
m_trafficGuard->flushAll();
releaseIfname(m_active->ifname());
m_active->deleteLater();
m_active = nullptr;
setConnectionState(Vpn::ConnectionState::Error);
if (error != ErrorCode::NoError) {
emit vpnProtocolError(error);
}
}
}