#include #include #include #include #include #include #include #include #include "ipc.h" #include "core/ipcclient.h" #include "protocols/openvpnprotocol.h" #include "protocols/openvpnovercloakprotocol.h" #include "protocols/shadowsocksvpnprotocol.h" #include "utils.h" #include "vpnconnection.h" VpnConnection::VpnConnection(QObject* parent) : QObject(parent) { QTimer::singleShot(0, this, [this](){ if (!IpcClient::init()) { qWarning() << "Error occured when init IPC client"; emit serviceIsNotReady(); } }); } VpnConnection::~VpnConnection() { m_vpnProtocol.clear(); } void VpnConnection::onBytesChanged(quint64 receivedBytes, quint64 sentBytes) { emit bytesChanged(receivedBytes, sentBytes); } void VpnConnection::onConnectionStateChanged(VpnProtocol::ConnectionState state) { if (IpcClient::Interface()) { if (state == VpnProtocol::Connected){ IpcClient::Interface()->flushDns(); if (m_settings.routeMode() == Settings::VpnOnlyForwardSites) { IpcClient::Interface()->routeDeleteList(m_vpnProtocol->vpnGateway(), QStringList() << "0.0.0.0"); //qDebug() << "VpnConnection::onConnectionStateChanged :: adding custom routes, count:" << forwardIps.size(); } IpcClient::Interface()->routeAddList(m_vpnProtocol->vpnGateway(), QStringList() << m_settings.primaryDns() << m_settings.secondaryDns()); if (m_settings.routeMode() == Settings::VpnOnlyForwardSites) { IpcClient::Interface()->routeAddList(m_vpnProtocol->vpnGateway(), m_settings.getVpnIps(Settings::VpnOnlyForwardSites)); } else if (m_settings.routeMode() == Settings::VpnAllExceptSites) { IpcClient::Interface()->routeAddList(m_vpnProtocol->routeGateway(), m_settings.getVpnIps(Settings::VpnAllExceptSites)); } } else if (state == VpnProtocol::Error) { IpcClient::Interface()->flushDns(); if (m_settings.routeMode() == Settings::VpnOnlyForwardSites) { IpcClient::Interface()->clearSavedRoutes(); } } } emit connectionStateChanged(state); } QSharedPointer VpnConnection::vpnProtocol() const { return m_vpnProtocol; } void VpnConnection::addRoutes(const QStringList &ips) { if (connectionState() == VpnProtocol::Connected && IpcClient::Interface()) { if (m_settings.routeMode() == Settings::VpnOnlyForwardSites) { IpcClient::Interface()->routeAddList(m_vpnProtocol->vpnGateway(), ips); } else if (m_settings.routeMode() == Settings::VpnAllExceptSites) { IpcClient::Interface()->routeAddList(m_vpnProtocol->routeGateway(), ips); } } } void VpnConnection::deleteRoutes(const QStringList &ips) { if (connectionState() == VpnProtocol::Connected && IpcClient::Interface()) { if (m_settings.routeMode() == Settings::VpnOnlyForwardSites) { IpcClient::Interface()->routeDeleteList(vpnProtocol()->vpnGateway(), ips); } else if (m_settings.routeMode() == Settings::VpnAllExceptSites) { IpcClient::Interface()->routeDeleteList(m_vpnProtocol->routeGateway(), ips); } } } void VpnConnection::flushDns() { if (IpcClient::Interface()) IpcClient::Interface()->flushDns(); } ErrorCode VpnConnection::lastError() const { if (!m_vpnProtocol.data()) { return ErrorCode::InternalError; } return m_vpnProtocol.data()->lastError(); } QMap VpnConnection::getLastVpnConfig(const QJsonObject &containerConfig) { QMap configs; for (Protocol proto: { Protocol::OpenVpn, Protocol::ShadowSocks, Protocol::Cloak, Protocol::WireGuard}) { QString cfg = containerConfig.value(protoToString(proto)).toObject().value(config_key::last_config).toString(); if (!cfg.isEmpty()) configs.insert(proto, cfg); } return configs; } QString VpnConnection::createVpnConfigurationForProto(int serverIndex, const ServerCredentials &credentials, DockerContainer container, const QJsonObject &containerConfig, Protocol proto, ErrorCode *errorCode) { ErrorCode e = ErrorCode::NoError; auto lastVpnConfig = getLastVpnConfig(containerConfig); QString configData; if (lastVpnConfig.contains(proto)) { configData = lastVpnConfig.value(proto); if (proto == Protocol::OpenVpn) { configData = OpenVpnConfigurator::processConfigWithLocalSettings(configData); } qDebug() << "VpnConnection::createVpnConfiguration: using saved config for" << protoToString(proto); } else { qDebug() << "VpnConnection::createVpnConfiguration: gen new config for" << protoToString(proto); if (proto == Protocol::OpenVpn) { configData = OpenVpnConfigurator::genOpenVpnConfig(credentials, container, containerConfig, &e); configData = OpenVpnConfigurator::processConfigWithLocalSettings(configData); } else if (proto == Protocol::Cloak) { configData = CloakConfigurator::genCloakConfig(credentials, container, containerConfig, &e); } else if (proto == Protocol::ShadowSocks) { configData = ShadowSocksConfigurator::genShadowSocksConfig(credentials, container, containerConfig, &e); } if (errorCode && e) { *errorCode = e; return ""; } if (serverIndex >= 0) { qDebug() << "VpnConnection::createVpnConfiguration: saving config for server #" << serverIndex << container << proto; QJsonObject protoObject = m_settings.protocolConfig(serverIndex, container, proto); protoObject.insert(config_key::last_config, configData); m_settings.setProtocolConfig(serverIndex, container, proto, protoObject); } } if (errorCode) *errorCode = e; return configData; } ErrorCode VpnConnection::createVpnConfiguration(int serverIndex, const ServerCredentials &credentials, DockerContainer container, const QJsonObject &containerConfig) { ErrorCode errorCode = ErrorCode::NoError; if (container == DockerContainer::OpenVpn || container == DockerContainer::OpenVpnOverShadowSocks || container == DockerContainer::OpenVpnOverCloak) { QString openVpnConfigData = createVpnConfigurationForProto( serverIndex, credentials, container, containerConfig, Protocol::OpenVpn, &errorCode); m_vpnConfiguration.insert(config::key_openvpn_config_data, openVpnConfigData); if (errorCode) { return errorCode; } QFile file(Utils::defaultVpnConfigFileName()); if (file.open(QIODevice::WriteOnly | QIODevice::Truncate)){ QTextStream stream(&file); stream << openVpnConfigData << endl; file.close(); } else { return ErrorCode::FailedToSaveConfigData; } } if (container == DockerContainer::OpenVpnOverShadowSocks) { QJsonObject ssConfigData = QJsonDocument::fromJson( createVpnConfigurationForProto( serverIndex, credentials, container, containerConfig, Protocol::ShadowSocks, &errorCode).toUtf8()). object(); m_vpnConfiguration.insert(config::key_shadowsocks_config_data, ssConfigData); } if (container == DockerContainer::OpenVpnOverCloak) { QJsonObject cloakConfigData = QJsonDocument::fromJson( createVpnConfigurationForProto( serverIndex, credentials, container, containerConfig, Protocol::Cloak, &errorCode).toUtf8()). object(); m_vpnConfiguration.insert(config::key_cloak_config_data, cloakConfigData); } //qDebug().noquote() << "VPN config" << QJsonDocument(m_vpnConfiguration).toJson(); return ErrorCode::NoError; } ErrorCode VpnConnection::connectToVpn(int serverIndex, const ServerCredentials &credentials, DockerContainer container, const QJsonObject &containerConfig) { qDebug() << "СonnectToVpn, Route mode is" << m_settings.routeMode(); emit connectionStateChanged(VpnProtocol::Connecting); ServerController::setupServerFirewall(credentials); if (m_vpnProtocol) { disconnect(m_vpnProtocol.data(), &VpnProtocol::protocolError, this, &VpnConnection::vpnProtocolError); m_vpnProtocol->stop(); m_vpnProtocol.reset(); } if (container == DockerContainer::None || container == DockerContainer::OpenVpn) { ErrorCode e = createVpnConfiguration(serverIndex, credentials, DockerContainer::OpenVpn, containerConfig); if (e) { emit connectionStateChanged(VpnProtocol::Error); return e; } m_vpnProtocol.reset(new OpenVpnProtocol(m_vpnConfiguration)); e = static_cast(m_vpnProtocol.data())->checkAndSetupTapDriver(); if (e) { emit connectionStateChanged(VpnProtocol::Error); return e; } } else if (container == DockerContainer::OpenVpnOverShadowSocks) { ErrorCode e = createVpnConfiguration(serverIndex, credentials, DockerContainer::OpenVpnOverShadowSocks, containerConfig); if (e) { emit connectionStateChanged(VpnProtocol::Error); return e; } m_vpnProtocol.reset(new ShadowSocksVpnProtocol(m_vpnConfiguration)); e = static_cast(m_vpnProtocol.data())->checkAndSetupTapDriver(); if (e) { emit connectionStateChanged(VpnProtocol::Error); return e; } } else if (container == DockerContainer::OpenVpnOverCloak) { ErrorCode e = createVpnConfiguration(serverIndex, credentials, DockerContainer::OpenVpnOverCloak, containerConfig); if (e) { emit connectionStateChanged(VpnProtocol::Error); return e; } m_vpnProtocol.reset(new OpenVpnOverCloakProtocol(m_vpnConfiguration)); e = static_cast(m_vpnProtocol.data())->checkAndSetupTapDriver(); if (e) { emit connectionStateChanged(VpnProtocol::Error); return e; } } connect(m_vpnProtocol.data(), &VpnProtocol::protocolError, this, &VpnConnection::vpnProtocolError); connect(m_vpnProtocol.data(), SIGNAL(connectionStateChanged(VpnProtocol::ConnectionState)), this, SLOT(onConnectionStateChanged(VpnProtocol::ConnectionState))); connect(m_vpnProtocol.data(), SIGNAL(bytesChanged(quint64, quint64)), this, SLOT(onBytesChanged(quint64, quint64))); ServerController::disconnectFromHost(credentials); return m_vpnProtocol.data()->start(); } QString VpnConnection::bytesPerSecToText(quint64 bytes) { double mbps = bytes * 8 / 1e6; return QString("%1 %2").arg(QString::number(mbps, 'f', 2)).arg(tr("Mbps")); // Mbit/s } void VpnConnection::disconnectFromVpn() { qDebug() << "Disconnect from VPN"; if (IpcClient::Interface()) { IpcClient::Interface()->flushDns(); // delete cached routes IpcClient::Interface()->clearSavedRoutes(); } if (!m_vpnProtocol.data()) { return; } m_vpnProtocol.data()->stop(); } VpnProtocol::ConnectionState VpnConnection::connectionState() { if (!m_vpnProtocol) return VpnProtocol::Disconnected; return m_vpnProtocol->connectionState(); } bool VpnConnection::isConnected() const { if (!m_vpnProtocol.data()) { return false; } return m_vpnProtocol.data()->isConnected(); } bool VpnConnection::isDisconnected() const { if (!m_vpnProtocol.data()) { return true; } return m_vpnProtocol.data()->isDisconnected(); }