Files
amnezia-client/client/vpnconnection.cpp
T

381 lines
13 KiB
C++
Raw Normal View History

2021-01-09 19:55:16 +03:00
#include <QApplication>
2020-12-26 15:03:51 +03:00
#include <QDebug>
2021-01-06 17:12:24 +03:00
#include <QFile>
2021-02-18 15:00:41 +03:00
#include <QJsonObject>
2020-12-26 15:03:51 +03:00
2021-04-04 23:12:36 +03:00
#include <configurators/openvpn_configurator.h>
#include <configurators/cloak_configurator.h>
2021-05-07 23:28:37 +03:00
#include <configurators/shadowsocks_configurator.h>
2021-06-12 11:59:36 +03:00
#include <configurators/wireguard_configurator.h>
2021-01-06 17:12:24 +03:00
#include <core/servercontroller.h>
2021-06-12 11:59:36 +03:00
#include <protocols/wireguardprotocol.h>
2021-01-06 17:12:24 +03:00
2021-02-02 01:47:40 +03:00
#include "ipc.h"
2021-02-18 15:00:41 +03:00
#include "core/ipcclient.h"
2021-01-06 17:12:24 +03:00
#include "protocols/openvpnprotocol.h"
2021-04-04 23:12:36 +03:00
#include "protocols/openvpnovercloakprotocol.h"
2021-01-15 23:36:35 +03:00
#include "protocols/shadowsocksvpnprotocol.h"
2021-04-04 23:12:36 +03:00
2021-01-06 17:12:24 +03:00
#include "utils.h"
2020-12-26 15:03:51 +03:00
#include "vpnconnection.h"
VpnConnection::VpnConnection(QObject* parent) : QObject(parent)
{
2021-02-18 15:00:41 +03:00
QTimer::singleShot(0, this, [this](){
2021-02-03 15:42:36 +03:00
if (!IpcClient::init()) {
qWarning() << "Error occured when init IPC client";
2021-02-18 15:00:41 +03:00
emit serviceIsNotReady();
2021-02-03 15:42:36 +03:00
}
});
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()
{
m_vpnProtocol.clear();
2020-12-26 15:03:51 +03:00
}
void VpnConnection::onBytesChanged(quint64 receivedBytes, quint64 sentBytes)
{
emit bytesChanged(receivedBytes, sentBytes);
}
void VpnConnection::onConnectionStateChanged(VpnProtocol::ConnectionState state)
{
2021-02-18 15:00:41 +03:00
if (IpcClient::Interface()) {
2021-06-01 18:18:09 +03:00
if (state == VpnProtocol::Connected){
2021-02-18 15:00:41 +03:00
IpcClient::Interface()->flushDns();
2021-06-12 11:59:36 +03:00
if (m_settings.routeMode() != Settings::VpnAllSites) {
2021-06-01 18:18:09 +03:00
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());
2021-01-26 15:01:15 +03:00
2021-06-01 18:18:09 +03:00
if (m_settings.routeMode() == Settings::VpnOnlyForwardSites) {
IpcClient::Interface()->routeAddList(m_vpnProtocol->vpnGateway(), m_settings.getVpnIps(Settings::VpnOnlyForwardSites));
}
else if (m_settings.routeMode() == Settings::VpnAllExceptSites) {
2021-06-12 11:59:36 +03:00
IpcClient::Interface()->routeAddList(m_vpnProtocol->vpnGateway(), QStringList() << "0.0.0.0/1");
IpcClient::Interface()->routeAddList(m_vpnProtocol->vpnGateway(), QStringList() << "128.0.0.0/1");
IpcClient::Interface()->routeAddList(m_vpnProtocol->routeGateway(), QStringList() << remoteAddress());
2021-06-01 18:18:09 +03:00
IpcClient::Interface()->routeAddList(m_vpnProtocol->routeGateway(), m_settings.getVpnIps(Settings::VpnAllExceptSites));
2021-02-18 15:00:41 +03:00
}
2021-06-01 18:18:09 +03:00
2021-02-18 15:00:41 +03:00
}
2021-06-01 18:18:09 +03:00
else if (state == VpnProtocol::Error) {
2021-02-18 15:00:41 +03:00
IpcClient::Interface()->flushDns();
2021-01-26 15:01:15 +03:00
2021-05-27 22:18:36 +03:00
if (m_settings.routeMode() == Settings::VpnOnlyForwardSites) {
2021-02-18 15:00:41 +03:00
IpcClient::Interface()->clearSavedRoutes();
}
}
}
2021-01-26 15:01:15 +03:00
2020-12-26 15:03:51 +03:00
emit connectionStateChanged(state);
}
2021-06-12 11:59:36 +03:00
const QString &VpnConnection::remoteAddress() const
{
return m_remoteAddress;
}
2021-02-18 15:00:41 +03:00
QSharedPointer<VpnProtocol> VpnConnection::vpnProtocol() const
{
return m_vpnProtocol;
}
2021-06-01 18:18:09 +03:00
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();
}
2021-01-06 17:12:24 +03:00
ErrorCode VpnConnection::lastError() const
2020-12-26 23:17:20 +03:00
{
if (!m_vpnProtocol.data()) {
2021-01-06 17:12:24 +03:00
return ErrorCode::InternalError;
2020-12-26 23:17:20 +03:00
}
return m_vpnProtocol.data()->lastError();
}
2021-05-07 23:28:37 +03:00
QMap<Protocol, QString> VpnConnection::getLastVpnConfig(const QJsonObject &containerConfig)
{
QMap<Protocol, QString> 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)) {
2021-05-14 04:27:30 -07:00
configData = lastVpnConfig.value(proto);
if (proto == Protocol::OpenVpn) {
configData = OpenVpnConfigurator::processConfigWithLocalSettings(configData);
}
2021-05-10 02:33:31 +03:00
qDebug() << "VpnConnection::createVpnConfiguration: using saved config for" << protoToString(proto);
2021-05-07 23:28:37 +03:00
}
else {
2021-05-10 02:33:31 +03:00
qDebug() << "VpnConnection::createVpnConfiguration: gen new config for" << protoToString(proto);
2021-05-07 23:28:37 +03:00
if (proto == Protocol::OpenVpn) {
configData = OpenVpnConfigurator::genOpenVpnConfig(credentials,
container, containerConfig, &e);
2021-05-20 15:59:58 +03:00
configData = OpenVpnConfigurator::processConfigWithLocalSettings(configData);
2021-05-07 23:28:37 +03:00
}
else if (proto == Protocol::Cloak) {
configData = CloakConfigurator::genCloakConfig(credentials,
container, containerConfig, &e);
}
else if (proto == Protocol::ShadowSocks) {
configData = ShadowSocksConfigurator::genShadowSocksConfig(credentials,
container, containerConfig, &e);
}
2021-06-12 11:59:36 +03:00
else if (proto == Protocol::WireGuard) {
configData = WireguardConfigurator::genWireguardConfig(credentials,
container, containerConfig, &e);
}
2021-05-07 23:28:37 +03:00
if (errorCode && e) {
*errorCode = e;
return "";
}
if (serverIndex >= 0) {
2021-05-10 02:33:31 +03:00
qDebug() << "VpnConnection::createVpnConfiguration: saving config for server #" << serverIndex << container << proto;
2021-05-07 23:28:37 +03:00
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)
2020-12-26 15:03:51 +03:00
{
2021-01-06 17:12:24 +03:00
ErrorCode errorCode = ErrorCode::NoError;
2021-05-07 23:28:37 +03:00
if (container == DockerContainer::OpenVpn ||
container == DockerContainer::OpenVpnOverShadowSocks ||
container == DockerContainer::OpenVpnOverCloak) {
QString openVpnConfigData =
createVpnConfigurationForProto(
serverIndex, credentials, container, containerConfig, Protocol::OpenVpn, &errorCode);
2021-04-20 02:09:47 +03:00
m_vpnConfiguration.insert(config::key_openvpn_config_data, openVpnConfigData);
2021-01-06 17:12:24 +03:00
if (errorCode) {
return errorCode;
}
2021-06-12 11:59:36 +03:00
QFile file(OpenVpnProtocol::defaultConfigFileName());
2021-01-06 17:12:24 +03:00
if (file.open(QIODevice::WriteOnly | QIODevice::Truncate)){
QTextStream stream(&file);
2021-02-18 15:00:41 +03:00
stream << openVpnConfigData << endl;
2021-04-04 23:12:36 +03:00
file.close();
2021-02-18 15:00:41 +03:00
}
else {
return ErrorCode::FailedToSaveConfigData;
2021-01-06 17:12:24 +03:00
}
}
2021-05-07 23:28:37 +03:00
if (container == DockerContainer::OpenVpnOverShadowSocks) {
QJsonObject ssConfigData = QJsonDocument::fromJson(
createVpnConfigurationForProto(
serverIndex, credentials, container, containerConfig, Protocol::ShadowSocks, &errorCode).toUtf8()).
object();
2021-04-20 02:09:47 +03:00
m_vpnConfiguration.insert(config::key_shadowsocks_config_data, ssConfigData);
2021-02-18 15:00:41 +03:00
}
2021-03-13 14:16:24 +03:00
2021-04-26 23:19:19 +03:00
if (container == DockerContainer::OpenVpnOverCloak) {
2021-05-07 23:28:37 +03:00
QJsonObject cloakConfigData = QJsonDocument::fromJson(
createVpnConfigurationForProto(
serverIndex, credentials, container, containerConfig, Protocol::Cloak, &errorCode).toUtf8()).
object();
2021-04-20 02:09:47 +03:00
m_vpnConfiguration.insert(config::key_cloak_config_data, cloakConfigData);
2021-04-04 23:12:36 +03:00
}
2021-06-12 11:59:36 +03:00
if (container == DockerContainer::WireGuard) {
QString wgConfigData = createVpnConfigurationForProto(
serverIndex, credentials, container, containerConfig, Protocol::WireGuard, &errorCode);
m_vpnConfiguration.insert(config::key_wireguard_config_data, wgConfigData);
}
2021-03-13 14:16:24 +03:00
//qDebug().noquote() << "VPN config" << QJsonDocument(m_vpnConfiguration).toJson();
2021-02-18 15:00:41 +03:00
return ErrorCode::NoError;
2021-01-06 17:12:24 +03:00
}
2021-06-01 18:18:09 +03:00
ErrorCode VpnConnection::connectToVpn(int serverIndex,
const ServerCredentials &credentials, DockerContainer container, const QJsonObject &containerConfig)
2021-01-06 17:12:24 +03:00
{
2021-06-12 11:59:36 +03:00
qDebug() << QString("СonnectToVpn, Server index is %1, container is %2, route mode is")
.arg(serverIndex).arg(containerToString(container)) << m_settings.routeMode();
m_remoteAddress = credentials.hostName;
2020-12-26 15:03:51 +03:00
2021-06-01 18:18:09 +03:00
emit connectionStateChanged(VpnProtocol::Connecting);
2021-02-18 15:00:41 +03:00
2021-06-12 11:59:36 +03:00
if (credentials.isValid()) {
ServerController::setupServerFirewall(credentials);
}
2021-05-10 02:33:31 +03:00
2021-02-18 15:00:41 +03:00
if (m_vpnProtocol) {
disconnect(m_vpnProtocol.data(), &VpnProtocol::protocolError, this, &VpnConnection::vpnProtocolError);
m_vpnProtocol->stop();
2021-02-25 18:05:42 +03:00
m_vpnProtocol.reset();
2021-02-18 15:00:41 +03:00
}
2021-04-26 23:19:19 +03:00
if (container == DockerContainer::None || container == DockerContainer::OpenVpn) {
2021-05-07 23:28:37 +03:00
ErrorCode e = createVpnConfiguration(serverIndex, credentials, DockerContainer::OpenVpn, containerConfig);
2021-01-06 17:12:24 +03:00
if (e) {
2021-06-01 18:18:09 +03:00
emit connectionStateChanged(VpnProtocol::Error);
2021-01-06 17:12:24 +03:00
return e;
}
2021-02-18 15:00:41 +03:00
m_vpnProtocol.reset(new OpenVpnProtocol(m_vpnConfiguration));
e = static_cast<OpenVpnProtocol *>(m_vpnProtocol.data())->checkAndSetupTapDriver();
if (e) {
2021-06-01 18:18:09 +03:00
emit connectionStateChanged(VpnProtocol::Error);
2021-02-18 15:00:41 +03:00
return e;
2021-01-08 16:51:58 +03:00
}
2021-01-06 17:12:24 +03:00
}
2021-05-07 23:28:37 +03:00
else if (container == DockerContainer::OpenVpnOverShadowSocks) {
ErrorCode e = createVpnConfiguration(serverIndex, credentials, DockerContainer::OpenVpnOverShadowSocks, containerConfig);
2021-01-15 23:36:35 +03:00
if (e) {
2021-06-01 18:18:09 +03:00
emit connectionStateChanged(VpnProtocol::Error);
2021-01-15 23:36:35 +03:00
return e;
}
2021-02-18 15:00:41 +03:00
m_vpnProtocol.reset(new ShadowSocksVpnProtocol(m_vpnConfiguration));
2021-02-25 18:05:42 +03:00
e = static_cast<OpenVpnProtocol *>(m_vpnProtocol.data())->checkAndSetupTapDriver();
if (e) {
2021-06-01 18:18:09 +03:00
emit connectionStateChanged(VpnProtocol::Error);
2021-02-25 18:05:42 +03:00
return e;
}
2020-12-26 15:03:51 +03:00
}
2021-04-26 23:19:19 +03:00
else if (container == DockerContainer::OpenVpnOverCloak) {
2021-05-07 23:28:37 +03:00
ErrorCode e = createVpnConfiguration(serverIndex, credentials, DockerContainer::OpenVpnOverCloak, containerConfig);
2021-04-04 23:12:36 +03:00
if (e) {
2021-06-01 18:18:09 +03:00
emit connectionStateChanged(VpnProtocol::Error);
2021-04-04 23:12:36 +03:00
return e;
}
m_vpnProtocol.reset(new OpenVpnOverCloakProtocol(m_vpnConfiguration));
e = static_cast<OpenVpnProtocol *>(m_vpnProtocol.data())->checkAndSetupTapDriver();
if (e) {
2021-06-01 18:18:09 +03:00
emit connectionStateChanged(VpnProtocol::Error);
2021-04-04 23:12:36 +03:00
return e;
}
}
2021-06-12 11:59:36 +03:00
else if (container == DockerContainer::WireGuard) {
ErrorCode e = createVpnConfiguration(serverIndex, credentials, DockerContainer::WireGuard, containerConfig);
if (e) {
emit connectionStateChanged(VpnProtocol::Error);
return e;
}
m_vpnProtocol.reset(new WireguardProtocol(m_vpnConfiguration));
}
2020-12-26 15:03:51 +03:00
2021-02-18 15:00:41 +03:00
connect(m_vpnProtocol.data(), &VpnProtocol::protocolError, this, &VpnConnection::vpnProtocolError);
2020-12-26 15:03:51 +03:00
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)));
2021-04-26 22:54:31 +03:00
ServerController::disconnectFromHost(credentials);
2020-12-26 23:17:20 +03:00
return m_vpnProtocol.data()->start();
2020-12-26 15:03:51 +03:00
}
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
}
void VpnConnection::disconnectFromVpn()
{
qDebug() << "Disconnect from VPN";
2021-05-20 15:59:58 +03:00
if (IpcClient::Interface()) {
IpcClient::Interface()->flushDns();
2021-02-24 21:58:32 +03:00
2021-06-01 18:18:09 +03:00
// delete cached routes
IpcClient::Interface()->clearSavedRoutes();
2021-02-24 21:58:32 +03:00
}
2021-01-26 15:01:15 +03:00
2020-12-26 15:03:51 +03:00
if (!m_vpnProtocol.data()) {
return;
}
m_vpnProtocol.data()->stop();
}
2021-01-15 23:36:35 +03:00
VpnProtocol::ConnectionState VpnConnection::connectionState()
{
2021-06-01 18:18:09 +03:00
if (!m_vpnProtocol) return VpnProtocol::Disconnected;
2021-01-15 23:36:35 +03:00
return m_vpnProtocol->connectionState();
}
2021-02-18 15:00:41 +03:00
bool VpnConnection::isConnected() const
2020-12-26 15:03:51 +03:00
{
if (!m_vpnProtocol.data()) {
return false;
}
2021-02-18 15:00:41 +03:00
return m_vpnProtocol.data()->isConnected();
2020-12-26 15:03:51 +03:00
}
2021-02-18 15:00:41 +03:00
bool VpnConnection::isDisconnected() const
2020-12-26 15:03:51 +03:00
{
if (!m_vpnProtocol.data()) {
return true;
}
2021-02-18 15:00:41 +03:00
return m_vpnProtocol.data()->isDisconnected();
2020-12-26 15:03:51 +03:00
}