Files
DefaultVPN/client/protocols/openvpnprotocol.cpp

323 lines
9.8 KiB
C++
Raw Normal View History

2020-12-26 15:03:51 +03:00
#include <QCoreApplication>
#include <QFileInfo>
#include <QProcess>
#include <QRegularExpression>
2021-01-07 20:53:42 +03:00
#include <QTcpSocket>
2020-12-26 15:03:51 +03:00
#include "debug.h"
2021-06-12 11:59:36 +03:00
#include "defines.h"
2020-12-26 15:03:51 +03:00
#include "utils.h"
2021-06-12 11:59:36 +03:00
#include "openvpnprotocol.h"
2020-12-26 15:03:51 +03:00
OpenVpnProtocol::OpenVpnProtocol(const QJsonObject &configuration, QObject* parent) :
VpnProtocol(configuration, parent)
2020-12-26 15:03:51 +03:00
{
readOpenVpnConfiguration(configuration);
2020-12-26 15:03:51 +03:00
connect(&m_managementServer, &ManagementServer::readyRead, this, &OpenVpnProtocol::onReadyReadDataFromManagementServer);
}
OpenVpnProtocol::~OpenVpnProtocol()
{
qDebug() << "OpenVpnProtocol::~OpenVpnProtocol()";
2021-01-15 23:36:35 +03:00
OpenVpnProtocol::stop();
QThread::msleep(200);
2020-12-26 15:03:51 +03:00
}
2021-06-12 11:59:36 +03:00
QString OpenVpnProtocol::defaultConfigFileName()
{
2021-06-19 16:38:35 +03:00
//qDebug() << "OpenVpnProtocol::defaultConfigFileName" << defaultConfigPath() + QString("/%1.ovpn").arg(APPLICATION_NAME);
2021-06-12 11:59:36 +03:00
return defaultConfigPath() + QString("/%1.ovpn").arg(APPLICATION_NAME);
}
QString OpenVpnProtocol::defaultConfigPath()
{
2021-06-19 16:38:35 +03:00
QString p = QStandardPaths::writableLocation(QStandardPaths::AppDataLocation) + "/config";
Utils::initializePath(p);
return p;
2021-06-12 11:59:36 +03:00
}
2020-12-26 15:03:51 +03:00
void OpenVpnProtocol::stop()
{
qDebug() << "OpenVpnProtocol::stop()";
2021-01-09 19:55:16 +03:00
// TODO: need refactoring
// sendTermSignal() will even return true while server connected ???
2021-06-01 18:18:09 +03:00
if ((m_connectionState == VpnProtocol::Preparing) ||
(m_connectionState == VpnProtocol::Connecting) ||
(m_connectionState == VpnProtocol::Connected) ||
(m_connectionState == VpnProtocol::Reconnecting)) {
2020-12-26 15:03:51 +03:00
if (!sendTermSignal()) {
killOpenVpnProcess();
}
m_managementServer.stop();
qApp->processEvents();
2021-06-01 18:18:09 +03:00
setConnectionState(VpnProtocol::Disconnecting);
2020-12-26 15:03:51 +03:00
}
}
ErrorCode OpenVpnProtocol::checkAndSetupTapDriver()
2020-12-26 15:03:51 +03:00
{
if (!IpcClient::Interface()) {
return ErrorCode::AmneziaServiceConnectionFailed;
}
QRemoteObjectPendingReply<QStringList> resultCheck = IpcClient::Interface()->getTapList();
resultCheck.waitForFinished();
if (resultCheck.returnValue().isEmpty()){
QRemoteObjectPendingReply<bool> resultInstall = IpcClient::Interface()->checkAndInstallDriver();
resultInstall.waitForFinished();
2021-06-03 20:23:44 +03:00
if (!resultInstall.returnValue()) return ErrorCode::OpenVpnTapAdapterError;
}
return ErrorCode::NoError;
2020-12-26 15:03:51 +03:00
}
void OpenVpnProtocol::killOpenVpnProcess()
2020-12-26 15:03:51 +03:00
{
2021-09-15 08:03:28 -07:00
#ifndef Q_OS_IOS
if (m_openVpnProcess){
m_openVpnProcess->close();
2020-12-26 15:03:51 +03:00
}
2021-09-15 08:03:28 -07:00
#endif
}
void OpenVpnProtocol::readOpenVpnConfiguration(const QJsonObject &configuration)
{
2021-04-20 02:09:47 +03:00
if (configuration.contains(config::key_openvpn_config_data)) {
m_configFile.open();
2021-04-20 02:09:47 +03:00
m_configFile.write(configuration.value(config::key_openvpn_config_data).toString().toUtf8());
m_configFile.close();
m_configFileName = m_configFile.fileName();
2020-12-26 15:03:51 +03:00
qDebug().noquote() << QString("Set config data") << m_configFileName;
2020-12-26 15:03:51 +03:00
}
2021-04-20 02:09:47 +03:00
else if (configuration.contains(config::key_openvpn_config_path)) {
m_configFileName = configuration.value(config::key_openvpn_config_path).toString();
QFileInfo file(m_configFileName);
2020-12-26 15:03:51 +03:00
if (file.fileName().isEmpty()) {
2021-06-12 11:59:36 +03:00
m_configFileName = defaultConfigFileName();
}
2020-12-26 15:03:51 +03:00
qDebug().noquote() << QString("Set config file: '%1'").arg(configPath());
}
2020-12-26 15:03:51 +03:00
}
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)
2020-12-26 15:03:51 +03:00
{
2021-01-07 20:53:42 +03:00
QIODevice *device = dynamic_cast<QIODevice*>(m_managementServer.socket().data());
if (device) {
QTextStream stream(device);
2021-02-21 09:44:53 -08:00
stream << command << Qt::endl;
2021-01-07 20:53:42 +03:00
}
2020-12-26 15:03:51 +03:00
}
void OpenVpnProtocol::updateRouteGateway(QString line)
{
2021-02-21 09:44:53 -08:00
// TODO: fix for macos
line = line.split("ROUTE_GATEWAY", QString::SkipEmptyParts).at(1);
if (!line.contains("/")) return;
m_routeGateway = line.split("/", QString::SkipEmptyParts).first();
m_routeGateway.replace(" ", "");
qDebug() << "Set VPN route gateway" << m_routeGateway;
}
2020-12-26 15:03:51 +03:00
QString OpenVpnProtocol::openVpnExecPath() const
{
#ifdef Q_OS_WIN
2021-06-01 18:18:09 +03:00
return Utils::executable("openvpn/openvpn", true);
2021-08-04 10:08:00 -07:00
#elif defined Q_OS_LINUX
return Utils::usrExecutable("openvpn");
2020-12-26 15:03:51 +03:00
#else
2021-06-01 18:18:09 +03:00
return Utils::executable("/openvpn", true);
2020-12-26 15:03:51 +03:00
#endif
}
2021-01-06 17:12:24 +03:00
ErrorCode OpenVpnProtocol::start()
2020-12-26 15:03:51 +03:00
{
2021-09-15 08:03:28 -07:00
#ifndef Q_OS_IOS
//qDebug() << "Start OpenVPN connection";
2021-01-15 23:36:35 +03:00
OpenVpnProtocol::stop();
2020-12-26 15:03:51 +03:00
if (!QFileInfo::exists(openVpnExecPath())) {
2021-01-06 17:12:24 +03:00
setLastError(ErrorCode::OpenVpnExecutableMissing);
return lastError();
2020-12-26 15:03:51 +03:00
}
if (!QFileInfo::exists(configPath())) {
2021-01-06 17:12:24 +03:00
setLastError(ErrorCode::OpenVpnConfigMissing);
return lastError();
2020-12-26 15:03:51 +03:00
}
QString vpnLogFileNamePath = Utils::systemLogPath() + "/openvpn.log";
Utils::createEmptyFile(vpnLogFileNamePath);
if (!m_managementServer.start(m_managementHost, m_managementPort)) {
2021-01-06 17:12:24 +03:00
setLastError(ErrorCode::OpenVpnManagementServerError);
return lastError();
2020-12-26 15:03:51 +03:00
}
setConnectionState(ConnectionState::Connecting);
2021-02-02 01:47:40 +03:00
m_openVpnProcess = IpcClient::CreatePrivilegedProcess();
2021-02-03 15:42:36 +03:00
if (!m_openVpnProcess) {
//qWarning() << "IpcProcess replica is not created!";
setLastError(ErrorCode::AmneziaServiceConnectionFailed);
2021-02-03 15:42:36 +03:00
return ErrorCode::AmneziaServiceConnectionFailed;
}
m_openVpnProcess->waitForSource(1000);
if (!m_openVpnProcess->isInitialized()) {
qWarning() << "IpcProcess replica is not connected!";
setLastError(ErrorCode::AmneziaServiceConnectionFailed);
2021-02-02 01:47:40 +03:00
return ErrorCode::AmneziaServiceConnectionFailed;
}
2021-02-03 15:42:36 +03:00
m_openVpnProcess->setProgram(openVpnExecPath());
2021-02-02 22:51:31 +03:00
QStringList arguments({"--config" , configPath(),
"--management", m_managementHost, QString::number(m_managementPort),
"--management-client",
2021-06-02 00:54:47 +03:00
"--log", vpnLogFileNamePath
2021-02-02 22:51:31 +03:00
});
2021-02-03 15:42:36 +03:00
m_openVpnProcess->setArguments(arguments);
2021-02-02 22:51:31 +03:00
qDebug() << arguments.join(" ");
2021-02-03 15:42:36 +03:00
connect(m_openVpnProcess.data(), &IpcProcessInterfaceReplica::errorOccurred, [&](QProcess::ProcessError error) {
2021-02-02 22:51:31 +03:00
qDebug() << "IpcProcessInterfaceReplica errorOccurred" << error;
});
2021-02-03 15:42:36 +03:00
connect(m_openVpnProcess.data(), &IpcProcessInterfaceReplica::stateChanged, [&](QProcess::ProcessState newState) {
2021-02-02 22:51:31 +03:00
qDebug() << "IpcProcessInterfaceReplica stateChanged" << newState;
});
connect(m_openVpnProcess.data(), &IpcProcessInterfaceReplica::finished, this, [&]() {
setConnectionState(ConnectionState::Disconnected);
});
2021-02-03 15:42:36 +03:00
m_openVpnProcess->start();
2021-02-02 01:47:40 +03:00
2021-02-02 22:51:31 +03:00
//startTimeoutTimer();
2020-12-26 15:03:51 +03:00
2021-01-06 17:12:24 +03:00
return ErrorCode::NoError;
2021-09-15 08:03:28 -07:00
#else
return ErrorCode::NotImplementedError;
#endif
2020-12-26 15:03:51 +03:00
}
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();
2021-06-01 18:18:09 +03:00
setConnectionState(VpnProtocol::Connected);
2020-12-26 15:03:51 +03:00
continue;
} else if (line.contains("EXITING,SIGTER")) {
//openVpnStateSigTermHandler();
2021-06-01 18:18:09 +03:00
setConnectionState(VpnProtocol::Disconnecting);
2020-12-26 15:03:51 +03:00
continue;
2021-01-09 19:55:16 +03:00
} else if (line.contains("RECONNECTING")) {
2021-06-01 18:18:09 +03:00
setConnectionState(VpnProtocol::Reconnecting);
2021-01-09 19:55:16 +03:00
continue;
2020-12-26 15:03:51 +03:00
}
}
if (line.contains("ROUTE_GATEWAY")) {
updateRouteGateway(line);
}
2021-02-21 09:44:53 -08:00
if (line.contains("PUSH: Received control message")) {
updateVpnGateway(line);
}
2021-01-08 16:51:58 +03:00
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;
2021-01-08 16:51:58 +03:00
}
2020-12-26 15:03:51 +03:00
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);
}
}
}
2021-02-21 09:44:53 -08:00
void OpenVpnProtocol::updateVpnGateway(const QString &line)
{
2021-02-21 09:44:53 -08:00
// 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'
2021-02-21 09:44:53 -08:00
QStringList params = line.split(",");
for (const QString &l : params) {
if (l.contains("ifconfig")) {
if (l.split(" ").size() == 3) {
2021-06-03 20:23:44 +03:00
m_vpnLocalAddress = l.split(" ").at(1);
2021-02-21 09:44:53 -08:00
m_vpnGateway = l.split(" ").at(2);
2021-06-03 20:23:44 +03:00
qDebug() << QString("Set vpn local address %1, gw %2").arg(m_vpnLocalAddress).arg(vpnGateway());
2021-02-21 09:44:53 -08:00
}
}
}
}