Files
amnezia-client/client/vpnconnection.cpp
T

451 lines
14 KiB
C++
Raw Normal View History

2022-12-12 16:16:12 +04:00
#include "qtimer.h"
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>
2022-01-22 20:00:06 +03:00
#include <QHostInfo>
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-10-02 21:56:47 +03:00
#include <configurators/vpn_configurator.h>
2021-01-06 17:12:24 +03:00
#include <core/servercontroller.h>
2022-02-22 02:08:57 +03:00
#ifdef AMNEZIA_DESKTOP
#include "ipc.h"
#include "core/ipcclient.h"
2021-06-12 11:59:36 +03:00
#include <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
2022-12-27 15:37:58 +03:00
#include "../../platforms/android/android_controller.h"
2021-09-30 18:16:41 +03:00
#endif
#ifdef Q_OS_IOS
#include <protocols/ios_vpnprotocol.h>
#endif
#include "utilities.h"
2020-12-26 15:03:51 +03:00
#include "vpnconnection.h"
2022-08-25 17:35:28 +03:00
VpnConnection::VpnConnection(std::shared_ptr<Settings> settings,
std::shared_ptr<VpnConfigurator> configurator, QObject* parent) : QObject(parent),
2022-08-25 17:35:28 +03:00
m_settings(settings),
2023-08-08 19:02:41 -07:00
m_configurator(configurator)
2021-12-20 15:43:36 +03:00
{
2023-08-08 19:02:41 -07:00
m_checkTimer.setInterval(1000);
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()
{
if (m_vpnProtocol != nullptr) {
m_vpnProtocol->deleteLater();
m_vpnProtocol.clear();
}
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
}
2023-05-14 21:11:19 +08:00
void VpnConnection::onConnectionStateChanged(Vpn::ConnectionState state)
2020-12-26 15:03:51 +03:00
{
2022-12-12 16:16:12 +04:00
2022-02-22 02:08:57 +03:00
#ifdef AMNEZIA_DESKTOP
2021-02-18 15:00:41 +03:00
if (IpcClient::Interface()) {
2023-05-14 21:11:19 +08:00
if (state == Vpn::ConnectionState::Connected){
2021-12-15 14:53:07 +03:00
IpcClient::Interface()->resetIpStack();
2021-02-18 15:00:41 +03:00
IpcClient::Interface()->flushDns();
2022-08-25 12:47:02 +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();
}
2022-02-01 19:48:59 +03:00
QString dns1 = m_vpnConfiguration.value(config_key::dns1).toString();
QString dns2 = m_vpnConfiguration.value(config_key::dns1).toString();
2021-06-01 18:18:09 +03:00
IpcClient::Interface()->routeAddList(m_vpnProtocol->vpnGateway(),
2022-02-01 19:48:59 +03:00
QStringList() << dns1 << dns2);
2021-01-26 15:01:15 +03:00
2022-08-25 12:47:02 +03:00
if (m_settings->routeMode() == Settings::VpnOnlyForwardSites) {
2022-01-22 20:00:06 +03:00
QTimer::singleShot(1000, m_vpnProtocol.data(), [this](){
2022-08-25 12:47:02 +03:00
addSitesRoutes(m_vpnProtocol->vpnGateway(), m_settings->routeMode());
2022-01-22 20:00:06 +03:00
});
2021-06-01 18:18:09 +03:00
}
2022-08-25 12:47:02 +03:00
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());
2022-08-25 12:47:02 +03:00
addSitesRoutes(m_vpnProtocol->routeGateway(), m_settings->routeMode());
2021-02-18 15:00:41 +03:00
}
2021-06-01 18:18:09 +03:00
2022-01-22 20:00:06 +03:00
2021-02-18 15:00:41 +03:00
}
2023-05-14 21:11:19 +08:00
else if (state == Vpn::ConnectionState::Error) {
2021-02-18 15:00:41 +03:00
IpcClient::Interface()->flushDns();
2021-01-26 15:01:15 +03:00
2022-08-25 12:47:02 +03:00
if (m_settings->routeMode() == Settings::VpnOnlyForwardSites) {
2021-02-18 15:00:41 +03:00
IpcClient::Interface()->clearSavedRoutes();
}
}
}
2022-02-22 02:08:57 +03:00
#endif
2022-12-12 16:16:12 +04:00
#ifdef Q_OS_IOS
2023-08-13 11:28:32 +05:00
if (state == Vpn::ConnectionState::Connected) {
2023-08-08 19:02:41 -07:00
m_checkTimer.start();
2022-12-12 14:41:54 +01: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
emit connectionStateChanged(state);
}
2021-06-12 11:59:36 +03:00
const QString &VpnConnection::remoteAddress() const
{
return m_remoteAddress;
}
2022-01-22 20:00:06 +03:00
void VpnConnection::addSitesRoutes(const QString &gw, Settings::RouteMode mode)
{
2022-02-22 02:08:57 +03:00
#ifdef AMNEZIA_DESKTOP
2022-01-22 20:00:06 +03:00
QStringList ips;
QStringList sites;
2022-08-25 12:47:02 +03:00
const QVariantMap &m = m_settings->vpnSites(mode);
2022-01-22 20:00:06 +03:00
for (auto i = m.constBegin(); i != m.constEnd(); ++i) {
if (Utils::checkIpSubnetFormat(i.key())) {
ips.append(i.key());
}
else {
if (Utils::checkIpSubnetFormat(i.value().toString())) {
ips.append(i.value().toString());
}
sites.append(i.key());
}
}
ips.removeDuplicates();
// add all IPs immediately
IpcClient::Interface()->routeAddList(gw, ips);
// re-resolve domains
for (const QString &site: sites) {
const auto &cbResolv = [this, site, gw, mode, ips](const QHostInfo &hostInfo){
const QList<QHostAddress> &addresses = hostInfo.addresses();
QString ipv4Addr;
for (const QHostAddress &addr: hostInfo.addresses()) {
if (addr.protocol() == QAbstractSocket::NetworkLayerProtocol::IPv4Protocol) {
const QString &ip = addr.toString();
//qDebug() << "VpnConnection::addSitesRoutes updating site" << site << ip;
if (!ips.contains(ip)) {
IpcClient::Interface()->routeAddList(gw, QStringList() << ip);
2022-08-25 12:47:02 +03:00
m_settings->addVpnSite(mode, site, ip);
2022-01-22 20:00:06 +03:00
}
flushDns();
break;
}
}
};
QHostInfo::lookupHost(site, this, cbResolv);
}
2022-02-22 02:08:57 +03:00
#endif
2022-01-22 20:00:06 +03:00
}
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)
{
2022-02-22 02:08:57 +03:00
#ifdef AMNEZIA_DESKTOP
2023-05-14 21:11:19 +08:00
if (connectionState() == Vpn::ConnectionState::Connected && IpcClient::Interface()) {
2022-08-25 12:47:02 +03:00
if (m_settings->routeMode() == Settings::VpnOnlyForwardSites) {
2021-06-01 18:18:09 +03:00
IpcClient::Interface()->routeAddList(m_vpnProtocol->vpnGateway(), ips);
}
2022-08-25 12:47:02 +03:00
else if (m_settings->routeMode() == Settings::VpnAllExceptSites) {
2021-06-01 18:18:09 +03:00
IpcClient::Interface()->routeAddList(m_vpnProtocol->routeGateway(), ips);
}
}
2022-02-22 02:08:57 +03:00
#endif
2021-06-01 18:18:09 +03:00
}
void VpnConnection::deleteRoutes(const QStringList &ips)
{
2022-02-22 02:08:57 +03:00
#ifdef AMNEZIA_DESKTOP
2023-05-14 21:11:19 +08:00
if (connectionState() == Vpn::ConnectionState::Connected && IpcClient::Interface()) {
2022-08-25 12:47:02 +03:00
if (m_settings->routeMode() == Settings::VpnOnlyForwardSites) {
2021-06-01 18:18:09 +03:00
IpcClient::Interface()->routeDeleteList(vpnProtocol()->vpnGateway(), ips);
}
2022-08-25 12:47:02 +03:00
else if (m_settings->routeMode() == Settings::VpnAllExceptSites) {
2021-06-01 18:18:09 +03:00
IpcClient::Interface()->routeDeleteList(m_vpnProtocol->routeGateway(), ips);
}
}
2022-02-22 02:08:57 +03:00
#endif
2021-06-01 18:18:09 +03:00
}
void VpnConnection::flushDns()
{
2022-02-22 02:08:57 +03:00
#ifdef AMNEZIA_DESKTOP
2021-06-01 18:18:09 +03:00
if (IpcClient::Interface()) IpcClient::Interface()->flushDns();
2022-02-22 02:08:57 +03:00
#endif
2021-06-01 18:18:09 +03:00
}
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();
}
QMap<Proto, QString> VpnConnection::getLastVpnConfig(const QJsonObject &containerConfig)
2021-05-07 23:28:37 +03:00
{
QMap<Proto, QString> configs;
for (Proto proto: ProtocolProps::allProtocols()) {
2021-05-07 23:28:37 +03:00
2021-09-20 21:51:28 +03:00
QString cfg = containerConfig.value(ProtocolProps::protoToString(proto)).toObject().value(config_key::last_config).toString();
2021-05-07 23:28:37 +03:00
if (!cfg.isEmpty()) configs.insert(proto, cfg);
}
return configs;
}
QString VpnConnection::createVpnConfigurationForProto(int serverIndex,
const ServerCredentials &credentials, DockerContainer container, const QJsonObject &containerConfig, Proto proto,
2021-05-07 23:28:37 +03:00
ErrorCode *errorCode)
{
2022-02-01 19:48:59 +03:00
QMap<Proto, QString> lastVpnConfig = getLastVpnConfig(containerConfig);
2021-05-07 23:28:37 +03:00
QString configData;
if (lastVpnConfig.contains(proto)) {
2021-05-14 04:27:30 -07:00
configData = lastVpnConfig.value(proto);
2022-08-25 17:35:28 +03:00
configData = m_configurator->processConfigWithLocalSettings(serverIndex, container, proto, configData);
2021-05-07 23:28:37 +03:00
}
else {
2022-08-25 17:35:28 +03:00
configData = m_configurator->genVpnProtocolConfig(credentials,
2023-08-08 19:02:41 -07:00
container, containerConfig, proto, errorCode);
2021-10-02 21:56:47 +03:00
2023-08-08 19:02:41 -07:00
if (errorCode && *errorCode) {
2021-05-07 23:28:37 +03:00
return "";
}
2023-08-08 19:02:41 -07:00
QString configDataBeforeLocalProcessing = configData;
configData = m_configurator->processConfigWithLocalSettings(serverIndex, container, proto, configData);
2021-05-07 23:28:37 +03:00
if (serverIndex >= 0) {
2021-05-10 02:33:31 +03:00
qDebug() << "VpnConnection::createVpnConfiguration: saving config for server #" << serverIndex << container << proto;
2022-08-25 12:47:02 +03:00
QJsonObject protoObject = m_settings->protocolConfig(serverIndex, container, proto);
2021-10-02 21:56:47 +03:00
protoObject.insert(config_key::last_config, configDataBeforeLocalProcessing);
2022-08-25 12:47:02 +03:00
m_settings->setProtocolConfig(serverIndex, container, proto, protoObject);
2021-05-07 23:28:37 +03:00
}
}
return configData;
}
2021-10-04 21:13:07 +03:00
QJsonObject VpnConnection::createVpnConfiguration(int serverIndex,
const ServerCredentials &credentials, DockerContainer container,
const QJsonObject &containerConfig, ErrorCode *errorCode)
2020-12-26 15:03:51 +03:00
{
2021-10-04 21:13:07 +03:00
QJsonObject vpnConfiguration;
2021-05-07 23:28:37 +03:00
for (ProtocolEnumNS::Proto proto : ContainerProps::protocolsForContainer(container)) {
2021-10-05 12:22:13 +03:00
QJsonObject vpnConfigData = QJsonDocument::fromJson(
2021-05-07 23:28:37 +03:00
createVpnConfigurationForProto(
2023-08-08 19:02:41 -07:00
serverIndex, credentials, container, containerConfig, proto, errorCode).toUtf8()).
2021-10-05 12:22:13 +03:00
object();
2021-05-07 23:28:37 +03:00
2023-08-08 19:02:41 -07:00
if (errorCode && *errorCode) {
2021-10-04 21:13:07 +03:00
return {};
2021-01-06 17:12:24 +03:00
}
2021-03-13 14:16:24 +03:00
2021-10-05 12:22:13 +03:00
vpnConfiguration.insert(ProtocolProps::key_proto_config_data(proto), vpnConfigData);
2021-04-04 23:12:36 +03:00
}
Proto proto = ContainerProps::defaultProtocol(container);
vpnConfiguration[config_key::vpnproto] = ProtocolProps::protoToString(proto);
2021-06-12 11:59:36 +03:00
2022-08-25 17:35:28 +03:00
auto dns = m_configurator->getDnsForConfig(serverIndex);
2022-02-01 19:48:59 +03:00
vpnConfiguration[config_key::dns1] = dns.first;
vpnConfiguration[config_key::dns2] = dns.second;
2021-10-04 21:13:07 +03:00
return vpnConfiguration;
2021-01-06 17:12:24 +03:00
}
2021-10-26 12:59:20 +03:00
void VpnConnection::connectToVpn(int serverIndex,
2021-06-01 18:18:09 +03:00
const ServerCredentials &credentials, DockerContainer container, const QJsonObject &containerConfig)
2021-01-06 17:12:24 +03:00
{
2023-04-11 09:50:44 -04:00
qDebug() << QString("ConnectToVpn, Server index is %1, container is %2, route mode is")
2022-08-25 12:47:02 +03:00
.arg(serverIndex).arg(ContainerProps::containerToString(container)) << m_settings->routeMode();
2021-10-26 12:59:20 +03:00
#if !defined (Q_OS_ANDROID) && !defined (Q_OS_IOS)
2021-10-26 12:59:20 +03:00
if (!m_IpcClient) {
2021-12-20 15:43:36 +03:00
m_IpcClient = new IpcClient(this);
2021-10-26 12:59:20 +03:00
}
if (!m_IpcClient->isSocketConnected()) {
if (!IpcClient::init(m_IpcClient)) {
2023-04-11 09:50:44 -04:00
qWarning() << "Error occurred when init IPC client";
2021-10-26 12:59:20 +03:00
emit serviceIsNotReady();
2023-05-14 21:11:19 +08:00
emit connectionStateChanged(Vpn::ConnectionState::Error);
2021-10-26 12:59:20 +03:00
return;
}
}
#endif
2021-06-12 11:59:36 +03:00
m_remoteAddress = credentials.hostName;
2023-05-14 21:11:19 +08:00
emit connectionStateChanged(Vpn::ConnectionState::Connecting);
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-10-04 21:13:07 +03:00
ErrorCode e = ErrorCode::NoError;
m_vpnConfiguration = createVpnConfiguration(serverIndex, credentials, container, containerConfig, &e);
2021-10-04 19:07:49 +03:00
if (e) {
2023-05-14 21:11:19 +08:00
emit connectionStateChanged(Vpn::ConnectionState::Error);
2021-10-26 12:59:20 +03:00
return;
2021-01-06 17:12:24 +03:00
}
#if !defined (Q_OS_ANDROID) && !defined (Q_OS_IOS)
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) {
emit connectionStateChanged(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();
#elif defined Q_OS_ANDROID
androidVpnProtocol = createDefaultAndroidVpnProtocol(container);
createAndroidConnections(container);
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
Proto proto = ContainerProps::defaultProtocol(container);
2023-08-08 19:02:41 -07:00
auto iosVpnProtocol = new IOSVpnProtocol(proto, m_vpnConfiguration);
if (!iosVpnProtocol->initialize()) {
qDebug() << QString("Init failed") ;
2023-08-13 11:28:32 +05:00
emit Vpn::ConnectionState::Error;
2023-08-08 19:02:41 -07:00
iosVpnProtocol->deleteLater();
return;
}
2023-08-08 19:02:41 -07:00
connect(&m_checkTimer, &QTimer::timeout, iosVpnProtocol, &IOSVpnProtocol::checkStatus);
m_vpnProtocol.reset(iosVpnProtocol);
2023-08-08 19:02:41 -07:00
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
2021-10-26 12:59:20 +03:00
e = m_vpnProtocol.data()->start();
if (e) emit connectionStateChanged(Vpn::ConnectionState::Error);
2020-12-26 15:03:51 +03:00
}
2023-03-15 17:46:22 +03:00
void VpnConnection::createProtocolConnections() {
connect(m_vpnProtocol.data(), &VpnProtocol::protocolError, this, &VpnConnection::vpnProtocolError);
2023-05-14 21:11:19 +08:00
connect(m_vpnProtocol.data(), SIGNAL(connectionStateChanged(Vpn::ConnectionState)), this, SLOT(onConnectionStateChanged(Vpn::ConnectionState)));
2023-03-15 17:46:22 +03:00
connect(m_vpnProtocol.data(), SIGNAL(bytesChanged(quint64, quint64)), this, SLOT(onBytesChanged(quint64, quint64)));
}
#ifdef Q_OS_ANDROID
void VpnConnection::restoreConnection() {
createAndroidConnections();
m_vpnProtocol.reset(androidVpnProtocol);
createProtocolConnections();
}
void VpnConnection::createAndroidConnections()
{
int serverIndex = m_settings->defaultServerIndex();
DockerContainer container = m_settings->defaultContainer(serverIndex);
createAndroidConnections(container);
}
void VpnConnection::createAndroidConnections(DockerContainer container)
{
androidVpnProtocol = createDefaultAndroidVpnProtocol(container);
connect(AndroidController::instance(), &AndroidController::connectionStateChanged, androidVpnProtocol, &AndroidVpnProtocol::setConnectionState);
connect(AndroidController::instance(), &AndroidController::statusUpdated, androidVpnProtocol, &AndroidVpnProtocol::connectionDataUpdated);
}
AndroidVpnProtocol* VpnConnection::createDefaultAndroidVpnProtocol(DockerContainer container)
{
Proto proto = ContainerProps::defaultProtocol(container);
AndroidVpnProtocol *androidVpnProtocol = new AndroidVpnProtocol(proto, m_vpnConfiguration);
return androidVpnProtocol;
}
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
}
void VpnConnection::disconnectFromVpn()
{
2022-02-22 02:08:57 +03:00
#ifdef AMNEZIA_DESKTOP
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
2021-06-19 16:38:35 +03:00
QRemoteObjectPendingReply<bool> response = IpcClient::Interface()->clearSavedRoutes();
response.waitForFinished(1000);
2021-02-24 21:58:32 +03:00
}
2022-02-22 02:08:57 +03:00
#endif
2020-12-26 15:03:51 +03:00
if (!m_vpnProtocol.data()) {
2023-05-14 21:11:19 +08:00
emit connectionStateChanged(Vpn::ConnectionState::Disconnected);
2022-02-22 02:08:57 +03:00
#ifdef Q_OS_ANDROID
AndroidController::instance()->stop();
#endif
2020-12-26 15:03:51 +03:00
return;
}
2023-08-08 16:41:00 -07:00
if (m_vpnProtocol) {
m_vpnProtocol->deleteLater();
}
m_vpnProtocol = nullptr;
2020-12-26 15:03:51 +03:00
}
2023-05-14 21:11:19 +08:00
Vpn::ConnectionState VpnConnection::connectionState()
2021-01-15 23:36:35 +03:00
{
2023-05-14 21:11:19 +08:00
if (!m_vpnProtocol) return Vpn::ConnectionState::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
}