mirror of
https://github.com/amnezia-vpn/amnezia-client.git
synced 2026-06-22 02:01:08 +07:00
Compare commits
1 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 277fdea3d4 |
@@ -10,7 +10,7 @@ env:
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
Build-Linux-Ubuntu:
|
Build-Linux-Ubuntu:
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-20.04
|
||||||
|
|
||||||
env:
|
env:
|
||||||
QT_VERSION: 6.6.2
|
QT_VERSION: 6.6.2
|
||||||
@@ -190,7 +190,7 @@ jobs:
|
|||||||
- name: 'Install go'
|
- name: 'Install go'
|
||||||
uses: actions/setup-go@v5
|
uses: actions/setup-go@v5
|
||||||
with:
|
with:
|
||||||
go-version: '1.24'
|
go-version: '1.22.1'
|
||||||
cache: false
|
cache: false
|
||||||
|
|
||||||
- name: 'Setup gomobile'
|
- name: 'Setup gomobile'
|
||||||
|
|||||||
@@ -1,41 +1,64 @@
|
|||||||
name: 'Upload a new version'
|
name: 'Upload a new version'
|
||||||
|
|
||||||
on:
|
on:
|
||||||
workflow_dispatch:
|
push:
|
||||||
inputs:
|
tags:
|
||||||
RELEASE_VERSION:
|
- '[0-9]+.[0-9]+.[0-9]+.[0-9]+'
|
||||||
description: 'Release version (e.g. 1.2.3.4)'
|
|
||||||
required: true
|
|
||||||
type: string
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
Upload-S3:
|
upload:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
|
name: upload
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout CMakeLists.txt
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
ref: ${{ inputs.RELEASE_VERSION }}
|
ref: ${{ github.ref_name }}
|
||||||
sparse-checkout: |
|
sparse-checkout: |
|
||||||
CMakeLists.txt
|
CMakeLists.txt
|
||||||
deploy/deploy_s3.sh
|
|
||||||
sparse-checkout-cone-mode: false
|
sparse-checkout-cone-mode: false
|
||||||
|
|
||||||
- name: Verify git tag
|
- name: Verify git tag
|
||||||
run: |
|
run: |
|
||||||
TAG_NAME=${{ inputs.RELEASE_VERSION }}
|
GIT_TAG=${{ github.ref_name }}
|
||||||
CMAKE_TAG=$(grep 'project.*VERSION' CMakeLists.txt | sed -E 's/.* ([0-9]+.[0-9]+.[0-9]+.[0-9]+)$/\1/')
|
CMAKE_TAG=$(grep 'project.*VERSION' CMakeLists.txt | sed -E 's/.* ([0-9]+.[0-9]+.[0-9]+.[0-9]+)$/\1/')
|
||||||
if [[ "$TAG_NAME" == "$CMAKE_TAG" ]]; then
|
|
||||||
echo "Git tag ($TAG_NAME) matches CMakeLists.txt version ($CMAKE_TAG)."
|
if [[ "$GIT_TAG" == "$CMAKE_TAG" ]]; then
|
||||||
|
echo "Git tag ($GIT_TAG) and version in CMakeLists.txt ($CMAKE_TAG) are the same. Continuing..."
|
||||||
else
|
else
|
||||||
echo "::error::Mismatch: Git tag ($TAG_NAME) != CMakeLists.txt version ($CMAKE_TAG). Exiting with error..."
|
echo "Git tag ($GIT_TAG) and version in CMakeLists.txt ($CMAKE_TAG) are not the same! Cancelling..."
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
- name: Setup Rclone
|
- name: Download artifacts from the "${{ github.ref_name }}" tag
|
||||||
uses: AnimMouse/setup-rclone@v1
|
uses: robinraju/release-downloader@v1.8
|
||||||
with:
|
with:
|
||||||
rclone_config: ${{ secrets.RCLONE_CONFIG }}
|
tag: ${{ github.ref_name }}
|
||||||
|
fileName: "AmneziaVPN_(Linux_|)${{ github.ref_name }}*"
|
||||||
|
out-file-path: ${{ github.ref_name }}
|
||||||
|
|
||||||
- name: Send dist to S3
|
- name: Upload beta version
|
||||||
run: bash deploy/deploy_s3.sh ${{ inputs.RELEASE_VERSION }}
|
uses: jakejarvis/s3-sync-action@master
|
||||||
|
if: contains(github.event.base_ref, 'dev')
|
||||||
|
with:
|
||||||
|
args: --include "AmneziaVPN*" --delete
|
||||||
|
env:
|
||||||
|
AWS_S3_BUCKET: updates
|
||||||
|
AWS_ACCESS_KEY_ID: ${{ secrets.CF_R2_ACCESS_KEY_ID }}
|
||||||
|
AWS_SECRET_ACCESS_KEY: ${{ secrets.CF_R2_SECRET_ACCESS_KEY }}
|
||||||
|
AWS_S3_ENDPOINT: https://${{ vars.CF_ACCOUNT_ID }}.r2.cloudflarestorage.com
|
||||||
|
SOURCE_DIR: ${{ github.ref_name }}
|
||||||
|
DEST_DIR: beta/${{ github.ref_name }}
|
||||||
|
|
||||||
|
- name: Upload stable version
|
||||||
|
uses: jakejarvis/s3-sync-action@master
|
||||||
|
if: contains(github.event.base_ref, 'master')
|
||||||
|
with:
|
||||||
|
args: --include "AmneziaVPN*" --delete
|
||||||
|
env:
|
||||||
|
AWS_S3_BUCKET: updates
|
||||||
|
AWS_ACCESS_KEY_ID: ${{ secrets.CF_R2_ACCESS_KEY_ID }}
|
||||||
|
AWS_SECRET_ACCESS_KEY: ${{ secrets.CF_R2_SECRET_ACCESS_KEY }}
|
||||||
|
AWS_S3_ENDPOINT: https://${{ vars.CF_ACCOUNT_ID }}.r2.cloudflarestorage.com
|
||||||
|
SOURCE_DIR: ${{ github.ref_name }}
|
||||||
|
DEST_DIR: stable/${{ github.ref_name }}
|
||||||
|
|||||||
+2
-2
@@ -2,7 +2,7 @@ cmake_minimum_required(VERSION 3.25.0 FATAL_ERROR)
|
|||||||
|
|
||||||
set(PROJECT AmneziaVPN)
|
set(PROJECT AmneziaVPN)
|
||||||
|
|
||||||
project(${PROJECT} VERSION 4.8.6.0
|
project(${PROJECT} VERSION 4.8.4.3
|
||||||
DESCRIPTION "AmneziaVPN"
|
DESCRIPTION "AmneziaVPN"
|
||||||
HOMEPAGE_URL "https://amnezia.org/"
|
HOMEPAGE_URL "https://amnezia.org/"
|
||||||
)
|
)
|
||||||
@@ -11,7 +11,7 @@ string(TIMESTAMP CURRENT_DATE "%Y-%m-%d")
|
|||||||
set(RELEASE_DATE "${CURRENT_DATE}")
|
set(RELEASE_DATE "${CURRENT_DATE}")
|
||||||
|
|
||||||
set(APP_MAJOR_VERSION ${CMAKE_PROJECT_VERSION_MAJOR}.${CMAKE_PROJECT_VERSION_MINOR}.${CMAKE_PROJECT_VERSION_PATCH})
|
set(APP_MAJOR_VERSION ${CMAKE_PROJECT_VERSION_MAJOR}.${CMAKE_PROJECT_VERSION_MINOR}.${CMAKE_PROJECT_VERSION_PATCH})
|
||||||
set(APP_ANDROID_VERSION_CODE 2083)
|
set(APP_ANDROID_VERSION_CODE 2080)
|
||||||
|
|
||||||
if(${CMAKE_SYSTEM_NAME} STREQUAL "Linux")
|
if(${CMAKE_SYSTEM_NAME} STREQUAL "Linux")
|
||||||
set(MZ_PLATFORM_NAME "linux")
|
set(MZ_PLATFORM_NAME "linux")
|
||||||
|
|||||||
+10
-10
@@ -6,11 +6,11 @@
|
|||||||
[](https://gitpod.io/#https://github.com/amnezia-vpn/amnezia-client)
|
[](https://gitpod.io/#https://github.com/amnezia-vpn/amnezia-client)
|
||||||
|
|
||||||
### [English](https://github.com/amnezia-vpn/amnezia-client/blob/dev/README.md) | Русский
|
### [English](https://github.com/amnezia-vpn/amnezia-client/blob/dev/README.md) | Русский
|
||||||
[AmneziaVPN](https://amnezia.org) — это open source VPN-клиент, ключевая особенность которого заключается в возможности развернуть собственный VPN на вашем сервере.
|
[AmneziaVPN](https://amnezia.org) — это open sourse VPN-клиент, ключевая особенность которого заключается в возможности развернуть собственный VPN на вашем сервере.
|
||||||
|
|
||||||
[](https://amnezia.org)
|
[](https://amnezia.org)
|
||||||
|
|
||||||
### [Сайт](https://amnezia.org) | [Зеркало сайта](https://storage.googleapis.com/amnezia/amnezia.org) | [Документация](https://docs.amnezia.org) | [Решение проблем](https://docs.amnezia.org/troubleshooting)
|
### [Сайт](https://amnezia.org) | [Зеркало на сайт](https://storage.googleapis.com/amnezia/amnezia.org) | [Документация](https://docs.amnezia.org) | [Решение проблем](https://docs.amnezia.org/troubleshooting)
|
||||||
|
|
||||||
> [!TIP]
|
> [!TIP]
|
||||||
> Если [сайт Amnezia](https://amnezia.org) заблокирован в вашем регионе, вы можете воспользоваться [ссылкой на зеркало](https://storage.googleapis.com/amnezia/amnezia.org).
|
> Если [сайт Amnezia](https://amnezia.org) заблокирован в вашем регионе, вы можете воспользоваться [ссылкой на зеркало](https://storage.googleapis.com/amnezia/amnezia.org).
|
||||||
@@ -30,7 +30,7 @@
|
|||||||
- Классические VPN-протоколы: OpenVPN, WireGuard и IKEv2.
|
- Классические VPN-протоколы: OpenVPN, WireGuard и IKEv2.
|
||||||
- Протоколы с маскировкой трафика (обфускацией): OpenVPN с плагином [Cloak](https://github.com/cbeuw/Cloak), Shadowsocks (OpenVPN over Shadowsocks), [AmneziaWG](https://docs.amnezia.org/documentation/amnezia-wg/) and XRay.
|
- Протоколы с маскировкой трафика (обфускацией): OpenVPN с плагином [Cloak](https://github.com/cbeuw/Cloak), Shadowsocks (OpenVPN over Shadowsocks), [AmneziaWG](https://docs.amnezia.org/documentation/amnezia-wg/) and XRay.
|
||||||
- Поддержка Split Tunneling — добавляйте любые сайты или приложения в список, чтобы включить VPN только для них.
|
- Поддержка Split Tunneling — добавляйте любые сайты или приложения в список, чтобы включить VPN только для них.
|
||||||
- Поддерживает платформы: Windows, macOS, Linux, Android, iOS.
|
- Поддерживает платформы: Windows, MacOS, Linux, Android, iOS.
|
||||||
- Поддержка конфигурации протокола AmneziaWG на [бета-прошивке Keenetic](https://docs.keenetic.com/ua/air/kn-1611/en/6319-latest-development-release.html#UUID-186c4108-5afd-c10b-f38a-cdff6c17fab3_section-idm33192196168192-improved).
|
- Поддержка конфигурации протокола AmneziaWG на [бета-прошивке Keenetic](https://docs.keenetic.com/ua/air/kn-1611/en/6319-latest-development-release.html#UUID-186c4108-5afd-c10b-f38a-cdff6c17fab3_section-idm33192196168192-improved).
|
||||||
|
|
||||||
## Ссылки
|
## Ссылки
|
||||||
@@ -38,10 +38,10 @@
|
|||||||
- [https://amnezia.org](https://amnezia.org) - Веб-сайт проекта | [Альтернативная ссылка (зеркало)](https://storage.googleapis.com/kldscp/amnezia.org)
|
- [https://amnezia.org](https://amnezia.org) - Веб-сайт проекта | [Альтернативная ссылка (зеркало)](https://storage.googleapis.com/kldscp/amnezia.org)
|
||||||
- [https://docs.amnezia.org](https://docs.amnezia.org) - Документация
|
- [https://docs.amnezia.org](https://docs.amnezia.org) - Документация
|
||||||
- [https://www.reddit.com/r/AmneziaVPN](https://www.reddit.com/r/AmneziaVPN) - Reddit
|
- [https://www.reddit.com/r/AmneziaVPN](https://www.reddit.com/r/AmneziaVPN) - Reddit
|
||||||
- [https://t.me/amnezia_vpn_en](https://t.me/amnezia_vpn_en) - Канал поддержки в Telegram (Английский)
|
- [https://t.me/amnezia_vpn_en](https://t.me/amnezia_vpn_en) - Канал поддржки в Telegram (Английский)
|
||||||
- [https://t.me/amnezia_vpn_ir](https://t.me/amnezia_vpn_ir) - Канал поддержки в Telegram (Фарси)
|
- [https://t.me/amnezia_vpn_ir](https://t.me/amnezia_vpn_ir) - Канал поддржки в Telegram (Фарси)
|
||||||
- [https://t.me/amnezia_vpn_mm](https://t.me/amnezia_vpn_mm) - Канал поддержки в Telegram (Мьянма)
|
- [https://t.me/amnezia_vpn_mm](https://t.me/amnezia_vpn_mm) - Канал поддржки в Telegram (Мьянма)
|
||||||
- [https://t.me/amnezia_vpn](https://t.me/amnezia_vpn) - Канал поддержки в Telegram (Русский)
|
- [https://t.me/amnezia_vpn](https://t.me/amnezia_vpn) - Канал поддржки в Telegram (Русский)
|
||||||
- [https://vpnpay.io/en/amnezia-premium/](https://vpnpay.io/en/amnezia-premium/) - Amnezia Premium | [Зеркало](https://storage.googleapis.com/kldscp/vpnpay.io/ru/amnezia-premium\)
|
- [https://vpnpay.io/en/amnezia-premium/](https://vpnpay.io/en/amnezia-premium/) - Amnezia Premium | [Зеркало](https://storage.googleapis.com/kldscp/vpnpay.io/ru/amnezia-premium\)
|
||||||
|
|
||||||
## Технологии
|
## Технологии
|
||||||
@@ -80,8 +80,8 @@ git submodule update --init --recursive
|
|||||||
Проверьте папку deploy для скриптов сборки.
|
Проверьте папку deploy для скриптов сборки.
|
||||||
|
|
||||||
### Как собрать iOS-приложение из исходного кода на MacOS
|
### Как собрать iOS-приложение из исходного кода на MacOS
|
||||||
1. Убедитесь, что у вас установлен Xcode версии 14 или выше.
|
1. Убедитесь, что у вас установлен XCode версии 14 или выше.
|
||||||
2. Для генерации проекта Xcode используется QT. Требуется версия QT 6.6.2. Установите QT для MacOS здесь или через QT Online Installer. Необходимые модули:
|
2. Для генерации проекта XCode используется QT. Требуется версия QT 6.6.2. Установите QT для MacOS здесь или через QT Online Installer. Необходимые модули:
|
||||||
- MacOS
|
- MacOS
|
||||||
- iOS
|
- iOS
|
||||||
- Модуль совместимости с Qt 5
|
- Модуль совместимости с Qt 5
|
||||||
@@ -117,7 +117,7 @@ $QT_IOS_BIN/qt-cmake . -B build-ios -GXcode -DQT_HOST_PATH=$QT_MACOS_ROOT_DIR
|
|||||||
export PATH=$(PATH):/path/to/GOPATH/bin
|
export PATH=$(PATH):/path/to/GOPATH/bin
|
||||||
```
|
```
|
||||||
|
|
||||||
6. Откройте проект в Xcode. Теперь вы можете тестировать, архивировать или публиковать приложение.
|
6. Откройте проект в XCode. Теперь вы можете тестировать, архивировать или публиковать приложение.
|
||||||
|
|
||||||
Если сборка завершится с ошибкой:
|
Если сборка завершится с ошибкой:
|
||||||
```
|
```
|
||||||
|
|||||||
+1
-1
Submodule client/3rd-prebuilt updated: efad1a5b5c...e555c78bcf
@@ -31,6 +31,10 @@ add_definitions(-DDEV_AGW_PUBLIC_KEY="$ENV{DEV_AGW_PUBLIC_KEY}")
|
|||||||
add_definitions(-DDEV_AGW_ENDPOINT="$ENV{DEV_AGW_ENDPOINT}")
|
add_definitions(-DDEV_AGW_ENDPOINT="$ENV{DEV_AGW_ENDPOINT}")
|
||||||
add_definitions(-DDEV_S3_ENDPOINT="$ENV{DEV_S3_ENDPOINT}")
|
add_definitions(-DDEV_S3_ENDPOINT="$ENV{DEV_S3_ENDPOINT}")
|
||||||
|
|
||||||
|
if(IOS)
|
||||||
|
set(PACKAGES ${PACKAGES} Multimedia)
|
||||||
|
endif()
|
||||||
|
|
||||||
if(WIN32 OR (APPLE AND NOT IOS) OR (LINUX AND NOT ANDROID))
|
if(WIN32 OR (APPLE AND NOT IOS) OR (LINUX AND NOT ANDROID))
|
||||||
set(PACKAGES ${PACKAGES} Widgets)
|
set(PACKAGES ${PACKAGES} Widgets)
|
||||||
endif()
|
endif()
|
||||||
@@ -44,6 +48,10 @@ set(LIBS ${LIBS}
|
|||||||
Qt6::Core5Compat Qt6::Concurrent
|
Qt6::Core5Compat Qt6::Concurrent
|
||||||
)
|
)
|
||||||
|
|
||||||
|
if(IOS)
|
||||||
|
set(LIBS ${LIBS} Qt6::Multimedia)
|
||||||
|
endif()
|
||||||
|
|
||||||
if(WIN32 OR (APPLE AND NOT IOS) OR (LINUX AND NOT ANDROID))
|
if(WIN32 OR (APPLE AND NOT IOS) OR (LINUX AND NOT ANDROID))
|
||||||
set(LIBS ${LIBS} Qt6::Widgets)
|
set(LIBS ${LIBS} Qt6::Widgets)
|
||||||
endif()
|
endif()
|
||||||
|
|||||||
@@ -13,10 +13,10 @@
|
|||||||
#include <QApplication>
|
#include <QApplication>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#include "core/networkUtilities.h"
|
|
||||||
#include "containers/containers_defs.h"
|
#include "containers/containers_defs.h"
|
||||||
#include "core/controllers/serverController.h"
|
#include "core/controllers/serverController.h"
|
||||||
#include "core/scripts_registry.h"
|
#include "core/scripts_registry.h"
|
||||||
|
#include "core/server_defs.h"
|
||||||
#include "settings.h"
|
#include "settings.h"
|
||||||
#include "utilities.h"
|
#include "utilities.h"
|
||||||
|
|
||||||
@@ -24,7 +24,6 @@
|
|||||||
#include <openssl/rsa.h>
|
#include <openssl/rsa.h>
|
||||||
#include <openssl/x509.h>
|
#include <openssl/x509.h>
|
||||||
|
|
||||||
|
|
||||||
OpenVpnConfigurator::OpenVpnConfigurator(std::shared_ptr<Settings> settings, const QSharedPointer<ServerController> &serverController,
|
OpenVpnConfigurator::OpenVpnConfigurator(std::shared_ptr<Settings> settings, const QSharedPointer<ServerController> &serverController,
|
||||||
QObject *parent)
|
QObject *parent)
|
||||||
: ConfiguratorBase(settings, serverController, parent)
|
: ConfiguratorBase(settings, serverController, parent)
|
||||||
@@ -120,14 +119,20 @@ QString OpenVpnConfigurator::processConfigWithLocalSettings(const QPair<QString,
|
|||||||
|
|
||||||
if (!m_settings->isSitesSplitTunnelingEnabled()) {
|
if (!m_settings->isSitesSplitTunnelingEnabled()) {
|
||||||
config.append("\nredirect-gateway def1 ipv6 bypass-dhcp\n");
|
config.append("\nredirect-gateway def1 ipv6 bypass-dhcp\n");
|
||||||
|
|
||||||
|
#if !defined(Q_OS_ANDROID) && !defined(Q_OS_IOS)
|
||||||
|
// Prevent ipv6 leak
|
||||||
|
config.append("ifconfig-ipv6 fd15:53b6:dead::2/64 fd15:53b6:dead::1\n");
|
||||||
|
#endif
|
||||||
config.append("block-ipv6\n");
|
config.append("block-ipv6\n");
|
||||||
} else if (m_settings->routeMode() == Settings::VpnOnlyForwardSites) {
|
} else if (m_settings->routeMode() == Settings::VpnOnlyForwardSites) {
|
||||||
|
|
||||||
// no redirect-gateway
|
// no redirect-gateway
|
||||||
} else if (m_settings->routeMode() == Settings::VpnAllExceptSites) {
|
} else if (m_settings->routeMode() == Settings::VpnAllExceptSites) {
|
||||||
#if !defined(Q_OS_ANDROID) && !defined(Q_OS_IOS)
|
#if !defined(Q_OS_ANDROID) && !defined(Q_OS_IOS)
|
||||||
config.append("\nredirect-gateway ipv6 !ipv4 bypass-dhcp\n");
|
config.append("\nredirect-gateway ipv6 !ipv4 bypass-dhcp\n");
|
||||||
// Prevent ipv6 leak
|
// Prevent ipv6 leak
|
||||||
|
config.append("ifconfig-ipv6 fd15:53b6:dead::2/64 fd15:53b6:dead::1\n");
|
||||||
#endif
|
#endif
|
||||||
config.append("block-ipv6\n");
|
config.append("block-ipv6\n");
|
||||||
}
|
}
|
||||||
@@ -164,6 +169,7 @@ QString OpenVpnConfigurator::processConfigWithExportSettings(const QPair<QString
|
|||||||
config.append("\nredirect-gateway def1 ipv6 bypass-dhcp\n");
|
config.append("\nredirect-gateway def1 ipv6 bypass-dhcp\n");
|
||||||
|
|
||||||
// Prevent ipv6 leak
|
// Prevent ipv6 leak
|
||||||
|
config.append("ifconfig-ipv6 fd15:53b6:dead::2/64 fd15:53b6:dead::1\n");
|
||||||
config.append("block-ipv6\n");
|
config.append("block-ipv6\n");
|
||||||
|
|
||||||
// remove block-outside-dns for all exported configs
|
// remove block-outside-dns for all exported configs
|
||||||
|
|||||||
@@ -3,7 +3,6 @@
|
|||||||
#include <QDebug>
|
#include <QDebug>
|
||||||
#include <QJsonDocument>
|
#include <QJsonDocument>
|
||||||
#include <QProcess>
|
#include <QProcess>
|
||||||
#include <QRegularExpression>
|
|
||||||
#include <QString>
|
#include <QString>
|
||||||
#include <QTemporaryDir>
|
#include <QTemporaryDir>
|
||||||
#include <QTemporaryFile>
|
#include <QTemporaryFile>
|
||||||
@@ -20,17 +19,13 @@
|
|||||||
#include "settings.h"
|
#include "settings.h"
|
||||||
#include "utilities.h"
|
#include "utilities.h"
|
||||||
|
|
||||||
WireguardConfigurator::WireguardConfigurator(std::shared_ptr<Settings> settings,
|
WireguardConfigurator::WireguardConfigurator(std::shared_ptr<Settings> settings, const QSharedPointer<ServerController> &serverController,
|
||||||
const QSharedPointer<ServerController> &serverController, bool isAwg,
|
bool isAwg, QObject *parent)
|
||||||
QObject *parent)
|
|
||||||
: ConfiguratorBase(settings, serverController, parent), m_isAwg(isAwg)
|
: ConfiguratorBase(settings, serverController, parent), m_isAwg(isAwg)
|
||||||
{
|
{
|
||||||
m_serverConfigPath =
|
m_serverConfigPath = m_isAwg ? amnezia::protocols::awg::serverConfigPath : amnezia::protocols::wireguard::serverConfigPath;
|
||||||
m_isAwg ? amnezia::protocols::awg::serverConfigPath : amnezia::protocols::wireguard::serverConfigPath;
|
m_serverPublicKeyPath = m_isAwg ? amnezia::protocols::awg::serverPublicKeyPath : amnezia::protocols::wireguard::serverPublicKeyPath;
|
||||||
m_serverPublicKeyPath =
|
m_serverPskKeyPath = m_isAwg ? amnezia::protocols::awg::serverPskKeyPath : amnezia::protocols::wireguard::serverPskKeyPath;
|
||||||
m_isAwg ? amnezia::protocols::awg::serverPublicKeyPath : amnezia::protocols::wireguard::serverPublicKeyPath;
|
|
||||||
m_serverPskKeyPath =
|
|
||||||
m_isAwg ? amnezia::protocols::awg::serverPskKeyPath : amnezia::protocols::wireguard::serverPskKeyPath;
|
|
||||||
m_configTemplate = m_isAwg ? ProtocolScriptType::awg_template : ProtocolScriptType::wireguard_template;
|
m_configTemplate = m_isAwg ? ProtocolScriptType::awg_template : ProtocolScriptType::wireguard_template;
|
||||||
|
|
||||||
m_protocolName = m_isAwg ? config_key::awg : config_key::wireguard;
|
m_protocolName = m_isAwg ? config_key::awg : config_key::wireguard;
|
||||||
@@ -68,31 +63,9 @@ WireguardConfigurator::ConnectionData WireguardConfigurator::genClientKeys()
|
|||||||
return connData;
|
return connData;
|
||||||
}
|
}
|
||||||
|
|
||||||
QList<QHostAddress> WireguardConfigurator::getIpsFromConf(const QString &input)
|
|
||||||
{
|
|
||||||
QRegularExpression regex("AllowedIPs = (\\d+\\.\\d+\\.\\d+\\.\\d+)");
|
|
||||||
QRegularExpressionMatchIterator matchIterator = regex.globalMatch(input);
|
|
||||||
|
|
||||||
QList<QHostAddress> ips;
|
|
||||||
|
|
||||||
while (matchIterator.hasNext()) {
|
|
||||||
QRegularExpressionMatch match = matchIterator.next();
|
|
||||||
const QString address_string { match.captured(1) };
|
|
||||||
const QHostAddress address { address_string };
|
|
||||||
if (address.isNull()) {
|
|
||||||
qWarning() << "Couldn't recognize the ip address: " << address_string;
|
|
||||||
} else {
|
|
||||||
ips << address;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return ips;
|
|
||||||
}
|
|
||||||
|
|
||||||
WireguardConfigurator::ConnectionData WireguardConfigurator::prepareWireguardConfig(const ServerCredentials &credentials,
|
WireguardConfigurator::ConnectionData WireguardConfigurator::prepareWireguardConfig(const ServerCredentials &credentials,
|
||||||
DockerContainer container,
|
DockerContainer container,
|
||||||
const QJsonObject &containerConfig,
|
const QJsonObject &containerConfig, ErrorCode &errorCode)
|
||||||
ErrorCode &errorCode)
|
|
||||||
{
|
{
|
||||||
WireguardConfigurator::ConnectionData connData = WireguardConfigurator::genClientKeys();
|
WireguardConfigurator::ConnectionData connData = WireguardConfigurator::genClientKeys();
|
||||||
connData.host = credentials.hostName;
|
connData.host = credentials.hostName;
|
||||||
@@ -103,45 +76,65 @@ WireguardConfigurator::ConnectionData WireguardConfigurator::prepareWireguardCon
|
|||||||
return connData;
|
return connData;
|
||||||
}
|
}
|
||||||
|
|
||||||
QString getIpsScript = QString("cat %1 | grep AllowedIPs").arg(m_serverConfigPath);
|
// Get list of already created clients (only IP addresses)
|
||||||
QString stdOut;
|
QString nextIpNumber;
|
||||||
auto cbReadStdOut = [&](const QString &data, libssh::Client &) {
|
{
|
||||||
stdOut += data + "\n";
|
QString script = QString("cat %1 | grep AllowedIPs").arg(m_serverConfigPath);
|
||||||
return ErrorCode::NoError;
|
QString stdOut;
|
||||||
};
|
auto cbReadStdOut = [&](const QString &data, libssh::Client &) {
|
||||||
|
stdOut += data + "\n";
|
||||||
|
return ErrorCode::NoError;
|
||||||
|
};
|
||||||
|
|
||||||
errorCode = m_serverController->runContainerScript(credentials, container, getIpsScript, cbReadStdOut);
|
errorCode = m_serverController->runContainerScript(credentials, container, script, cbReadStdOut);
|
||||||
if (errorCode != ErrorCode::NoError) {
|
if (errorCode != ErrorCode::NoError) {
|
||||||
return connData;
|
return connData;
|
||||||
}
|
}
|
||||||
auto ips = getIpsFromConf(stdOut);
|
|
||||||
|
|
||||||
QHostAddress nextIp = [&] {
|
stdOut.replace("AllowedIPs = ", "");
|
||||||
QHostAddress result;
|
stdOut.replace("/32", "");
|
||||||
QHostAddress lastIp;
|
QStringList ips = stdOut.split("\n", Qt::SkipEmptyParts);
|
||||||
if (ips.empty()) {
|
|
||||||
lastIp.setAddress(containerConfig.value(m_protocolName)
|
// remove extra IPs from each line for case when user manually edited the wg0.conf
|
||||||
.toObject()
|
// and added there more IPs for route his itnernal networks, like:
|
||||||
.value(config_key::subnet_address)
|
// ...
|
||||||
.toString(protocols::wireguard::defaultSubnetAddress));
|
// AllowedIPs = 10.8.1.6/32, 192.168.1.0/24, 192.168.2.0/24, ...
|
||||||
|
// ...
|
||||||
|
// without this code - next IP would be 1 if last item in 'ips' has format above
|
||||||
|
QStringList vpnIps;
|
||||||
|
for (const auto &ip : ips) {
|
||||||
|
vpnIps.append(ip.split(",", Qt::SkipEmptyParts).first().trimmed());
|
||||||
|
}
|
||||||
|
ips = vpnIps;
|
||||||
|
|
||||||
|
// Calc next IP address
|
||||||
|
if (ips.isEmpty()) {
|
||||||
|
nextIpNumber = "2";
|
||||||
} else {
|
} else {
|
||||||
lastIp = ips.last();
|
int next = ips.last().split(".").last().toInt() + 1;
|
||||||
|
if (next > 254) {
|
||||||
|
errorCode = ErrorCode::AddressPoolError;
|
||||||
|
return connData;
|
||||||
|
}
|
||||||
|
nextIpNumber = QString::number(next);
|
||||||
}
|
}
|
||||||
quint8 lastOctet = static_cast<quint8>(lastIp.toIPv4Address());
|
}
|
||||||
switch (lastOctet) {
|
|
||||||
case 254: result.setAddress(lastIp.toIPv4Address() + 3); break;
|
QString subnetIp = containerConfig.value(m_protocolName).toObject().value(config_key::subnet_address).toString(protocols::wireguard::defaultSubnetAddress);
|
||||||
case 255: result.setAddress(lastIp.toIPv4Address() + 2); break;
|
{
|
||||||
default: result.setAddress(lastIp.toIPv4Address() + 1); break;
|
QStringList l = subnetIp.split(".", Qt::SkipEmptyParts);
|
||||||
|
if (l.isEmpty()) {
|
||||||
|
errorCode = ErrorCode::AddressPoolError;
|
||||||
|
return connData;
|
||||||
}
|
}
|
||||||
|
l.removeLast();
|
||||||
|
l.append(nextIpNumber);
|
||||||
|
|
||||||
return result;
|
connData.clientIP = l.join(".");
|
||||||
}();
|
}
|
||||||
|
|
||||||
connData.clientIP = nextIp.toString();
|
|
||||||
|
|
||||||
// Get keys
|
// Get keys
|
||||||
connData.serverPubKey =
|
connData.serverPubKey = m_serverController->getTextFileFromContainer(container, credentials, m_serverPublicKeyPath, errorCode);
|
||||||
m_serverController->getTextFileFromContainer(container, credentials, m_serverPublicKeyPath, errorCode);
|
|
||||||
connData.serverPubKey.replace("\n", "");
|
connData.serverPubKey.replace("\n", "");
|
||||||
if (errorCode != ErrorCode::NoError) {
|
if (errorCode != ErrorCode::NoError) {
|
||||||
return connData;
|
return connData;
|
||||||
@@ -168,12 +161,10 @@ WireguardConfigurator::ConnectionData WireguardConfigurator::prepareWireguardCon
|
|||||||
return connData;
|
return connData;
|
||||||
}
|
}
|
||||||
|
|
||||||
QString script = QString("sudo docker exec -i $CONTAINER_NAME bash -c 'wg syncconf wg0 <(wg-quick strip %1)'")
|
QString script = QString("sudo docker exec -i $CONTAINER_NAME bash -c 'wg syncconf wg0 <(wg-quick strip %1)'").arg(m_serverConfigPath);
|
||||||
.arg(m_serverConfigPath);
|
|
||||||
|
|
||||||
errorCode = m_serverController->runScript(
|
errorCode = m_serverController->runScript(
|
||||||
credentials,
|
credentials, m_serverController->replaceVars(script, m_serverController->genVarsForScript(credentials, container)));
|
||||||
m_serverController->replaceVars(script, m_serverController->genVarsForScript(credentials, container)));
|
|
||||||
|
|
||||||
return connData;
|
return connData;
|
||||||
}
|
}
|
||||||
@@ -182,8 +173,8 @@ QString WireguardConfigurator::createConfig(const ServerCredentials &credentials
|
|||||||
const QJsonObject &containerConfig, ErrorCode &errorCode)
|
const QJsonObject &containerConfig, ErrorCode &errorCode)
|
||||||
{
|
{
|
||||||
QString scriptData = amnezia::scriptData(m_configTemplate, container);
|
QString scriptData = amnezia::scriptData(m_configTemplate, container);
|
||||||
QString config = m_serverController->replaceVars(
|
QString config =
|
||||||
scriptData, m_serverController->genVarsForScript(credentials, container, containerConfig));
|
m_serverController->replaceVars(scriptData, m_serverController->genVarsForScript(credentials, container, containerConfig));
|
||||||
|
|
||||||
ConnectionData connData = prepareWireguardConfig(credentials, container, containerConfig, errorCode);
|
ConnectionData connData = prepareWireguardConfig(credentials, container, containerConfig, errorCode);
|
||||||
if (errorCode != ErrorCode::NoError) {
|
if (errorCode != ErrorCode::NoError) {
|
||||||
@@ -217,16 +208,16 @@ QString WireguardConfigurator::createConfig(const ServerCredentials &credentials
|
|||||||
return QJsonDocument(jConfig).toJson();
|
return QJsonDocument(jConfig).toJson();
|
||||||
}
|
}
|
||||||
|
|
||||||
QString WireguardConfigurator::processConfigWithLocalSettings(const QPair<QString, QString> &dns,
|
QString WireguardConfigurator::processConfigWithLocalSettings(const QPair<QString, QString> &dns, const bool isApiConfig,
|
||||||
const bool isApiConfig, QString &protocolConfigString)
|
QString &protocolConfigString)
|
||||||
{
|
{
|
||||||
processConfigWithDnsSettings(dns, protocolConfigString);
|
processConfigWithDnsSettings(dns, protocolConfigString);
|
||||||
|
|
||||||
return protocolConfigString;
|
return protocolConfigString;
|
||||||
}
|
}
|
||||||
|
|
||||||
QString WireguardConfigurator::processConfigWithExportSettings(const QPair<QString, QString> &dns,
|
QString WireguardConfigurator::processConfigWithExportSettings(const QPair<QString, QString> &dns, const bool isApiConfig,
|
||||||
const bool isApiConfig, QString &protocolConfigString)
|
QString &protocolConfigString)
|
||||||
{
|
{
|
||||||
processConfigWithDnsSettings(dns, protocolConfigString);
|
processConfigWithDnsSettings(dns, protocolConfigString);
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
#ifndef WIREGUARD_CONFIGURATOR_H
|
#ifndef WIREGUARD_CONFIGURATOR_H
|
||||||
#define WIREGUARD_CONFIGURATOR_H
|
#define WIREGUARD_CONFIGURATOR_H
|
||||||
|
|
||||||
#include <QHostAddress>
|
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
#include <QProcessEnvironment>
|
#include <QProcessEnvironment>
|
||||||
|
|
||||||
@@ -13,8 +12,8 @@ class WireguardConfigurator : public ConfiguratorBase
|
|||||||
{
|
{
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
public:
|
public:
|
||||||
WireguardConfigurator(std::shared_ptr<Settings> settings, const QSharedPointer<ServerController> &serverController,
|
WireguardConfigurator(std::shared_ptr<Settings> settings, const QSharedPointer<ServerController> &serverController, bool isAwg,
|
||||||
bool isAwg, QObject *parent = nullptr);
|
QObject *parent = nullptr);
|
||||||
|
|
||||||
struct ConnectionData
|
struct ConnectionData
|
||||||
{
|
{
|
||||||
@@ -27,18 +26,15 @@ public:
|
|||||||
QString port;
|
QString port;
|
||||||
};
|
};
|
||||||
|
|
||||||
QString createConfig(const ServerCredentials &credentials, DockerContainer container,
|
QString createConfig(const ServerCredentials &credentials, DockerContainer container, const QJsonObject &containerConfig,
|
||||||
const QJsonObject &containerConfig, ErrorCode &errorCode);
|
ErrorCode &errorCode);
|
||||||
|
|
||||||
QString processConfigWithLocalSettings(const QPair<QString, QString> &dns, const bool isApiConfig,
|
QString processConfigWithLocalSettings(const QPair<QString, QString> &dns, const bool isApiConfig, QString &protocolConfigString);
|
||||||
QString &protocolConfigString);
|
QString processConfigWithExportSettings(const QPair<QString, QString> &dns, const bool isApiConfig, QString &protocolConfigString);
|
||||||
QString processConfigWithExportSettings(const QPair<QString, QString> &dns, const bool isApiConfig,
|
|
||||||
QString &protocolConfigString);
|
|
||||||
|
|
||||||
static ConnectionData genClientKeys();
|
static ConnectionData genClientKeys();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
QList<QHostAddress> getIpsFromConf(const QString &input);
|
|
||||||
ConnectionData prepareWireguardConfig(const ServerCredentials &credentials, DockerContainer container,
|
ConnectionData prepareWireguardConfig(const ServerCredentials &credentials, DockerContainer container,
|
||||||
const QJsonObject &containerConfig, ErrorCode &errorCode);
|
const QJsonObject &containerConfig, ErrorCode &errorCode);
|
||||||
|
|
||||||
|
|||||||
@@ -10,8 +10,7 @@ namespace apiDefs
|
|||||||
AmneziaFreeV3,
|
AmneziaFreeV3,
|
||||||
AmneziaPremiumV1,
|
AmneziaPremiumV1,
|
||||||
AmneziaPremiumV2,
|
AmneziaPremiumV2,
|
||||||
SelfHosted,
|
SelfHosted
|
||||||
ExternalPremium
|
|
||||||
};
|
};
|
||||||
|
|
||||||
enum ConfigSource {
|
enum ConfigSource {
|
||||||
@@ -44,13 +43,6 @@ namespace apiDefs
|
|||||||
constexpr QLatin1String maxDeviceCount("max_device_count");
|
constexpr QLatin1String maxDeviceCount("max_device_count");
|
||||||
constexpr QLatin1String subscriptionEndDate("subscription_end_date");
|
constexpr QLatin1String subscriptionEndDate("subscription_end_date");
|
||||||
constexpr QLatin1String issuedConfigs("issued_configs");
|
constexpr QLatin1String issuedConfigs("issued_configs");
|
||||||
|
|
||||||
constexpr QLatin1String supportInfo("support_info");
|
|
||||||
constexpr QLatin1String email("email");
|
|
||||||
constexpr QLatin1String billingEmail("billing_email");
|
|
||||||
constexpr QLatin1String website("website");
|
|
||||||
constexpr QLatin1String websiteName("website_name");
|
|
||||||
constexpr QLatin1String telegram("telegram");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
const int requestTimeoutMsecs = 12 * 1000; // 12 secs
|
const int requestTimeoutMsecs = 12 * 1000; // 12 secs
|
||||||
|
|||||||
@@ -32,17 +32,15 @@ apiDefs::ConfigType apiUtils::getConfigType(const QJsonObject &serverConfigObjec
|
|||||||
|
|
||||||
constexpr QLatin1String servicePremium("amnezia-premium");
|
constexpr QLatin1String servicePremium("amnezia-premium");
|
||||||
constexpr QLatin1String serviceFree("amnezia-free");
|
constexpr QLatin1String serviceFree("amnezia-free");
|
||||||
constexpr QLatin1String serviceExternalPremium("external-premium");
|
|
||||||
|
|
||||||
auto apiConfigObject = serverConfigObject.value(apiDefs::key::apiConfig).toObject();
|
auto apiConfigObject = serverConfigObject.value(apiDefs::key::apiConfig).toObject();
|
||||||
|
auto stackType = apiConfigObject.value(apiDefs::key::stackType).toString();
|
||||||
auto serviceType = apiConfigObject.value(apiDefs::key::serviceType).toString();
|
auto serviceType = apiConfigObject.value(apiDefs::key::serviceType).toString();
|
||||||
|
|
||||||
if (serviceType == servicePremium) {
|
if (serviceType == servicePremium || stackType == stackPremium) {
|
||||||
return apiDefs::ConfigType::AmneziaPremiumV2;
|
return apiDefs::ConfigType::AmneziaPremiumV2;
|
||||||
} else if (serviceType == serviceFree) {
|
} else if (serviceType == serviceFree || stackType == stackFree) {
|
||||||
return apiDefs::ConfigType::AmneziaFreeV3;
|
return apiDefs::ConfigType::AmneziaFreeV3;
|
||||||
} else if (serviceType == serviceExternalPremium) {
|
|
||||||
return apiDefs::ConfigType::ExternalPremium;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
default: {
|
default: {
|
||||||
@@ -68,7 +66,6 @@ amnezia::ErrorCode apiUtils::checkNetworkReplyErrors(const QList<QSslError> &ssl
|
|||||||
return amnezia::ErrorCode::NoError;
|
return amnezia::ErrorCode::NoError;
|
||||||
} else if (reply->error() == QNetworkReply::NetworkError::OperationCanceledError
|
} else if (reply->error() == QNetworkReply::NetworkError::OperationCanceledError
|
||||||
|| reply->error() == QNetworkReply::NetworkError::TimeoutError) {
|
|| reply->error() == QNetworkReply::NetworkError::TimeoutError) {
|
||||||
qDebug() << reply->error();
|
|
||||||
return amnezia::ErrorCode::ApiConfigTimeoutError;
|
return amnezia::ErrorCode::ApiConfigTimeoutError;
|
||||||
} else {
|
} else {
|
||||||
QString err = reply->errorString();
|
QString err = reply->errorString();
|
||||||
@@ -88,10 +85,3 @@ amnezia::ErrorCode apiUtils::checkNetworkReplyErrors(const QList<QSslError> &ssl
|
|||||||
qDebug() << "something went wrong";
|
qDebug() << "something went wrong";
|
||||||
return amnezia::ErrorCode::InternalError;
|
return amnezia::ErrorCode::InternalError;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool apiUtils::isPremiumServer(const QJsonObject &serverConfigObject)
|
|
||||||
{
|
|
||||||
static const QSet<apiDefs::ConfigType> premiumTypes = { apiDefs::ConfigType::AmneziaPremiumV1, apiDefs::ConfigType::AmneziaPremiumV2,
|
|
||||||
apiDefs::ConfigType::ExternalPremium };
|
|
||||||
return premiumTypes.contains(getConfigType(serverConfigObject));
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -13,8 +13,6 @@ namespace apiUtils
|
|||||||
|
|
||||||
bool isSubscriptionExpired(const QString &subscriptionEndDate);
|
bool isSubscriptionExpired(const QString &subscriptionEndDate);
|
||||||
|
|
||||||
bool isPremiumServer(const QJsonObject &serverConfigObject);
|
|
||||||
|
|
||||||
apiDefs::ConfigType getConfigType(const QJsonObject &serverConfigObject);
|
apiDefs::ConfigType getConfigType(const QJsonObject &serverConfigObject);
|
||||||
apiDefs::ConfigSource getConfigSource(const QJsonObject &serverConfigObject);
|
apiDefs::ConfigSource getConfigSource(const QJsonObject &serverConfigObject);
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
#include "coreController.h"
|
#include "coreController.h"
|
||||||
|
|
||||||
#include <QDirIterator>
|
|
||||||
#include <QTranslator>
|
#include <QTranslator>
|
||||||
|
|
||||||
#if defined(Q_OS_ANDROID)
|
#if defined(Q_OS_ANDROID)
|
||||||
@@ -48,9 +47,6 @@ void CoreController::initModels()
|
|||||||
m_sitesModel.reset(new SitesModel(m_settings, this));
|
m_sitesModel.reset(new SitesModel(m_settings, this));
|
||||||
m_engine->rootContext()->setContextProperty("SitesModel", m_sitesModel.get());
|
m_engine->rootContext()->setContextProperty("SitesModel", m_sitesModel.get());
|
||||||
|
|
||||||
m_allowedDnsModel.reset(new AllowedDnsModel(m_settings, this));
|
|
||||||
m_engine->rootContext()->setContextProperty("AllowedDnsModel", m_allowedDnsModel.get());
|
|
||||||
|
|
||||||
m_appSplitTunnelingModel.reset(new AppSplitTunnelingModel(m_settings, this));
|
m_appSplitTunnelingModel.reset(new AppSplitTunnelingModel(m_settings, this));
|
||||||
m_engine->rootContext()->setContextProperty("AppSplitTunnelingModel", m_appSplitTunnelingModel.get());
|
m_engine->rootContext()->setContextProperty("AppSplitTunnelingModel", m_appSplitTunnelingModel.get());
|
||||||
|
|
||||||
@@ -133,9 +129,6 @@ void CoreController::initControllers()
|
|||||||
m_sitesController.reset(new SitesController(m_settings, m_vpnConnection, m_sitesModel));
|
m_sitesController.reset(new SitesController(m_settings, m_vpnConnection, m_sitesModel));
|
||||||
m_engine->rootContext()->setContextProperty("SitesController", m_sitesController.get());
|
m_engine->rootContext()->setContextProperty("SitesController", m_sitesController.get());
|
||||||
|
|
||||||
m_allowedDnsController.reset(new AllowedDnsController(m_settings, m_allowedDnsModel));
|
|
||||||
m_engine->rootContext()->setContextProperty("AllowedDnsController", m_allowedDnsController.get());
|
|
||||||
|
|
||||||
m_appSplitTunnelingController.reset(new AppSplitTunnelingController(m_settings, m_appSplitTunnelingModel));
|
m_appSplitTunnelingController.reset(new AppSplitTunnelingController(m_settings, m_appSplitTunnelingModel));
|
||||||
m_engine->rootContext()->setContextProperty("AppSplitTunnelingController", m_appSplitTunnelingController.get());
|
m_engine->rootContext()->setContextProperty("AppSplitTunnelingController", m_appSplitTunnelingController.get());
|
||||||
|
|
||||||
@@ -220,7 +213,6 @@ void CoreController::initSignalHandlers()
|
|||||||
initAutoConnectHandler();
|
initAutoConnectHandler();
|
||||||
initAmneziaDnsToggledHandler();
|
initAmneziaDnsToggledHandler();
|
||||||
initPrepareConfigHandler();
|
initPrepareConfigHandler();
|
||||||
initStrictKillSwitchHandler();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void CoreController::initNotificationHandler()
|
void CoreController::initNotificationHandler()
|
||||||
@@ -246,23 +238,7 @@ void CoreController::updateTranslator(const QLocale &locale)
|
|||||||
QCoreApplication::removeTranslator(m_translator.get());
|
QCoreApplication::removeTranslator(m_translator.get());
|
||||||
}
|
}
|
||||||
|
|
||||||
QStringList availableTranslations;
|
QString strFileName = QString(":/translations/amneziavpn") + QLatin1String("_") + locale.name() + ".qm";
|
||||||
QDirIterator it(":/translations", QStringList("amneziavpn_*.qm"), QDir::Files);
|
|
||||||
while (it.hasNext()) {
|
|
||||||
availableTranslations << it.next();
|
|
||||||
}
|
|
||||||
|
|
||||||
// This code allow to load translation for the language only, without country code
|
|
||||||
const QString lang = locale.name().split("_").first();
|
|
||||||
const QString translationFilePrefix = QString(":/translations/amneziavpn_") + lang;
|
|
||||||
QString strFileName = QString(":/translations/amneziavpn_%1.qm").arg(locale.name());
|
|
||||||
for (const QString &translation : availableTranslations) {
|
|
||||||
if (translation.contains(translationFilePrefix)) {
|
|
||||||
strFileName = translation;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (m_translator->load(strFileName)) {
|
if (m_translator->load(strFileName)) {
|
||||||
if (QCoreApplication::installTranslator(m_translator.get())) {
|
if (QCoreApplication::installTranslator(m_translator.get())) {
|
||||||
m_settings->setAppLanguage(locale);
|
m_settings->setAppLanguage(locale);
|
||||||
@@ -363,12 +339,6 @@ void CoreController::initPrepareConfigHandler()
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
void CoreController::initStrictKillSwitchHandler()
|
|
||||||
{
|
|
||||||
connect(m_settingsController.get(), &SettingsController::strictKillSwitchEnabledChanged,
|
|
||||||
m_vpnConnection.get(), &VpnConnection::onKillSwitchModeChanged);
|
|
||||||
}
|
|
||||||
|
|
||||||
QSharedPointer<PageController> CoreController::pageController() const
|
QSharedPointer<PageController> CoreController::pageController() const
|
||||||
{
|
{
|
||||||
return m_pageController;
|
return m_pageController;
|
||||||
|
|||||||
@@ -8,7 +8,6 @@
|
|||||||
#include "ui/controllers/api/apiConfigsController.h"
|
#include "ui/controllers/api/apiConfigsController.h"
|
||||||
#include "ui/controllers/api/apiSettingsController.h"
|
#include "ui/controllers/api/apiSettingsController.h"
|
||||||
#include "ui/controllers/appSplitTunnelingController.h"
|
#include "ui/controllers/appSplitTunnelingController.h"
|
||||||
#include "ui/controllers/allowedDnsController.h"
|
|
||||||
#include "ui/controllers/connectionController.h"
|
#include "ui/controllers/connectionController.h"
|
||||||
#include "ui/controllers/exportController.h"
|
#include "ui/controllers/exportController.h"
|
||||||
#include "ui/controllers/focusController.h"
|
#include "ui/controllers/focusController.h"
|
||||||
@@ -19,7 +18,6 @@
|
|||||||
#include "ui/controllers/sitesController.h"
|
#include "ui/controllers/sitesController.h"
|
||||||
#include "ui/controllers/systemController.h"
|
#include "ui/controllers/systemController.h"
|
||||||
|
|
||||||
#include "ui/models/allowed_dns_model.h"
|
|
||||||
#include "ui/models/containers_model.h"
|
#include "ui/models/containers_model.h"
|
||||||
#include "ui/models/languageModel.h"
|
#include "ui/models/languageModel.h"
|
||||||
#include "ui/models/protocols/cloakConfigModel.h"
|
#include "ui/models/protocols/cloakConfigModel.h"
|
||||||
@@ -82,7 +80,6 @@ private:
|
|||||||
void initAutoConnectHandler();
|
void initAutoConnectHandler();
|
||||||
void initAmneziaDnsToggledHandler();
|
void initAmneziaDnsToggledHandler();
|
||||||
void initPrepareConfigHandler();
|
void initPrepareConfigHandler();
|
||||||
void initStrictKillSwitchHandler();
|
|
||||||
|
|
||||||
QQmlApplicationEngine *m_engine {}; // TODO use parent child system here?
|
QQmlApplicationEngine *m_engine {}; // TODO use parent child system here?
|
||||||
std::shared_ptr<Settings> m_settings;
|
std::shared_ptr<Settings> m_settings;
|
||||||
@@ -105,7 +102,6 @@ private:
|
|||||||
QScopedPointer<SitesController> m_sitesController;
|
QScopedPointer<SitesController> m_sitesController;
|
||||||
QScopedPointer<SystemController> m_systemController;
|
QScopedPointer<SystemController> m_systemController;
|
||||||
QScopedPointer<AppSplitTunnelingController> m_appSplitTunnelingController;
|
QScopedPointer<AppSplitTunnelingController> m_appSplitTunnelingController;
|
||||||
QScopedPointer<AllowedDnsController> m_allowedDnsController;
|
|
||||||
|
|
||||||
QScopedPointer<ApiSettingsController> m_apiSettingsController;
|
QScopedPointer<ApiSettingsController> m_apiSettingsController;
|
||||||
QScopedPointer<ApiConfigsController> m_apiConfigsController;
|
QScopedPointer<ApiConfigsController> m_apiConfigsController;
|
||||||
@@ -116,7 +112,6 @@ private:
|
|||||||
QSharedPointer<LanguageModel> m_languageModel;
|
QSharedPointer<LanguageModel> m_languageModel;
|
||||||
QSharedPointer<ProtocolsModel> m_protocolsModel;
|
QSharedPointer<ProtocolsModel> m_protocolsModel;
|
||||||
QSharedPointer<SitesModel> m_sitesModel;
|
QSharedPointer<SitesModel> m_sitesModel;
|
||||||
QSharedPointer<AllowedDnsModel> m_allowedDnsModel;
|
|
||||||
QSharedPointer<AppSplitTunnelingModel> m_appSplitTunnelingModel;
|
QSharedPointer<AppSplitTunnelingModel> m_appSplitTunnelingModel;
|
||||||
QSharedPointer<ClientManagementModel> m_clientManagementModel;
|
QSharedPointer<ClientManagementModel> m_clientManagementModel;
|
||||||
|
|
||||||
|
|||||||
@@ -7,7 +7,6 @@
|
|||||||
#include <QJsonDocument>
|
#include <QJsonDocument>
|
||||||
#include <QJsonObject>
|
#include <QJsonObject>
|
||||||
#include <QNetworkReply>
|
#include <QNetworkReply>
|
||||||
#include <QUrl>
|
|
||||||
|
|
||||||
#include "QBlockCipher.h"
|
#include "QBlockCipher.h"
|
||||||
#include "QRsa.h"
|
#include "QRsa.h"
|
||||||
@@ -15,11 +14,6 @@
|
|||||||
#include "amnezia_application.h"
|
#include "amnezia_application.h"
|
||||||
#include "core/api/apiUtils.h"
|
#include "core/api/apiUtils.h"
|
||||||
#include "utilities.h"
|
#include "utilities.h"
|
||||||
#include "core/networkUtilities.h"
|
|
||||||
|
|
||||||
#ifdef AMNEZIA_DESKTOP
|
|
||||||
#include "core/ipcclient.h"
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace
|
namespace
|
||||||
{
|
{
|
||||||
@@ -32,10 +26,6 @@ namespace
|
|||||||
constexpr char apiPayload[] = "api_payload";
|
constexpr char apiPayload[] = "api_payload";
|
||||||
constexpr char keyPayload[] = "key_payload";
|
constexpr char keyPayload[] = "key_payload";
|
||||||
}
|
}
|
||||||
|
|
||||||
constexpr QLatin1String errorResponsePattern1("No active configuration found for");
|
|
||||||
constexpr QLatin1String errorResponsePattern2("No non-revoked public key found for");
|
|
||||||
constexpr QLatin1String errorResponsePattern3("Account not found.");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
GatewayController::GatewayController(const QString &gatewayEndpoint, bool isDevEnvironment, int requestTimeoutMsecs, QObject *parent)
|
GatewayController::GatewayController(const QString &gatewayEndpoint, bool isDevEnvironment, int requestTimeoutMsecs, QObject *parent)
|
||||||
@@ -56,17 +46,6 @@ ErrorCode GatewayController::get(const QString &endpoint, QByteArray &responseBo
|
|||||||
|
|
||||||
request.setUrl(QString(endpoint).arg(m_gatewayEndpoint));
|
request.setUrl(QString(endpoint).arg(m_gatewayEndpoint));
|
||||||
|
|
||||||
// bypass killSwitch exceptions for API-gateway
|
|
||||||
#ifdef AMNEZIA_DESKTOP
|
|
||||||
{
|
|
||||||
QString host = QUrl(request.url()).host();
|
|
||||||
QString ip = NetworkUtilities::getIPAddress(host);
|
|
||||||
if (!ip.isEmpty()) {
|
|
||||||
IpcClient::Interface()->addKillSwitchAllowedRange(QStringList{ip});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
QNetworkReply *reply;
|
QNetworkReply *reply;
|
||||||
reply = amnApp->networkManager()->get(request);
|
reply = amnApp->networkManager()->get(request);
|
||||||
|
|
||||||
@@ -118,17 +97,6 @@ ErrorCode GatewayController::post(const QString &endpoint, const QJsonObject api
|
|||||||
|
|
||||||
request.setUrl(endpoint.arg(m_gatewayEndpoint));
|
request.setUrl(endpoint.arg(m_gatewayEndpoint));
|
||||||
|
|
||||||
// bypass killSwitch exceptions for API-gateway
|
|
||||||
#ifdef AMNEZIA_DESKTOP
|
|
||||||
{
|
|
||||||
QString host = QUrl(request.url()).host();
|
|
||||||
QString ip = NetworkUtilities::getIPAddress(host);
|
|
||||||
if (!ip.isEmpty()) {
|
|
||||||
IpcClient::Interface()->addKillSwitchAllowedRange(QStringList{ip});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
QSimpleCrypto::QBlockCipher blockCipher;
|
QSimpleCrypto::QBlockCipher blockCipher;
|
||||||
QByteArray key = blockCipher.generatePrivateSalt(32);
|
QByteArray key = blockCipher.generatePrivateSalt(32);
|
||||||
QByteArray iv = blockCipher.generatePrivateSalt(32);
|
QByteArray iv = blockCipher.generatePrivateSalt(32);
|
||||||
@@ -226,16 +194,16 @@ QStringList GatewayController::getProxyUrls()
|
|||||||
QList<QSslError> sslErrors;
|
QList<QSslError> sslErrors;
|
||||||
QNetworkReply *reply;
|
QNetworkReply *reply;
|
||||||
|
|
||||||
QStringList proxyStorageUrls;
|
QStringList proxyStorageUrl;
|
||||||
if (m_isDevEnvironment) {
|
if (m_isDevEnvironment) {
|
||||||
proxyStorageUrls = QString(DEV_S3_ENDPOINT).split(", ");
|
proxyStorageUrl = QStringList { DEV_S3_ENDPOINT };
|
||||||
} else {
|
} else {
|
||||||
proxyStorageUrls = QString(PROD_S3_ENDPOINT).split(", ");
|
proxyStorageUrl = QStringList { PROD_S3_ENDPOINT };
|
||||||
}
|
}
|
||||||
|
|
||||||
QByteArray key = m_isDevEnvironment ? DEV_AGW_PUBLIC_KEY : PROD_AGW_PUBLIC_KEY;
|
QByteArray key = m_isDevEnvironment ? DEV_AGW_PUBLIC_KEY : PROD_AGW_PUBLIC_KEY;
|
||||||
|
|
||||||
for (const auto &proxyStorageUrl : proxyStorageUrls) {
|
for (const auto &proxyStorageUrl : proxyStorageUrl) {
|
||||||
request.setUrl(proxyStorageUrl);
|
request.setUrl(proxyStorageUrl);
|
||||||
reply = amnApp->networkManager()->get(request);
|
reply = amnApp->networkManager()->get(request);
|
||||||
|
|
||||||
@@ -279,9 +247,6 @@ QStringList GatewayController::getProxyUrls()
|
|||||||
}
|
}
|
||||||
return endpoints;
|
return endpoints;
|
||||||
} else {
|
} else {
|
||||||
apiUtils::checkNetworkReplyErrors(sslErrors, reply);
|
|
||||||
qDebug() << "go to the next storage endpoint";
|
|
||||||
|
|
||||||
reply->deleteLater();
|
reply->deleteLater();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -292,29 +257,17 @@ bool GatewayController::shouldBypassProxy(QNetworkReply *reply, const QByteArray
|
|||||||
const QByteArray &iv, const QByteArray &salt)
|
const QByteArray &iv, const QByteArray &salt)
|
||||||
{
|
{
|
||||||
if (reply->error() == QNetworkReply::NetworkError::OperationCanceledError || reply->error() == QNetworkReply::NetworkError::TimeoutError) {
|
if (reply->error() == QNetworkReply::NetworkError::OperationCanceledError || reply->error() == QNetworkReply::NetworkError::TimeoutError) {
|
||||||
qDebug() << "timeout occurred";
|
qDebug() << "Timeout occurred";
|
||||||
qDebug() << reply->error();
|
|
||||||
return true;
|
return true;
|
||||||
} else if (responseBody.contains("html")) {
|
} else if (responseBody.contains("html")) {
|
||||||
qDebug() << "the response contains an html tag";
|
qDebug() << "The response contains an html tag";
|
||||||
return true;
|
return true;
|
||||||
} else if (reply->error() == QNetworkReply::NetworkError::ContentNotFoundError) {
|
} else if (reply->error() == QNetworkReply::NetworkError::NoError && checkEncryption) {
|
||||||
if (responseBody.contains(errorResponsePattern1) || responseBody.contains(errorResponsePattern2)
|
|
||||||
|| responseBody.contains(errorResponsePattern3)) {
|
|
||||||
return false;
|
|
||||||
} else {
|
|
||||||
qDebug() << reply->error();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
} else if (reply->error() != QNetworkReply::NetworkError::NoError) {
|
|
||||||
qDebug() << reply->error();
|
|
||||||
return true;
|
|
||||||
} else if (checkEncryption) {
|
|
||||||
try {
|
try {
|
||||||
QSimpleCrypto::QBlockCipher blockCipher;
|
QSimpleCrypto::QBlockCipher blockCipher;
|
||||||
static_cast<void>(blockCipher.decryptAesBlockCipher(responseBody, key, iv, "", salt));
|
static_cast<void>(blockCipher.decryptAesBlockCipher(responseBody, key, iv, "", salt));
|
||||||
} catch (...) {
|
} catch (...) {
|
||||||
qDebug() << "failed to decrypt the data";
|
qDebug() << "Failed to decrypt the data";
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -335,7 +288,7 @@ void GatewayController::bypassProxy(const QString &endpoint, QNetworkReply *repl
|
|||||||
QByteArray responseBody;
|
QByteArray responseBody;
|
||||||
|
|
||||||
for (const QString &proxyUrl : proxyUrls) {
|
for (const QString &proxyUrl : proxyUrls) {
|
||||||
qDebug() << "go to the next proxy endpoint";
|
qDebug() << "Go to the next endpoint";
|
||||||
reply->deleteLater(); // delete the previous reply
|
reply->deleteLater(); // delete the previous reply
|
||||||
reply = requestFunction(endpoint.arg(proxyUrl));
|
reply = requestFunction(endpoint.arg(proxyUrl));
|
||||||
|
|
||||||
|
|||||||
@@ -439,22 +439,15 @@ ErrorCode ServerController::buildContainerWorker(const ServerCredentials &creden
|
|||||||
stdOut += data + "\n";
|
stdOut += data + "\n";
|
||||||
return ErrorCode::NoError;
|
return ErrorCode::NoError;
|
||||||
};
|
};
|
||||||
auto cbReadStdErr = [&](const QString &data, libssh::Client &) {
|
|
||||||
stdOut += data + "\n";
|
|
||||||
return ErrorCode::NoError;
|
|
||||||
};
|
|
||||||
|
|
||||||
ErrorCode error =
|
errorCode =
|
||||||
runScript(credentials,
|
runScript(credentials,
|
||||||
replaceVars(amnezia::scriptData(SharedScriptType::build_container), genVarsForScript(credentials, container, config)),
|
replaceVars(amnezia::scriptData(SharedScriptType::build_container), genVarsForScript(credentials, container, config)),
|
||||||
cbReadStdOut, cbReadStdErr);
|
cbReadStdOut);
|
||||||
|
if (errorCode)
|
||||||
|
return errorCode;
|
||||||
|
|
||||||
if (stdOut.contains("doesn't work on cgroups v2"))
|
return errorCode;
|
||||||
return ErrorCode::ServerDockerOnCgroupsV2;
|
|
||||||
if (stdOut.contains("cgroup mountpoint does not exist"))
|
|
||||||
return ErrorCode::ServerCgroupMountpoint;
|
|
||||||
|
|
||||||
return error;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ErrorCode ServerController::runContainerWorker(const ServerCredentials &credentials, DockerContainer container, QJsonObject &config)
|
ErrorCode ServerController::runContainerWorker(const ServerCredentials &credentials, DockerContainer container, QJsonObject &config)
|
||||||
@@ -716,7 +709,7 @@ ErrorCode ServerController::isServerPortBusy(const ServerCredentials &credential
|
|||||||
QString transportProto = containerConfig.value(config_key::transport_proto).toString(defaultTransportProto);
|
QString transportProto = containerConfig.value(config_key::transport_proto).toString(defaultTransportProto);
|
||||||
|
|
||||||
// TODO reimplement with netstat
|
// TODO reimplement with netstat
|
||||||
QString script = QString("which lsof > /dev/null 2>&1 || true && sudo lsof -i -P -n 2>/dev/null | grep -E ':%1 ").arg(port);
|
QString script = QString("which lsof &>/dev/null || true && sudo lsof -i -P -n 2>/dev/null | grep -E ':%1 ").arg(port);
|
||||||
for (auto &port : fixedPorts) {
|
for (auto &port : fixedPorts) {
|
||||||
script = script.append("|:%1").arg(port);
|
script = script.append("|:%1").arg(port);
|
||||||
}
|
}
|
||||||
@@ -764,6 +757,10 @@ ErrorCode ServerController::isServerPortBusy(const ServerCredentials &credential
|
|||||||
|
|
||||||
ErrorCode ServerController::isUserInSudo(const ServerCredentials &credentials, DockerContainer container)
|
ErrorCode ServerController::isUserInSudo(const ServerCredentials &credentials, DockerContainer container)
|
||||||
{
|
{
|
||||||
|
if (credentials.userName == "root") {
|
||||||
|
return ErrorCode::NoError;
|
||||||
|
}
|
||||||
|
|
||||||
QString stdOut;
|
QString stdOut;
|
||||||
auto cbReadStdOut = [&](const QString &data, libssh::Client &) {
|
auto cbReadStdOut = [&](const QString &data, libssh::Client &) {
|
||||||
stdOut += data + "\n";
|
stdOut += data + "\n";
|
||||||
@@ -777,16 +774,8 @@ ErrorCode ServerController::isUserInSudo(const ServerCredentials &credentials, D
|
|||||||
const QString scriptData = amnezia::scriptData(SharedScriptType::check_user_in_sudo);
|
const QString scriptData = amnezia::scriptData(SharedScriptType::check_user_in_sudo);
|
||||||
ErrorCode error = runScript(credentials, replaceVars(scriptData, genVarsForScript(credentials)), cbReadStdOut, cbReadStdErr);
|
ErrorCode error = runScript(credentials, replaceVars(scriptData, genVarsForScript(credentials)), cbReadStdOut, cbReadStdErr);
|
||||||
|
|
||||||
if (credentials.userName != "root" && stdOut.contains("sudo:") && !stdOut.contains("uname:") && stdOut.contains("not found"))
|
if (!stdOut.contains("sudo"))
|
||||||
return ErrorCode::ServerSudoPackageIsNotPreinstalled;
|
|
||||||
if (credentials.userName != "root" && !stdOut.contains("sudo") && !stdOut.contains("wheel"))
|
|
||||||
return ErrorCode::ServerUserNotInSudo;
|
return ErrorCode::ServerUserNotInSudo;
|
||||||
if (stdOut.contains("can't cd to") || stdOut.contains("Permission denied") || stdOut.contains("No such file or directory"))
|
|
||||||
return ErrorCode::ServerUserDirectoryNotAccessible;
|
|
||||||
if (stdOut.contains("sudoers") || stdOut.contains("is not allowed to run sudo on"))
|
|
||||||
return ErrorCode::ServerUserNotAllowedInSudoers;
|
|
||||||
if (stdOut.contains("password is required"))
|
|
||||||
return ErrorCode::ServerUserPasswordRequired;
|
|
||||||
|
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -54,12 +54,6 @@ namespace amnezia
|
|||||||
ServerCancelInstallation = 204,
|
ServerCancelInstallation = 204,
|
||||||
ServerUserNotInSudo = 205,
|
ServerUserNotInSudo = 205,
|
||||||
ServerPacketManagerError = 206,
|
ServerPacketManagerError = 206,
|
||||||
ServerSudoPackageIsNotPreinstalled = 207,
|
|
||||||
ServerUserDirectoryNotAccessible = 208,
|
|
||||||
ServerUserNotAllowedInSudoers = 209,
|
|
||||||
ServerUserPasswordRequired = 210,
|
|
||||||
ServerDockerOnCgroupsV2 = 211,
|
|
||||||
ServerCgroupMountpoint = 212,
|
|
||||||
|
|
||||||
// Ssh connection errors
|
// Ssh connection errors
|
||||||
SshRequestDeniedError = 300,
|
SshRequestDeniedError = 300,
|
||||||
|
|||||||
@@ -20,14 +20,8 @@ QString errorString(ErrorCode code) {
|
|||||||
case(ErrorCode::ServerContainerMissingError): errorMessage = QObject::tr("Server error: Docker container missing"); break;
|
case(ErrorCode::ServerContainerMissingError): errorMessage = QObject::tr("Server error: Docker container missing"); break;
|
||||||
case(ErrorCode::ServerDockerFailedError): errorMessage = QObject::tr("Server error: Docker failed"); break;
|
case(ErrorCode::ServerDockerFailedError): errorMessage = QObject::tr("Server error: Docker failed"); break;
|
||||||
case(ErrorCode::ServerCancelInstallation): errorMessage = QObject::tr("Installation canceled by user"); break;
|
case(ErrorCode::ServerCancelInstallation): errorMessage = QObject::tr("Installation canceled by user"); break;
|
||||||
case(ErrorCode::ServerUserNotInSudo): errorMessage = QObject::tr("The user is not a member of the sudo group"); break;
|
case(ErrorCode::ServerUserNotInSudo): errorMessage = QObject::tr("The user does not have permission to use sudo"); break;
|
||||||
case(ErrorCode::ServerPacketManagerError): errorMessage = QObject::tr("Server error: Package manager error"); break;
|
case(ErrorCode::ServerPacketManagerError): errorMessage = QObject::tr("Server error: Packet manager error"); break;
|
||||||
case(ErrorCode::ServerSudoPackageIsNotPreinstalled): errorMessage = QObject::tr("The sudo package is not pre-installed on the server"); break;
|
|
||||||
case(ErrorCode::ServerUserDirectoryNotAccessible): errorMessage = QObject::tr("The server user's home directory is not accessible"); break;
|
|
||||||
case(ErrorCode::ServerUserNotAllowedInSudoers): errorMessage = QObject::tr("Action not allowed in sudoers"); break;
|
|
||||||
case(ErrorCode::ServerUserPasswordRequired): errorMessage = QObject::tr("The user's password is required"); break;
|
|
||||||
case(ErrorCode::ServerDockerOnCgroupsV2): errorMessage = QObject::tr("Docker error: runc doesn't work on cgroups v2"); break;
|
|
||||||
case(ErrorCode::ServerCgroupMountpoint): errorMessage = QObject::tr("Server error: cgroup mountpoint does not exist"); break;
|
|
||||||
|
|
||||||
// Libssh errors
|
// Libssh errors
|
||||||
case(ErrorCode::SshRequestDeniedError): errorMessage = QObject::tr("SSH request was denied"); break;
|
case(ErrorCode::SshRequestDeniedError): errorMessage = QObject::tr("SSH request was denied"); break;
|
||||||
|
|||||||
@@ -12,7 +12,6 @@
|
|||||||
#include <winsock.h>
|
#include <winsock.h>
|
||||||
#include <QNetworkInterface>
|
#include <QNetworkInterface>
|
||||||
#include "qendian.h"
|
#include "qendian.h"
|
||||||
#include <QSettings>
|
|
||||||
#endif
|
#endif
|
||||||
#ifdef Q_OS_LINUX
|
#ifdef Q_OS_LINUX
|
||||||
#include <arpa/inet.h>
|
#include <arpa/inet.h>
|
||||||
@@ -186,17 +185,6 @@ int NetworkUtilities::AdapterIndexTo(const QHostAddress& dst) {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool NetworkUtilities::checkIpv6Enabled() {
|
|
||||||
#ifdef Q_OS_WIN
|
|
||||||
QSettings RegHLM("HKEY_LOCAL_MACHINE\\SYSTEM\\CurrentControlSet\\Services\\Tcpip6\\Parameters",
|
|
||||||
QSettings::NativeFormat);
|
|
||||||
int ret = RegHLM.value("DisabledComponents", 0).toInt();
|
|
||||||
qDebug() << "Check for Windows disabled IPv6 return " << ret;
|
|
||||||
return (ret != 255);
|
|
||||||
#endif
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef Q_OS_WIN
|
#ifdef Q_OS_WIN
|
||||||
DWORD GetAdaptersAddressesWrapper(const ULONG Family,
|
DWORD GetAdaptersAddressesWrapper(const ULONG Family,
|
||||||
const ULONG Flags,
|
const ULONG Flags,
|
||||||
|
|||||||
@@ -16,7 +16,6 @@ public:
|
|||||||
static QString getStringBetween(const QString &s, const QString &a, const QString &b);
|
static QString getStringBetween(const QString &s, const QString &a, const QString &b);
|
||||||
static bool checkIPv4Format(const QString &ip);
|
static bool checkIPv4Format(const QString &ip);
|
||||||
static bool checkIpSubnetFormat(const QString &ip);
|
static bool checkIpSubnetFormat(const QString &ip);
|
||||||
static bool checkIpv6Enabled();
|
|
||||||
static QString getGatewayAndIface();
|
static QString getGatewayAndIface();
|
||||||
// Returns the Interface Index that could Route to dst
|
// Returns the Interface Index that could Route to dst
|
||||||
static int AdapterIndexTo(const QHostAddress& dst);
|
static int AdapterIndexTo(const QHostAddress& dst);
|
||||||
@@ -30,6 +29,7 @@ public:
|
|||||||
|
|
||||||
static QString netMaskFromIpWithSubnet(const QString ip);
|
static QString netMaskFromIpWithSubnet(const QString ip);
|
||||||
static QString ipAddressFromIpWithSubnet(const QString ip);
|
static QString ipAddressFromIpWithSubnet(const QString ip);
|
||||||
|
|
||||||
static QStringList summarizeRoutes(const QStringList &ips, const QString cidr);
|
static QStringList summarizeRoutes(const QStringList &ips, const QString cidr);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -371,9 +371,6 @@ bool Daemon::parseConfig(const QJsonObject& obj, InterfaceConfig& config) {
|
|||||||
if (!parseStringList(obj, "vpnDisabledApps", config.m_vpnDisabledApps)) {
|
if (!parseStringList(obj, "vpnDisabledApps", config.m_vpnDisabledApps)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (!parseStringList(obj, "allowedDnsServers", config.m_allowedDnsServers)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
config.m_killSwitchEnabled = QVariant(obj.value("killSwitchOption").toString()).toBool();
|
config.m_killSwitchEnabled = QVariant(obj.value("killSwitchOption").toString()).toBool();
|
||||||
|
|
||||||
|
|||||||
@@ -48,13 +48,6 @@ QJsonObject InterfaceConfig::toJson() const {
|
|||||||
}
|
}
|
||||||
json.insert("excludedAddresses", jsExcludedAddresses);
|
json.insert("excludedAddresses", jsExcludedAddresses);
|
||||||
|
|
||||||
|
|
||||||
QJsonArray jsAllowedDnsServers;
|
|
||||||
for (const QString& i : m_allowedDnsServers) {
|
|
||||||
jsAllowedDnsServers.append(QJsonValue(i));
|
|
||||||
}
|
|
||||||
json.insert("allowedDnsServers", jsAllowedDnsServers);
|
|
||||||
|
|
||||||
QJsonArray disabledApps;
|
QJsonArray disabledApps;
|
||||||
for (const QString& i : m_vpnDisabledApps) {
|
for (const QString& i : m_vpnDisabledApps) {
|
||||||
disabledApps.append(QJsonValue(i));
|
disabledApps.append(QJsonValue(i));
|
||||||
|
|||||||
@@ -6,7 +6,6 @@
|
|||||||
#define INTERFACECONFIG_H
|
#define INTERFACECONFIG_H
|
||||||
|
|
||||||
#include <QList>
|
#include <QList>
|
||||||
#include <QMap>
|
|
||||||
#include <QString>
|
#include <QString>
|
||||||
|
|
||||||
#include "ipaddress.h"
|
#include "ipaddress.h"
|
||||||
@@ -38,7 +37,6 @@ class InterfaceConfig {
|
|||||||
QList<IPAddress> m_allowedIPAddressRanges;
|
QList<IPAddress> m_allowedIPAddressRanges;
|
||||||
QStringList m_excludedAddresses;
|
QStringList m_excludedAddresses;
|
||||||
QStringList m_vpnDisabledApps;
|
QStringList m_vpnDisabledApps;
|
||||||
QStringList m_allowedDnsServers;
|
|
||||||
bool m_killSwitchEnabled;
|
bool m_killSwitchEnabled;
|
||||||
#if defined(MZ_ANDROID) || defined(MZ_IOS)
|
#if defined(MZ_ANDROID) || defined(MZ_IOS)
|
||||||
QString m_installationId;
|
QString m_installationId;
|
||||||
|
|||||||
@@ -123,7 +123,6 @@ void LocalSocketController::activate(const QJsonObject &rawConfig) {
|
|||||||
|
|
||||||
int appSplitTunnelType = rawConfig.value(amnezia::config_key::appSplitTunnelType).toInt();
|
int appSplitTunnelType = rawConfig.value(amnezia::config_key::appSplitTunnelType).toInt();
|
||||||
QJsonArray splitTunnelApps = rawConfig.value(amnezia::config_key::splitTunnelApps).toArray();
|
QJsonArray splitTunnelApps = rawConfig.value(amnezia::config_key::splitTunnelApps).toArray();
|
||||||
QJsonArray allowedDns = rawConfig.value(amnezia::config_key::allowedDnsServers).toArray();
|
|
||||||
|
|
||||||
QJsonObject wgConfig = rawConfig.value(protocolName + "_config_data").toObject();
|
QJsonObject wgConfig = rawConfig.value(protocolName + "_config_data").toObject();
|
||||||
|
|
||||||
@@ -227,8 +226,6 @@ void LocalSocketController::activate(const QJsonObject &rawConfig) {
|
|||||||
|
|
||||||
json.insert("vpnDisabledApps", splitTunnelApps);
|
json.insert("vpnDisabledApps", splitTunnelApps);
|
||||||
|
|
||||||
json.insert("allowedDnsServers", allowedDns);
|
|
||||||
|
|
||||||
json.insert(amnezia::config_key::killSwitchOption, rawConfig.value(amnezia::config_key::killSwitchOption));
|
json.insert(amnezia::config_key::killSwitchOption, rawConfig.value(amnezia::config_key::killSwitchOption));
|
||||||
|
|
||||||
if (protocolName == amnezia::config_key::awg) {
|
if (protocolName == amnezia::config_key::awg) {
|
||||||
|
|||||||
@@ -31,9 +31,7 @@ IPUtilsLinux::~IPUtilsLinux() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool IPUtilsLinux::addInterfaceIPs(const InterfaceConfig& config) {
|
bool IPUtilsLinux::addInterfaceIPs(const InterfaceConfig& config) {
|
||||||
bool ret = addIP4AddressToDevice(config);
|
return addIP4AddressToDevice(config) && addIP6AddressToDevice(config);
|
||||||
addIP6AddressToDevice(config);
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool IPUtilsLinux::setMTUAndUp(const InterfaceConfig& config) {
|
bool IPUtilsLinux::setMTUAndUp(const InterfaceConfig& config) {
|
||||||
|
|||||||
@@ -455,6 +455,9 @@ void LinuxFirewall::updateDNSServers(const QStringList& servers)
|
|||||||
|
|
||||||
void LinuxFirewall::updateAllowNets(const QStringList& servers)
|
void LinuxFirewall::updateAllowNets(const QStringList& servers)
|
||||||
{
|
{
|
||||||
|
static QStringList existingServers {};
|
||||||
|
|
||||||
|
existingServers = servers;
|
||||||
execute(QStringLiteral("iptables -F %1.110.allowNets").arg(kAnchorName));
|
execute(QStringLiteral("iptables -F %1.110.allowNets").arg(kAnchorName));
|
||||||
for (const QString& rule : getAllowRule(servers))
|
for (const QString& rule : getAllowRule(servers))
|
||||||
execute(QStringLiteral("iptables -A %1.110.allowNets %2").arg(kAnchorName, rule));
|
execute(QStringLiteral("iptables -A %1.110.allowNets %2").arg(kAnchorName, rule));
|
||||||
|
|||||||
@@ -17,8 +17,6 @@
|
|||||||
#include "leakdetector.h"
|
#include "leakdetector.h"
|
||||||
#include "logger.h"
|
#include "logger.h"
|
||||||
|
|
||||||
#include "killswitch.h"
|
|
||||||
|
|
||||||
constexpr const int WG_TUN_PROC_TIMEOUT = 5000;
|
constexpr const int WG_TUN_PROC_TIMEOUT = 5000;
|
||||||
constexpr const char* WG_RUNTIME_DIR = "/var/run/amneziawg";
|
constexpr const char* WG_RUNTIME_DIR = "/var/run/amneziawg";
|
||||||
|
|
||||||
@@ -184,7 +182,7 @@ bool WireguardUtilsLinux::deleteInterface() {
|
|||||||
QFile::remove(wgRuntimeDir.filePath(QString(WG_INTERFACE) + ".name"));
|
QFile::remove(wgRuntimeDir.filePath(QString(WG_INTERFACE) + ".name"));
|
||||||
|
|
||||||
// double-check + ensure our firewall is installed and enabled
|
// double-check + ensure our firewall is installed and enabled
|
||||||
KillSwitch::instance()->disableKillSwitch();
|
LinuxFirewall::uninstall();
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -16,8 +16,6 @@
|
|||||||
#include "leakdetector.h"
|
#include "leakdetector.h"
|
||||||
#include "logger.h"
|
#include "logger.h"
|
||||||
|
|
||||||
#include "killswitch.h"
|
|
||||||
|
|
||||||
constexpr const int WG_TUN_PROC_TIMEOUT = 5000;
|
constexpr const int WG_TUN_PROC_TIMEOUT = 5000;
|
||||||
constexpr const char* WG_RUNTIME_DIR = "/var/run/amneziawg";
|
constexpr const char* WG_RUNTIME_DIR = "/var/run/amneziawg";
|
||||||
|
|
||||||
@@ -182,7 +180,7 @@ bool WireguardUtilsMacos::deleteInterface() {
|
|||||||
QFile::remove(wgRuntimeDir.filePath(QString(WG_INTERFACE) + ".name"));
|
QFile::remove(wgRuntimeDir.filePath(QString(WG_INTERFACE) + ".name"));
|
||||||
|
|
||||||
// double-check + ensure our firewall is installed and enabled
|
// double-check + ensure our firewall is installed and enabled
|
||||||
KillSwitch::instance()->disableKillSwitch();
|
MacOSFirewall::uninstall();
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,8 +29,6 @@
|
|||||||
#include "logger.h"
|
#include "logger.h"
|
||||||
#include "platforms/windows/windowsutils.h"
|
#include "platforms/windows/windowsutils.h"
|
||||||
|
|
||||||
#include "killswitch.h"
|
|
||||||
|
|
||||||
#define IPV6_ADDRESS_SIZE 16
|
#define IPV6_ADDRESS_SIZE 16
|
||||||
|
|
||||||
// ID for the Firewall Sublayer
|
// ID for the Firewall Sublayer
|
||||||
@@ -182,29 +180,16 @@ bool WindowsFirewall::enableInterface(int vpnAdapterIndex) {
|
|||||||
} \
|
} \
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.info() << "Enabling Killswitch Using Adapter:" << vpnAdapterIndex;
|
logger.info() << "Enabling firewall Using Adapter:" << vpnAdapterIndex;
|
||||||
if (vpnAdapterIndex < 0)
|
|
||||||
{
|
|
||||||
IPAddress allv4("0.0.0.0/0");
|
|
||||||
if (!blockTrafficTo(allv4, MED_WEIGHT,
|
|
||||||
"Block Internet", "killswitch")) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
IPAddress allv6("::/0");
|
|
||||||
if (!blockTrafficTo(allv6, MED_WEIGHT,
|
|
||||||
"Block Internet", "killswitch")) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
} else
|
|
||||||
FW_OK(allowTrafficOfAdapter(vpnAdapterIndex, MED_WEIGHT,
|
FW_OK(allowTrafficOfAdapter(vpnAdapterIndex, MED_WEIGHT,
|
||||||
"Allow usage of VPN Adapter"));
|
"Allow usage of VPN Adapter"));
|
||||||
FW_OK(allowDHCPTraffic(MED_WEIGHT, "Allow DHCP Traffic"));
|
FW_OK(allowDHCPTraffic(MED_WEIGHT, "Allow DHCP Traffic"));
|
||||||
FW_OK(allowHyperVTraffic(MAX_WEIGHT, "Allow Hyper-V Traffic"));
|
FW_OK(allowHyperVTraffic(MED_WEIGHT, "Allow Hyper-V Traffic"));
|
||||||
FW_OK(allowTrafficForAppOnAll(getCurrentPath(), MAX_WEIGHT,
|
FW_OK(allowTrafficForAppOnAll(getCurrentPath(), MAX_WEIGHT,
|
||||||
"Allow all for AmneziaVPN.exe"));
|
"Allow all for AmneziaVPN.exe"));
|
||||||
FW_OK(blockTrafficOnPort(53, MED_WEIGHT, "Block all DNS"));
|
FW_OK(blockTrafficOnPort(53, MED_WEIGHT, "Block all DNS"));
|
||||||
FW_OK(allowLoopbackTraffic(MED_WEIGHT,
|
FW_OK(
|
||||||
"Allow Loopback traffic on device %1"));
|
allowLoopbackTraffic(MED_WEIGHT, "Allow Loopback traffic on device %1"));
|
||||||
|
|
||||||
logger.debug() << "Killswitch on! Rules:" << m_activeRules.length();
|
logger.debug() << "Killswitch on! Rules:" << m_activeRules.length();
|
||||||
return true;
|
return true;
|
||||||
@@ -241,37 +226,6 @@ bool WindowsFirewall::enableLanBypass(const QList<IPAddress>& ranges) {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Allow unprotected traffic sent to the following address ranges.
|
|
||||||
bool WindowsFirewall::allowTrafficRange(const QStringList& ranges) {
|
|
||||||
// Start the firewall transaction
|
|
||||||
auto result = FwpmTransactionBegin(m_sessionHandle, NULL);
|
|
||||||
if (result != ERROR_SUCCESS) {
|
|
||||||
disableKillSwitch();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
auto cleanup = qScopeGuard([&] {
|
|
||||||
FwpmTransactionAbort0(m_sessionHandle);
|
|
||||||
disableKillSwitch();
|
|
||||||
});
|
|
||||||
|
|
||||||
for (const QString& addr : ranges) {
|
|
||||||
logger.debug() << "Allow killswitch exclude: " << addr;
|
|
||||||
if (!allowTrafficTo(QHostAddress(addr), LOW_WEIGHT + 1, "Allow killswitch bypass traffic")) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
result = FwpmTransactionCommit0(m_sessionHandle);
|
|
||||||
if (result != ERROR_SUCCESS) {
|
|
||||||
logger.error() << "FwpmTransactionCommit0 failed with error:" << result;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
cleanup.dismiss();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool WindowsFirewall::enablePeerTraffic(const InterfaceConfig& config) {
|
bool WindowsFirewall::enablePeerTraffic(const InterfaceConfig& config) {
|
||||||
// Start the firewall transaction
|
// Start the firewall transaction
|
||||||
auto result = FwpmTransactionBegin(m_sessionHandle, NULL);
|
auto result = FwpmTransactionBegin(m_sessionHandle, NULL);
|
||||||
@@ -308,20 +262,12 @@ bool WindowsFirewall::enablePeerTraffic(const InterfaceConfig& config) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const QString& dns : config.m_allowedDnsServers) {
|
|
||||||
logger.debug() << "Allow DNS: " << dns;
|
|
||||||
if (!allowTrafficTo(QHostAddress(dns), 53, HIGH_WEIGHT,
|
|
||||||
"Allow DNS-Server", config.m_serverPublicKey)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!config.m_excludedAddresses.empty()) {
|
if (!config.m_excludedAddresses.empty()) {
|
||||||
for (const QString& i : config.m_excludedAddresses) {
|
for (const QString& i : config.m_excludedAddresses) {
|
||||||
logger.debug() << "excludedAddresses range: " << i;
|
logger.debug() << "excludedAddresses range: " << i;
|
||||||
|
|
||||||
if (!allowTrafficTo(i, HIGH_WEIGHT,
|
if (!allowTrafficTo(i, HIGH_WEIGHT,
|
||||||
"Allow Ecxlude route", config.m_serverPublicKey)) {
|
"Allow Ecxlude route", config.m_serverPublicKey)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -367,41 +313,37 @@ bool WindowsFirewall::disablePeerTraffic(const QString& pubkey) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool WindowsFirewall::disableKillSwitch() {
|
bool WindowsFirewall::disableKillSwitch() {
|
||||||
return KillSwitch::instance()->disableKillSwitch();
|
auto result = FwpmTransactionBegin(m_sessionHandle, NULL);
|
||||||
}
|
auto cleanup = qScopeGuard([&] {
|
||||||
|
|
||||||
bool WindowsFirewall::allowAllTraffic() {
|
|
||||||
auto result = FwpmTransactionBegin(m_sessionHandle, NULL);
|
|
||||||
auto cleanup = qScopeGuard([&] {
|
|
||||||
if (result != ERROR_SUCCESS) {
|
|
||||||
FwpmTransactionAbort0(m_sessionHandle);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
if (result != ERROR_SUCCESS) {
|
if (result != ERROR_SUCCESS) {
|
||||||
logger.error() << "FwpmTransactionBegin0 failed. Return value:.\n"
|
FwpmTransactionAbort0(m_sessionHandle);
|
||||||
<< result;
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
});
|
||||||
|
if (result != ERROR_SUCCESS) {
|
||||||
|
logger.error() << "FwpmTransactionBegin0 failed. Return value:.\n"
|
||||||
|
<< result;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
for (const auto& filterID : m_peerRules.values()) {
|
for (const auto& filterID : m_peerRules.values()) {
|
||||||
FwpmFilterDeleteById0(m_sessionHandle, filterID);
|
FwpmFilterDeleteById0(m_sessionHandle, filterID);
|
||||||
}
|
}
|
||||||
|
|
||||||
for (const auto& filterID : qAsConst(m_activeRules)) {
|
for (const auto& filterID : qAsConst(m_activeRules)) {
|
||||||
FwpmFilterDeleteById0(m_sessionHandle, filterID);
|
FwpmFilterDeleteById0(m_sessionHandle, filterID);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Commit!
|
// Commit!
|
||||||
result = FwpmTransactionCommit0(m_sessionHandle);
|
result = FwpmTransactionCommit0(m_sessionHandle);
|
||||||
if (result != ERROR_SUCCESS) {
|
if (result != ERROR_SUCCESS) {
|
||||||
logger.error() << "FwpmTransactionCommit0 failed. Return value:.\n"
|
logger.error() << "FwpmTransactionCommit0 failed. Return value:.\n"
|
||||||
<< result;
|
<< result;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
m_peerRules.clear();
|
m_peerRules.clear();
|
||||||
m_activeRules.clear();
|
m_activeRules.clear();
|
||||||
logger.debug() << "Firewall Disabled!";
|
logger.debug() << "Firewall Disabled!";
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool WindowsFirewall::allowTrafficForAppOnAll(const QString& exePath,
|
bool WindowsFirewall::allowTrafficForAppOnAll(const QString& exePath,
|
||||||
|
|||||||
@@ -43,8 +43,6 @@ class WindowsFirewall final : public QObject {
|
|||||||
bool enablePeerTraffic(const InterfaceConfig& config);
|
bool enablePeerTraffic(const InterfaceConfig& config);
|
||||||
bool disablePeerTraffic(const QString& pubkey);
|
bool disablePeerTraffic(const QString& pubkey);
|
||||||
bool disableKillSwitch();
|
bool disableKillSwitch();
|
||||||
bool allowAllTraffic();
|
|
||||||
bool allowTrafficRange(const QStringList& ranges);
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static bool initSublayer();
|
static bool initSublayer();
|
||||||
|
|||||||
@@ -14,6 +14,8 @@
|
|||||||
|
|
||||||
#include "leakdetector.h"
|
#include "leakdetector.h"
|
||||||
#include "logger.h"
|
#include "logger.h"
|
||||||
|
#include "platforms/windows/windowscommons.h"
|
||||||
|
#include "windowsdaemon.h"
|
||||||
#include "windowsfirewall.h"
|
#include "windowsfirewall.h"
|
||||||
|
|
||||||
#pragma comment(lib, "iphlpapi.lib")
|
#pragma comment(lib, "iphlpapi.lib")
|
||||||
@@ -267,13 +269,6 @@ bool WireguardUtilsWindows::updateRoutePrefix(const IPAddress& prefix) {
|
|||||||
if (result == ERROR_OBJECT_ALREADY_EXISTS) {
|
if (result == ERROR_OBJECT_ALREADY_EXISTS) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Case for ipv6 route with disabled ipv6
|
|
||||||
if (prefix.address().protocol() == QAbstractSocket::IPv6Protocol
|
|
||||||
&& result == ERROR_NOT_FOUND) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (result != NO_ERROR) {
|
if (result != NO_ERROR) {
|
||||||
logger.error() << "Failed to create route to"
|
logger.error() << "Failed to create route to"
|
||||||
<< prefix.toString()
|
<< prefix.toString()
|
||||||
|
|||||||
@@ -171,11 +171,6 @@ ErrorCode OpenVpnProtocol::start()
|
|||||||
return lastError();
|
return lastError();
|
||||||
}
|
}
|
||||||
|
|
||||||
#if defined(Q_OS_LINUX) || defined(Q_OS_MACOS)
|
|
||||||
IpcClient::Interface()->addKillSwitchAllowedRange(QStringList(NetworkUtilities::getIPAddress(
|
|
||||||
m_configData.value(amnezia::config_key::hostName).toString())));
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// Detect default gateway
|
// Detect default gateway
|
||||||
#ifdef Q_OS_MAC
|
#ifdef Q_OS_MAC
|
||||||
QProcess p;
|
QProcess p;
|
||||||
|
|||||||
@@ -95,8 +95,6 @@ namespace amnezia
|
|||||||
constexpr char splitTunnelApps[] = "splitTunnelApps";
|
constexpr char splitTunnelApps[] = "splitTunnelApps";
|
||||||
constexpr char appSplitTunnelType[] = "appSplitTunnelType";
|
constexpr char appSplitTunnelType[] = "appSplitTunnelType";
|
||||||
|
|
||||||
constexpr char allowedDnsServers[] = "allowedDnsServers";
|
|
||||||
|
|
||||||
constexpr char killSwitchOption[] = "killSwitchOption";
|
constexpr char killSwitchOption[] = "killSwitchOption";
|
||||||
|
|
||||||
constexpr char crc[] = "crc";
|
constexpr char crc[] = "crc";
|
||||||
|
|||||||
@@ -129,7 +129,6 @@
|
|||||||
<file>ui/qml/Components/SettingsContainersListView.qml</file>
|
<file>ui/qml/Components/SettingsContainersListView.qml</file>
|
||||||
<file>ui/qml/Components/ShareConnectionDrawer.qml</file>
|
<file>ui/qml/Components/ShareConnectionDrawer.qml</file>
|
||||||
<file>ui/qml/Components/TransportProtoSelector.qml</file>
|
<file>ui/qml/Components/TransportProtoSelector.qml</file>
|
||||||
<file>ui/qml/Components/AddSitePanel.qml</file>
|
|
||||||
<file>ui/qml/Config/GlobalConfig.qml</file>
|
<file>ui/qml/Config/GlobalConfig.qml</file>
|
||||||
<file>ui/qml/Config/qmldir</file>
|
<file>ui/qml/Config/qmldir</file>
|
||||||
<file>ui/qml/Controls2/BackButtonType.qml</file>
|
<file>ui/qml/Controls2/BackButtonType.qml</file>
|
||||||
@@ -144,9 +143,7 @@
|
|||||||
<file>ui/qml/Controls2/DropDownType.qml</file>
|
<file>ui/qml/Controls2/DropDownType.qml</file>
|
||||||
<file>ui/qml/Controls2/FlickableType.qml</file>
|
<file>ui/qml/Controls2/FlickableType.qml</file>
|
||||||
<file>ui/qml/Controls2/Header2Type.qml</file>
|
<file>ui/qml/Controls2/Header2Type.qml</file>
|
||||||
<file>ui/qml/Controls2/BaseHeaderType.qml</file>
|
<file>ui/qml/Controls2/HeaderType.qml</file>
|
||||||
<file>ui/qml/Controls2/HeaderTypeWithButton.qml</file>
|
|
||||||
<file>ui/qml/Controls2/HeaderTypeWithSwitcher.qml</file>
|
|
||||||
<file>ui/qml/Controls2/HorizontalRadioButton.qml</file>
|
<file>ui/qml/Controls2/HorizontalRadioButton.qml</file>
|
||||||
<file>ui/qml/Controls2/ImageButtonType.qml</file>
|
<file>ui/qml/Controls2/ImageButtonType.qml</file>
|
||||||
<file>ui/qml/Controls2/LabelWithButtonType.qml</file>
|
<file>ui/qml/Controls2/LabelWithButtonType.qml</file>
|
||||||
@@ -202,8 +199,6 @@
|
|||||||
<file>ui/qml/Pages2/PageSettingsBackup.qml</file>
|
<file>ui/qml/Pages2/PageSettingsBackup.qml</file>
|
||||||
<file>ui/qml/Pages2/PageSettingsConnection.qml</file>
|
<file>ui/qml/Pages2/PageSettingsConnection.qml</file>
|
||||||
<file>ui/qml/Pages2/PageSettingsDns.qml</file>
|
<file>ui/qml/Pages2/PageSettingsDns.qml</file>
|
||||||
<file>ui/qml/Pages2/PageSettingsKillSwitch.qml</file>
|
|
||||||
<file>ui/qml/Pages2/PageSettingsKillSwitchExceptions.qml</file>
|
|
||||||
<file>ui/qml/Pages2/PageSettingsLogging.qml</file>
|
<file>ui/qml/Pages2/PageSettingsLogging.qml</file>
|
||||||
<file>ui/qml/Pages2/PageSettingsServerData.qml</file>
|
<file>ui/qml/Pages2/PageSettingsServerData.qml</file>
|
||||||
<file>ui/qml/Pages2/PageSettingsServerInfo.qml</file>
|
<file>ui/qml/Pages2/PageSettingsServerInfo.qml</file>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
#include "secure_qsettings.h"
|
#include "secure_qsettings.h"
|
||||||
|
|
||||||
#include "../client/3rd/QSimpleCrypto/src/include/QAead.h"
|
#include "QAead.h"
|
||||||
#include "../client/3rd/QSimpleCrypto/src/include/QBlockCipher.h"
|
#include "QBlockCipher.h"
|
||||||
#include "utilities.h"
|
#include "utilities.h"
|
||||||
#include <QDataStream>
|
#include <QDataStream>
|
||||||
#include <QDebug>
|
#include <QDebug>
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
#include <QObject>
|
#include <QObject>
|
||||||
#include <QSettings>
|
#include <QSettings>
|
||||||
|
|
||||||
#include "../client/3rd/qtkeychain/qtkeychain/keychain.h"
|
#include "keychain.h"
|
||||||
|
|
||||||
class SecureQSettings : public QObject
|
class SecureQSettings : public QObject
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,13 +1,2 @@
|
|||||||
if which apt-get > /dev/null 2>&1; then pm=$(which apt-get); opt="--version";\
|
CUR_USER=$(whoami);\
|
||||||
elif which dnf > /dev/null 2>&1; then pm=$(which dnf); opt="--version";\
|
groups $CUR_USER
|
||||||
elif which yum > /dev/null 2>&1; then pm=$(which yum); opt="--version";\
|
|
||||||
elif which pacman > /dev/null 2>&1; then pm=$(which pacman); opt="--version";\
|
|
||||||
else pm="uname"; opt="-a";\
|
|
||||||
fi;\
|
|
||||||
CUR_USER=$(whoami 2>/dev/null || echo $HOME | sed 's/.*\///');\
|
|
||||||
echo $LANG | grep -qE '^(en_US.UTF-8|C.UTF-8|C)$' || export LC_ALL=C;\
|
|
||||||
sudo -K;\
|
|
||||||
cd ~;\
|
|
||||||
if [ "$CUR_USER" = "root" ] || ( groups "$CUR_USER" | grep -E '\<(sudo|wheel)\>' ); then \
|
|
||||||
sudo -nu $CUR_USER $pm $opt > /dev/null; sudo -n $pm $opt > /dev/null;\
|
|
||||||
fi
|
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
CUR_USER=$(whoami 2>/dev/null || echo $HOME | sed 's/.*\///');\
|
CUR_USER=$(whoami);\
|
||||||
sudo mkdir -p $DOCKERFILE_FOLDER;\
|
sudo mkdir -p $DOCKERFILE_FOLDER;\
|
||||||
sudo chown $CUR_USER $DOCKERFILE_FOLDER;\
|
sudo chown $CUR_USER $DOCKERFILE_FOLDER;\
|
||||||
if ! sudo docker network ls | grep -q amnezia-dns-net; then sudo docker network create \
|
if ! sudo docker network ls | grep -q amnezia-dns-net; then sudo docker network create \
|
||||||
|
|||||||
@@ -443,16 +443,6 @@ void Settings::setKillSwitchEnabled(bool enabled)
|
|||||||
setValue("Conf/killSwitchEnabled", enabled);
|
setValue("Conf/killSwitchEnabled", enabled);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Settings::isStrictKillSwitchEnabled() const
|
|
||||||
{
|
|
||||||
return value("Conf/strictKillSwitchEnabled", false).toBool();
|
|
||||||
}
|
|
||||||
|
|
||||||
void Settings::setStrictKillSwitchEnabled(bool enabled)
|
|
||||||
{
|
|
||||||
setValue("Conf/strictKillSwitchEnabled", enabled);
|
|
||||||
}
|
|
||||||
|
|
||||||
QString Settings::getInstallationUuid(const bool needCreate)
|
QString Settings::getInstallationUuid(const bool needCreate)
|
||||||
{
|
{
|
||||||
auto uuid = value("Conf/installationUuid", "").toString();
|
auto uuid = value("Conf/installationUuid", "").toString();
|
||||||
@@ -558,13 +548,3 @@ void Settings::disableHomeAdLabel()
|
|||||||
{
|
{
|
||||||
setValue("Conf/homeAdLabelVisible", false);
|
setValue("Conf/homeAdLabelVisible", false);
|
||||||
}
|
}
|
||||||
|
|
||||||
QStringList Settings::allowedDnsServers() const
|
|
||||||
{
|
|
||||||
return value("Conf/allowedDnsServers").toStringList();
|
|
||||||
}
|
|
||||||
|
|
||||||
void Settings::setAllowedDnsServers(const QStringList &servers)
|
|
||||||
{
|
|
||||||
setValue("Conf/allowedDnsServers", servers);
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -213,10 +213,6 @@ public:
|
|||||||
|
|
||||||
bool isKillSwitchEnabled() const;
|
bool isKillSwitchEnabled() const;
|
||||||
void setKillSwitchEnabled(bool enabled);
|
void setKillSwitchEnabled(bool enabled);
|
||||||
|
|
||||||
bool isStrictKillSwitchEnabled() const;
|
|
||||||
void setStrictKillSwitchEnabled(bool enabled);
|
|
||||||
|
|
||||||
QString getInstallationUuid(const bool needCreate);
|
QString getInstallationUuid(const bool needCreate);
|
||||||
|
|
||||||
void resetGatewayEndpoint();
|
void resetGatewayEndpoint();
|
||||||
@@ -229,9 +225,6 @@ public:
|
|||||||
bool isHomeAdLabelVisible();
|
bool isHomeAdLabelVisible();
|
||||||
void disableHomeAdLabel();
|
void disableHomeAdLabel();
|
||||||
|
|
||||||
QStringList allowedDnsServers() const;
|
|
||||||
void setAllowedDnsServers(const QStringList &servers);
|
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void saveLogsChanged(bool enabled);
|
void saveLogsChanged(bool enabled);
|
||||||
void screenshotsEnabledChanged(bool enabled);
|
void screenshotsEnabledChanged(bool enabled);
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@@ -3,14 +3,10 @@
|
|||||||
<TS version="2.1" language="ru_RU">
|
<TS version="2.1" language="ru_RU">
|
||||||
<context>
|
<context>
|
||||||
<name>AdLabel</name>
|
<name>AdLabel</name>
|
||||||
<message>
|
|
||||||
<source>Amnezia Premium - for access to any website</source>
|
|
||||||
<translation type="vanished">Amnezia Premium - для доступа к любым сайтам</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Components/AdLabel.qml" line="57"/>
|
<location filename="../ui/qml/Components/AdLabel.qml" line="57"/>
|
||||||
<source>Amnezia Premium - for access to all websites and online resources</source>
|
<source>Amnezia Premium - for access to any website</source>
|
||||||
<translation>Amnezia Premium - доступ ко всем сайтам и онлайн ресурсам</translation>
|
<translation>Amnezia Premium - для доступа к любым сайтам</translation>
|
||||||
</message>
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
@@ -60,12 +56,12 @@
|
|||||||
<translation>%1 успешно установлен.</translation>
|
<translation>%1 успешно установлен.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/controllers/api/apiConfigsController.cpp" line="257"/>
|
<location filename="../ui/controllers/api/apiConfigsController.cpp" line="258"/>
|
||||||
<source>API config reloaded</source>
|
<source>API config reloaded</source>
|
||||||
<translation>Конфигурация API перезагружена</translation>
|
<translation>Конфигурация API перезагружена</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/controllers/api/apiConfigsController.cpp" line="261"/>
|
<location filename="../ui/controllers/api/apiConfigsController.cpp" line="262"/>
|
||||||
<source>Successfully changed the country of connection to %1</source>
|
<source>Successfully changed the country of connection to %1</source>
|
||||||
<translation>Страна подключения изменена на %1</translation>
|
<translation>Страна подключения изменена на %1</translation>
|
||||||
</message>
|
</message>
|
||||||
@@ -94,8 +90,9 @@
|
|||||||
<translation type="vanished">Amnezia Free - это бесплатный VPN для обхода блокировок в странах с высоким уровнем интернет-цензуры</translation>
|
<translation type="vanished">Amnezia Free - это бесплатный VPN для обхода блокировок в странах с высоким уровнем интернет-цензуры</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
|
<location filename="../ui/models/api/apiServicesModel.cpp" line="68"/>
|
||||||
<source>Amnezia Premium is VPN for comfortable work, downloading large files and watching videos in 8K resolution. Works for any sites with no restrictions. Speed up to %1 MBit/s. Unlimited traffic.</source>
|
<source>Amnezia Premium is VPN for comfortable work, downloading large files and watching videos in 8K resolution. Works for any sites with no restrictions. Speed up to %1 MBit/s. Unlimited traffic.</source>
|
||||||
<translation type="vanished">Amnezia Premium — VPN для комфортной работы, скачивания больших файлов и просмотра видео в высоком разрешении. Скорость до %1 Мбит/с. Безлимитный трафик.</translation>
|
<translation>Amnezia Premium — VPN для комфортной работы, скачивания больших файлов и просмотра видео в высоком разрешении. Скорость до %1 Мбит/с. Безлимитный трафик.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/models/api/apiServicesModel.cpp" line="72"/>
|
<location filename="../ui/models/api/apiServicesModel.cpp" line="72"/>
|
||||||
@@ -103,19 +100,10 @@
|
|||||||
<source>AmneziaFree provides free unlimited access to a basic set of web sites, such as Facebook, Instagram, Twitter (X), Discord, Telegram, and others. YouTube is not included in the free plan.</source>
|
<source>AmneziaFree provides free unlimited access to a basic set of web sites, such as Facebook, Instagram, Twitter (X), Discord, Telegram, and others. YouTube is not included in the free plan.</source>
|
||||||
<translation>AmneziaFree предоставляет бесплатный неограниченный доступ к базовому набору сайтов и приложений, таким как Facebook, Instagram, Twitter (X), Discord, Telegram и другим. YouTube не включен в бесплатный тариф.</translation>
|
<translation>AmneziaFree предоставляет бесплатный неограниченный доступ к базовому набору сайтов и приложений, таким как Facebook, Instagram, Twitter (X), Discord, Telegram и другим. YouTube не включен в бесплатный тариф.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
|
||||||
<source>Amnezia Premium is VPN for comfortable work, downloading large files and watching videos in 8K resolution. Works for any sites with no restrictions.</source>
|
|
||||||
<translation type="vanished">Amnezia Premium — VPN для комфортной работы, скачивания больших файлов и просмотра видео в высоком разрешении. Работает для любых сайтов без ограничений.</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
|
||||||
<location filename="../ui/models/api/apiServicesModel.cpp" line="68"/>
|
|
||||||
<source>Amnezia Premium is classic VPN for seamless work, downloading large files, and watching videos. Access all websites and online resources. Speeds up to %1 Mbps.</source>
|
|
||||||
<translation>Amnezia Premium - это классический VPN для комфортной работы, загрузки больших файлов и просмотра видео. Доступ ко всем сайтам и онлайн ресурсам. Скорость - до %1 Мбит/с.</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/models/api/apiServicesModel.cpp" line="82"/>
|
<location filename="../ui/models/api/apiServicesModel.cpp" line="82"/>
|
||||||
<source>Amnezia Premium is classic VPN for for seamless work, downloading large files, and watching videos. Access all websites and online resources.</source>
|
<source>Amnezia Premium is VPN for comfortable work, downloading large files and watching videos in 8K resolution. Works for any sites with no restrictions.</source>
|
||||||
<translation>Amnezia Premium - это классический VPN для комфортной работы, загрузки больших файлов и просмотра видео. Доступ ко всем сайтам и онлайн ресурсам.</translation>
|
<translation>Amnezia Premium — VPN для комфортной работы, скачивания больших файлов и просмотра видео в высоком разрешении. Работает для любых сайтов без ограничений.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/models/api/apiServicesModel.cpp" line="97"/>
|
<location filename="../ui/models/api/apiServicesModel.cpp" line="97"/>
|
||||||
@@ -352,23 +340,14 @@ Can't be disabled for current server</source>
|
|||||||
<translation type="vanished">Неверный файл конфигурации</translation>
|
<translation type="vanished">Неверный файл конфигурации</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/controllers/importController.cpp" line="650"/>
|
<location filename="../ui/controllers/importController.cpp" line="651"/>
|
||||||
<source>Scanned %1 of %2.</source>
|
<source>Scanned %1 of %2.</source>
|
||||||
<translation>Отсканировано %1 из %2.</translation>
|
<translation>Отсканировано %1 из %2.</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/controllers/importController.cpp" line="685"/>
|
<location filename="../ui/controllers/importController.cpp" line="686"/>
|
||||||
<source>This configuration contains an OpenVPN setup. OpenVPN configurations can include malicious scripts, so only add it if you fully trust the provider of this config. </source>
|
|
||||||
<translation>Эта конфигурация содержит настройки OpenVPN. Конфигурации OpenVPN могут содержать вредоносные скрипты, поэтому добавляйте их только в том случае, если полностью доверяете источнику этого файла. </translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
|
||||||
<location filename="../ui/controllers/importController.cpp" line="689"/>
|
|
||||||
<source><br>In the imported configuration, potentially dangerous lines were found:</source>
|
|
||||||
<translation><br>В импортированной конфигурации обнаружены потенциально опасные строки:</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
|
||||||
<source>In the imported configuration, potentially dangerous lines were found:</source>
|
<source>In the imported configuration, potentially dangerous lines were found:</source>
|
||||||
<translation type="vanished">В импортированной конфигурации были обнаружены потенциально опасные строки:</translation>
|
<translation>В импортированной конфигурации были обнаружены потенциально опасные строки:</translation>
|
||||||
</message>
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
@@ -538,12 +517,12 @@ Already installed containers were found on the server. All installed containers
|
|||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageDevMenu.qml" line="68"/>
|
<location filename="../ui/qml/Pages2/PageDevMenu.qml" line="68"/>
|
||||||
<source>Gateway endpoint</source>
|
<source>Gateway endpoint</source>
|
||||||
<translation>Gateway endpoint</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageDevMenu.qml" line="97"/>
|
<location filename="../ui/qml/Pages2/PageDevMenu.qml" line="97"/>
|
||||||
<source>Dev gateway environment</source>
|
<source>Dev gateway environment</source>
|
||||||
<translation>Dev gateway environment</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
@@ -673,47 +652,47 @@ Already installed containers were found on the server. All installed containers
|
|||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="146"/>
|
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="146"/>
|
||||||
<source>Jc - Junk packet count</source>
|
<source>Jc - Junk packet count</source>
|
||||||
<translation>Jc - Junk packet count</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="168"/>
|
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="168"/>
|
||||||
<source>Jmin - Junk packet minimum size</source>
|
<source>Jmin - Junk packet minimum size</source>
|
||||||
<translation>Jmin - Junk packet minimum size</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="186"/>
|
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="186"/>
|
||||||
<source>Jmax - Junk packet maximum size</source>
|
<source>Jmax - Junk packet maximum size</source>
|
||||||
<translation>Jmax - Junk packet maximum size</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="204"/>
|
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="204"/>
|
||||||
<source>S1 - Init packet junk size</source>
|
<source>S1 - Init packet junk size</source>
|
||||||
<translation>S1 - Init packet junk size</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="228"/>
|
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="228"/>
|
||||||
<source>S2 - Response packet junk size</source>
|
<source>S2 - Response packet junk size</source>
|
||||||
<translation>S2 - Response packet junk size</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="252"/>
|
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="252"/>
|
||||||
<source>H1 - Init packet magic header</source>
|
<source>H1 - Init packet magic header</source>
|
||||||
<translation>H1 - Init packet magic header</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="270"/>
|
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="270"/>
|
||||||
<source>H2 - Response packet magic header</source>
|
<source>H2 - Response packet magic header</source>
|
||||||
<translation>H2 - Response packet magic header</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="288"/>
|
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="288"/>
|
||||||
<source>H4 - Transport packet magic header</source>
|
<source>H4 - Transport packet magic header</source>
|
||||||
<translation>H4 - Transport packet magic header</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="306"/>
|
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="306"/>
|
||||||
<source>H3 - Underload packet magic header</source>
|
<source>H3 - Underload packet magic header</source>
|
||||||
<translation>H3 - Underload packet magic header</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="354"/>
|
<location filename="../ui/qml/Pages2/PageProtocolAwgSettings.qml" line="354"/>
|
||||||
@@ -1482,7 +1461,7 @@ Already installed containers were found on the server. All installed containers
|
|||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettings.qml" line="123"/>
|
<location filename="../ui/qml/Pages2/PageSettings.qml" line="123"/>
|
||||||
<source>Dev console</source>
|
<source>Dev console</source>
|
||||||
<translation>Dev console</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettings.qml" line="142"/>
|
<location filename="../ui/qml/Pages2/PageSettings.qml" line="142"/>
|
||||||
@@ -1609,13 +1588,9 @@ Already installed containers were found on the server. All installed containers
|
|||||||
</context>
|
</context>
|
||||||
<context>
|
<context>
|
||||||
<name>PageSettingsApiDevices</name>
|
<name>PageSettingsApiDevices</name>
|
||||||
<message>
|
|
||||||
<source>Active devices</source>
|
|
||||||
<translation type="vanished">Активные устройства</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsApiDevices.qml" line="45"/>
|
<location filename="../ui/qml/Pages2/PageSettingsApiDevices.qml" line="45"/>
|
||||||
<source>Active Devices</source>
|
<source>Active devices</source>
|
||||||
<translation>Активные устройства</translation>
|
<translation>Активные устройства</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
@@ -1766,14 +1741,10 @@ Already installed containers were found on the server. All installed containers
|
|||||||
<source>Save AmneziaVPN config</source>
|
<source>Save AmneziaVPN config</source>
|
||||||
<translation>Сохранить конфигурацию AmneziaVPN</translation>
|
<translation>Сохранить конфигурацию AmneziaVPN</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
|
||||||
<source>Configuration files</source>
|
|
||||||
<translation type="vanished">Файл конфигурации</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsApiNativeConfigs.qml" line="48"/>
|
<location filename="../ui/qml/Pages2/PageSettingsApiNativeConfigs.qml" line="48"/>
|
||||||
<source>Configuration Files</source>
|
<source>Configuration files</source>
|
||||||
<translation>Файлы конфигурации</translation>
|
<translation>Файл конфигурации</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsApiNativeConfigs.qml" line="49"/>
|
<location filename="../ui/qml/Pages2/PageSettingsApiNativeConfigs.qml" line="49"/>
|
||||||
@@ -1861,8 +1832,9 @@ Already installed containers were found on the server. All installed containers
|
|||||||
<translation type="vanished">Период работы</translation>
|
<translation type="vanished">Период работы</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
|
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="37"/>
|
||||||
<source>Valid until</source>
|
<source>Valid until</source>
|
||||||
<translation type="vanished">Действует до</translation>
|
<translation>Действует до</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<source>Speed</source>
|
<source>Speed</source>
|
||||||
@@ -1873,12 +1845,14 @@ Already installed containers were found on the server. All installed containers
|
|||||||
<translation type="vanished">Скопировано</translation>
|
<translation type="vanished">Скопировано</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
|
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="29"/>
|
||||||
<source>Subscription status</source>
|
<source>Subscription status</source>
|
||||||
<translation type="vanished">Статус подписки</translation>
|
<translation>Статус подписки</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
|
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="45"/>
|
||||||
<source>Active connections</source>
|
<source>Active connections</source>
|
||||||
<translation type="vanished">Активные соединения</translation>
|
<translation>Активные соединения</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="171"/>
|
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="171"/>
|
||||||
@@ -1886,8 +1860,9 @@ Already installed containers were found on the server. All installed containers
|
|||||||
<translation>Сетевые адреса одного или нескольких серверов были обновлены. Пожалуйста, удалите старые конфигурацию и загрузите новые файлы</translation>
|
<translation>Сетевые адреса одного или нескольких серверов были обновлены. Пожалуйста, удалите старые конфигурацию и загрузите новые файлы</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
|
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="186"/>
|
||||||
<source>Subscription key</source>
|
<source>Subscription key</source>
|
||||||
<translation type="vanished">Ключ для подключения</translation>
|
<translation>Ключ для подключения</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="190"/>
|
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="190"/>
|
||||||
@@ -1895,8 +1870,9 @@ Already installed containers were found on the server. All installed containers
|
|||||||
<translation>Ключ подписки Amnezia Premium</translation>
|
<translation>Ключ подписки Amnezia Premium</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
|
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="194"/>
|
||||||
<source>Save VPN key to file</source>
|
<source>Save VPN key to file</source>
|
||||||
<translation type="vanished">Сохранить VPN-ключ в файле</translation>
|
<translation>Сохранить VPN-ключ в файле</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="195"/>
|
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="195"/>
|
||||||
@@ -1904,51 +1880,18 @@ Already installed containers were found on the server. All installed containers
|
|||||||
<translation>Скопировать VPN ключ</translation>
|
<translation>Скопировать VPN ключ</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
|
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="216"/>
|
||||||
<source>Configuration files</source>
|
<source>Configuration files</source>
|
||||||
<translation type="vanished">Файл конфигурации</translation>
|
<translation>Файл конфигурации</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="218"/>
|
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="218"/>
|
||||||
<source>Manage configuration files</source>
|
<source>Manage configuration files</source>
|
||||||
<translation>Управление файлами конфигурации</translation>
|
<translation>Управление файлами конфигурации</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
|
||||||
<source>Active devices</source>
|
|
||||||
<translation type="vanished">Активные устройства</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="29"/>
|
|
||||||
<source>Subscription Status</source>
|
|
||||||
<translation>Статус подписки</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="37"/>
|
|
||||||
<source>Valid Until</source>
|
|
||||||
<translation>Действительна до</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="45"/>
|
|
||||||
<source>Active Connections</source>
|
|
||||||
<translation>Активные соединения</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="186"/>
|
|
||||||
<source>Subscription Key</source>
|
|
||||||
<translation>Ключ для подключения</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="194"/>
|
|
||||||
<source>Save VPN key as a file</source>
|
|
||||||
<translation>Сохранить VPN-ключ в файл</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="216"/>
|
|
||||||
<source>Configuration Files</source>
|
|
||||||
<translation>Файлы конфигурации</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="236"/>
|
<location filename="../ui/qml/Pages2/PageSettingsApiServerInfo.qml" line="236"/>
|
||||||
<source>Active Devices</source>
|
<source>Active devices</source>
|
||||||
<translation>Активные устройства</translation>
|
<translation>Активные устройства</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
@@ -2038,13 +1981,9 @@ Already installed containers were found on the server. All installed containers
|
|||||||
<source>Telegram</source>
|
<source>Telegram</source>
|
||||||
<translation></translation>
|
<translation></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
|
||||||
<source>Email Support</source>
|
|
||||||
<translation type="vanished">Email</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsApiSupport.qml" line="30"/>
|
<location filename="../ui/qml/Pages2/PageSettingsApiSupport.qml" line="30"/>
|
||||||
<source>Email</source>
|
<source>Email Support</source>
|
||||||
<translation>Email</translation>
|
<translation>Email</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
@@ -2393,12 +2332,8 @@ Already installed containers were found on the server. All installed containers
|
|||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsConnection.qml" line="143"/>
|
<location filename="../ui/qml/Pages2/PageSettingsConnection.qml" line="143"/>
|
||||||
<source>Cannot change KillSwitch settings during active connection</source>
|
|
||||||
<translation>Невозможно изменить настройки KillSwitch во время активного подключения</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
|
||||||
<source>Cannot change killSwitch settings during active connection</source>
|
<source>Cannot change killSwitch settings during active connection</source>
|
||||||
<translation type="vanished">Невозможно изменить настройки аварийного выключателя во время активного соединения</translation>
|
<translation>Невозможно изменить настройки аварийного выключателя во время активного соединения</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsConnection.qml" line="86"/>
|
<location filename="../ui/qml/Pages2/PageSettingsConnection.qml" line="86"/>
|
||||||
@@ -2561,12 +2496,12 @@ Already installed containers were found on the server. All installed containers
|
|||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsLogging.qml" line="175"/>
|
<location filename="../ui/qml/Pages2/PageSettingsLogging.qml" line="175"/>
|
||||||
<source>Client logs</source>
|
<source>Client logs</source>
|
||||||
<translation>Логи приложения</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsLogging.qml" line="176"/>
|
<location filename="../ui/qml/Pages2/PageSettingsLogging.qml" line="176"/>
|
||||||
<source>AmneziaVPN logs</source>
|
<source>AmneziaVPN logs</source>
|
||||||
<translation>AmneziaVPN logs</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsLogging.qml" line="142"/>
|
<location filename="../ui/qml/Pages2/PageSettingsLogging.qml" line="142"/>
|
||||||
@@ -2581,12 +2516,12 @@ Already installed containers were found on the server. All installed containers
|
|||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsLogging.qml" line="204"/>
|
<location filename="../ui/qml/Pages2/PageSettingsLogging.qml" line="204"/>
|
||||||
<source>Service logs</source>
|
<source>Service logs</source>
|
||||||
<translation>Логи службы</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsLogging.qml" line="205"/>
|
<location filename="../ui/qml/Pages2/PageSettingsLogging.qml" line="205"/>
|
||||||
<source>AmneziaVPN-service logs</source>
|
<source>AmneziaVPN-service logs</source>
|
||||||
<translation>AmneziaVPN-service logs</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSettingsLogging.qml" line="78"/>
|
<location filename="../ui/qml/Pages2/PageSettingsLogging.qml" line="78"/>
|
||||||
@@ -3071,7 +3006,7 @@ It's okay as long as it's from someone you trust.</source>
|
|||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSetupWizardConfigSource.qml" line="109"/>
|
<location filename="../ui/qml/Pages2/PageSetupWizardConfigSource.qml" line="109"/>
|
||||||
<source>Support tag</source>
|
<source>Support tag</source>
|
||||||
<translation>Support tag</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageSetupWizardConfigSource.qml" line="120"/>
|
<location filename="../ui/qml/Pages2/PageSetupWizardConfigSource.qml" line="120"/>
|
||||||
@@ -3659,7 +3594,7 @@ and will not be shared or disclosed to the Amnezia or any third parties</source>
|
|||||||
<message>
|
<message>
|
||||||
<location filename="../ui/qml/Pages2/PageShare.qml" line="726"/>
|
<location filename="../ui/qml/Pages2/PageShare.qml" line="726"/>
|
||||||
<source>Allowed IPs: %1</source>
|
<source>Allowed IPs: %1</source>
|
||||||
<translation>Разрешенные подсети: %1</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<source>Creation date: </source>
|
<source>Creation date: </source>
|
||||||
@@ -4103,66 +4038,46 @@ and will not be shared or disclosed to the Amnezia or any third parties</source>
|
|||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="23"/>
|
<location filename="../core/errorstrings.cpp" line="23"/>
|
||||||
<source>The user is not a member of the sudo group</source>
|
<source>The user does not have permission to use sudo</source>
|
||||||
<translation>Пользователь не входит в группу sudo</translation>
|
<translation>У пользователя нет прав на использование sudo</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="24"/>
|
<location filename="../core/errorstrings.cpp" line="24"/>
|
||||||
<source>Server error: Package manager error</source>
|
<source>Server error: Packet manager error</source>
|
||||||
<translation>Ошибка сервера: Ошибка менеджера пакетов</translation>
|
<translation>Ошибка сервера: ошибка менеджера пакетов</translation>
|
||||||
</message>
|
|
||||||
<message>
|
|
||||||
<location filename="../core/errorstrings.cpp" line="25"/>
|
|
||||||
<source>The sudo package is not pre-installed on the server</source>
|
|
||||||
<translation>Пакет sudo не установлен на сервере по умолчанию</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
|
||||||
<location filename="../core/errorstrings.cpp" line="26"/>
|
|
||||||
<source>The server user's home directory is not accessible</source>
|
|
||||||
<translation>Домашний каталог пользователя сервера недоступен</translation>
|
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="27"/>
|
<location filename="../core/errorstrings.cpp" line="27"/>
|
||||||
<source>Action not allowed in sudoers</source>
|
|
||||||
<translation>Действие не разрешено в sudoers</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
|
||||||
<location filename="../core/errorstrings.cpp" line="28"/>
|
|
||||||
<source>The user's password is required</source>
|
|
||||||
<translation>Требуется пароль пользователя</translation>
|
|
||||||
</message>
|
|
||||||
<message>
|
|
||||||
<location filename="../core/errorstrings.cpp" line="31"/>
|
|
||||||
<source>SSH request was denied</source>
|
<source>SSH request was denied</source>
|
||||||
<translation>SSH-запрос был отклонён</translation>
|
<translation>SSH-запрос был отклонён</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="32"/>
|
<location filename="../core/errorstrings.cpp" line="28"/>
|
||||||
<source>SSH request was interrupted</source>
|
<source>SSH request was interrupted</source>
|
||||||
<translation>SSH-запрос был прерван</translation>
|
<translation>SSH-запрос был прерван</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="33"/>
|
<location filename="../core/errorstrings.cpp" line="29"/>
|
||||||
<source>SSH internal error</source>
|
<source>SSH internal error</source>
|
||||||
<translation>Внутренняя ошибка SSH</translation>
|
<translation>Внутренняя ошибка SSH</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="34"/>
|
<location filename="../core/errorstrings.cpp" line="30"/>
|
||||||
<source>Invalid private key or invalid passphrase entered</source>
|
<source>Invalid private key or invalid passphrase entered</source>
|
||||||
<translation>Введен неверный закрытый ключ или неверная парольная фраза</translation>
|
<translation>Введен неверный закрытый ключ или неверная парольная фраза</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="35"/>
|
<location filename="../core/errorstrings.cpp" line="31"/>
|
||||||
<source>The selected private key format is not supported, use openssh ED25519 key types or PEM key types</source>
|
<source>The selected private key format is not supported, use openssh ED25519 key types or PEM key types</source>
|
||||||
<translation>Выбранный формат закрытого ключа не поддерживается, используйте типы ключей openssh ED25519 или PEM</translation>
|
<translation>Выбранный формат закрытого ключа не поддерживается, используйте типы ключей openssh ED25519 или PEM</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="36"/>
|
<location filename="../core/errorstrings.cpp" line="32"/>
|
||||||
<source>Timeout connecting to server</source>
|
<source>Timeout connecting to server</source>
|
||||||
<translation>Тайм-аут подключения к серверу</translation>
|
<translation>Тайм-аут подключения к серверу</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="39"/>
|
<location filename="../core/errorstrings.cpp" line="35"/>
|
||||||
<source>SCP error: Generic failure</source>
|
<source>SCP error: Generic failure</source>
|
||||||
<translation>Ошибка SCP: общий сбой</translation>
|
<translation>Ошибка SCP: общий сбой</translation>
|
||||||
</message>
|
</message>
|
||||||
@@ -4219,23 +4134,23 @@ and will not be shared or disclosed to the Amnezia or any third parties</source>
|
|||||||
<translation type="vanished">Sftp error: No media was in remote drive</translation>
|
<translation type="vanished">Sftp error: No media was in remote drive</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="57"/>
|
<location filename="../core/errorstrings.cpp" line="53"/>
|
||||||
<source>The config does not contain any containers and credentials for connecting to the server</source>
|
<source>The config does not contain any containers and credentials for connecting to the server</source>
|
||||||
<translation>Конфигурация не содержит каких-либо контейнеров и учетных данных для подключения к серверу</translation>
|
<translation>Конфигурация не содержит каких-либо контейнеров и учетных данных для подключения к серверу</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="65"/>
|
<location filename="../core/errorstrings.cpp" line="61"/>
|
||||||
<location filename="../core/errorstrings.cpp" line="74"/>
|
<location filename="../core/errorstrings.cpp" line="70"/>
|
||||||
<source>Error when retrieving configuration from API</source>
|
<source>Error when retrieving configuration from API</source>
|
||||||
<translation>Ошибка при получении конфигурации из API</translation>
|
<translation>Ошибка при получении конфигурации из API</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="66"/>
|
<location filename="../core/errorstrings.cpp" line="62"/>
|
||||||
<source>This config has already been added to the application</source>
|
<source>This config has already been added to the application</source>
|
||||||
<translation>Данная конфигурация уже была добавлена в приложение</translation>
|
<translation>Данная конфигурация уже была добавлена в приложение</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="89"/>
|
<location filename="../core/errorstrings.cpp" line="85"/>
|
||||||
<source>ErrorCode: %1. </source>
|
<source>ErrorCode: %1. </source>
|
||||||
<translation>Код ошибки: %1. </translation>
|
<translation>Код ошибки: %1. </translation>
|
||||||
</message>
|
</message>
|
||||||
@@ -4244,139 +4159,139 @@ and will not be shared or disclosed to the Amnezia or any third parties</source>
|
|||||||
<translation type="vanished">Failed to save config to disk</translation>
|
<translation type="vanished">Failed to save config to disk</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="42"/>
|
<location filename="../core/errorstrings.cpp" line="38"/>
|
||||||
<source>OpenVPN config missing</source>
|
<source>OpenVPN config missing</source>
|
||||||
<translation>Отсутствует конфигурация OpenVPN</translation>
|
<translation>Отсутствует конфигурация OpenVPN</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="43"/>
|
<location filename="../core/errorstrings.cpp" line="39"/>
|
||||||
<source>OpenVPN management server error</source>
|
<source>OpenVPN management server error</source>
|
||||||
<translation>Серверная ошибка управлением OpenVPN</translation>
|
<translation>Серверная ошибка управлением OpenVPN</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="46"/>
|
<location filename="../core/errorstrings.cpp" line="42"/>
|
||||||
<source>OpenVPN executable missing</source>
|
<source>OpenVPN executable missing</source>
|
||||||
<translation>Отсутствует исполняемый файл OpenVPN</translation>
|
<translation>Отсутствует исполняемый файл OpenVPN</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="47"/>
|
<location filename="../core/errorstrings.cpp" line="43"/>
|
||||||
<source>Shadowsocks (ss-local) executable missing</source>
|
<source>Shadowsocks (ss-local) executable missing</source>
|
||||||
<translation>Отсутствует исполняемый файл Shadowsocks (ss-local)</translation>
|
<translation>Отсутствует исполняемый файл Shadowsocks (ss-local)</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="48"/>
|
<location filename="../core/errorstrings.cpp" line="44"/>
|
||||||
<source>Cloak (ck-client) executable missing</source>
|
<source>Cloak (ck-client) executable missing</source>
|
||||||
<translation>Отсутствует исполняемый файл Cloak (ck-client)</translation>
|
<translation>Отсутствует исполняемый файл Cloak (ck-client)</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="49"/>
|
<location filename="../core/errorstrings.cpp" line="45"/>
|
||||||
<source>Amnezia helper service error</source>
|
<source>Amnezia helper service error</source>
|
||||||
<translation>Ошибка вспомогательной службы Amnezia</translation>
|
<translation>Ошибка вспомогательной службы Amnezia</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="50"/>
|
<location filename="../core/errorstrings.cpp" line="46"/>
|
||||||
<source>OpenSSL failed</source>
|
<source>OpenSSL failed</source>
|
||||||
<translation>Ошибка OpenSSL</translation>
|
<translation>Ошибка OpenSSL</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="53"/>
|
<location filename="../core/errorstrings.cpp" line="49"/>
|
||||||
<source>Can't connect: another VPN connection is active</source>
|
<source>Can't connect: another VPN connection is active</source>
|
||||||
<translation>Невозможно подключиться: активно другое VPN-соединение</translation>
|
<translation>Невозможно подключиться: активно другое VPN-соединение</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="54"/>
|
<location filename="../core/errorstrings.cpp" line="50"/>
|
||||||
<source>Can't setup OpenVPN TAP network adapter</source>
|
<source>Can't setup OpenVPN TAP network adapter</source>
|
||||||
<translation>Невозможно настроить сетевой адаптер OpenVPN TAP</translation>
|
<translation>Невозможно настроить сетевой адаптер OpenVPN TAP</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="55"/>
|
<location filename="../core/errorstrings.cpp" line="51"/>
|
||||||
<source>VPN pool error: no available addresses</source>
|
<source>VPN pool error: no available addresses</source>
|
||||||
<translation>Ошибка пула VPN: нет доступных адресов</translation>
|
<translation>Ошибка пула VPN: нет доступных адресов</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="58"/>
|
<location filename="../core/errorstrings.cpp" line="54"/>
|
||||||
<source>Unable to open config file</source>
|
<source>Unable to open config file</source>
|
||||||
<translation>Не удалось открыть файл конфигурации</translation>
|
<translation>Не удалось открыть файл конфигурации</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="59"/>
|
<location filename="../core/errorstrings.cpp" line="55"/>
|
||||||
<source>VPN Protocols is not installed.
|
<source>VPN Protocols is not installed.
|
||||||
Please install VPN container at first</source>
|
Please install VPN container at first</source>
|
||||||
<translation>VPN-протоколы не установлены.
|
<translation>VPN-протоколы не установлены.
|
||||||
Пожалуйста, установите протокол</translation>
|
Пожалуйста, установите протокол</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="62"/>
|
<location filename="../core/errorstrings.cpp" line="58"/>
|
||||||
<source>VPN connection error</source>
|
<source>VPN connection error</source>
|
||||||
<translation>Ошибка VPN-соединения</translation>
|
<translation>Ошибка VPN-соединения</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="67"/>
|
<location filename="../core/errorstrings.cpp" line="63"/>
|
||||||
<source>In the response from the server, an empty config was received</source>
|
<source>In the response from the server, an empty config was received</source>
|
||||||
<translation>В ответе от сервера была получена пустая конфигурация</translation>
|
<translation>В ответе от сервера была получена пустая конфигурация</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="68"/>
|
<location filename="../core/errorstrings.cpp" line="64"/>
|
||||||
<source>SSL error occurred</source>
|
<source>SSL error occurred</source>
|
||||||
<translation>Произошла ошибка SSL</translation>
|
<translation>Произошла ошибка SSL</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="69"/>
|
<location filename="../core/errorstrings.cpp" line="65"/>
|
||||||
<source>Server response timeout on api request</source>
|
<source>Server response timeout on api request</source>
|
||||||
<translation>Тайм-аут ответа сервера на запрос API</translation>
|
<translation>Тайм-аут ответа сервера на запрос API</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="70"/>
|
<location filename="../core/errorstrings.cpp" line="66"/>
|
||||||
<source>Missing AGW public key</source>
|
<source>Missing AGW public key</source>
|
||||||
<translation>Отсутствует публичный ключ AGW</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="71"/>
|
<location filename="../core/errorstrings.cpp" line="67"/>
|
||||||
<source>Failed to decrypt response payload</source>
|
<source>Failed to decrypt response payload</source>
|
||||||
<translation></translation>
|
<translation></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="72"/>
|
<location filename="../core/errorstrings.cpp" line="68"/>
|
||||||
<source>Missing list of available services</source>
|
<source>Missing list of available services</source>
|
||||||
<translation>Отсутствует список доступных сервисов</translation>
|
<translation type="unfinished"></translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="73"/>
|
<location filename="../core/errorstrings.cpp" line="69"/>
|
||||||
<source>The limit of allowed configurations per subscription has been exceeded</source>
|
<source>The limit of allowed configurations per subscription has been exceeded</source>
|
||||||
<translation>Превышен лимит разрешенных конфигураций для одной подписки</translation>
|
<translation>Превышен лимит разрешенных конфигураций для одной подписки</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="77"/>
|
<location filename="../core/errorstrings.cpp" line="73"/>
|
||||||
<source>QFile error: The file could not be opened</source>
|
<source>QFile error: The file could not be opened</source>
|
||||||
<translation>Ошибка QFile: не удалось открыть файл</translation>
|
<translation>Ошибка QFile: не удалось открыть файл</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="78"/>
|
<location filename="../core/errorstrings.cpp" line="74"/>
|
||||||
<source>QFile error: An error occurred when reading from the file</source>
|
<source>QFile error: An error occurred when reading from the file</source>
|
||||||
<translation>Ошибка QFile: произошла ошибка при чтении из файла</translation>
|
<translation>Ошибка QFile: произошла ошибка при чтении из файла</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="79"/>
|
<location filename="../core/errorstrings.cpp" line="75"/>
|
||||||
<source>QFile error: The file could not be accessed</source>
|
<source>QFile error: The file could not be accessed</source>
|
||||||
<translation>Ошибка QFile: не удалось получить доступ к файлу</translation>
|
<translation>Ошибка QFile: не удалось получить доступ к файлу</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="80"/>
|
<location filename="../core/errorstrings.cpp" line="76"/>
|
||||||
<source>QFile error: An unspecified error occurred</source>
|
<source>QFile error: An unspecified error occurred</source>
|
||||||
<translation>Ошибка QFile: произошла неизвестная ошибка</translation>
|
<translation>Ошибка QFile: произошла неизвестная ошибка</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="81"/>
|
<location filename="../core/errorstrings.cpp" line="77"/>
|
||||||
<source>QFile error: A fatal error occurred</source>
|
<source>QFile error: A fatal error occurred</source>
|
||||||
<translation>Ошибка QFile: произошла фатальная ошибка</translation>
|
<translation>Ошибка QFile: произошла фатальная ошибка</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="82"/>
|
<location filename="../core/errorstrings.cpp" line="78"/>
|
||||||
<source>QFile error: The operation was aborted</source>
|
<source>QFile error: The operation was aborted</source>
|
||||||
<translation>Ошибка QFile: операция была прервана</translation>
|
<translation>Ошибка QFile: операция была прервана</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../core/errorstrings.cpp" line="86"/>
|
<location filename="../core/errorstrings.cpp" line="82"/>
|
||||||
<source>Internal error</source>
|
<source>Internal error</source>
|
||||||
<translation>Внутренняя ошибка</translation>
|
<translation>Внутренняя ошибка</translation>
|
||||||
</message>
|
</message>
|
||||||
@@ -5138,37 +5053,37 @@ This means that AmneziaWG keeps the fast performance of the original while addin
|
|||||||
<translation>Имя хоста не похоже на IP-адрес или доменное имя</translation>
|
<translation>Имя хоста не похоже на IP-адрес или доменное имя</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/controllers/sitesController.cpp" line="66"/>
|
<location filename="../ui/controllers/sitesController.cpp" line="67"/>
|
||||||
<source>New site added: %1</source>
|
<source>New site added: %1</source>
|
||||||
<translation>Добавлен новый сайт: %1</translation>
|
<translation>Добавлен новый сайт: %1</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/controllers/sitesController.cpp" line="78"/>
|
<location filename="../ui/controllers/sitesController.cpp" line="80"/>
|
||||||
<source>Site removed: %1</source>
|
<source>Site removed: %1</source>
|
||||||
<translation>Сайт удален: %1</translation>
|
<translation>Сайт удален: %1</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/controllers/sitesController.cpp" line="85"/>
|
<location filename="../ui/controllers/sitesController.cpp" line="87"/>
|
||||||
<source>Can't open file: %1</source>
|
<source>Can't open file: %1</source>
|
||||||
<translation>Невозможно открыть файл: %1</translation>
|
<translation>Невозможно открыть файл: %1</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/controllers/sitesController.cpp" line="91"/>
|
<location filename="../ui/controllers/sitesController.cpp" line="93"/>
|
||||||
<source>Failed to parse JSON data from file: %1</source>
|
<source>Failed to parse JSON data from file: %1</source>
|
||||||
<translation>Не удалось разобрать JSON-данные из файла: %1</translation>
|
<translation>Не удалось разобрать JSON-данные из файла: %1</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/controllers/sitesController.cpp" line="96"/>
|
<location filename="../ui/controllers/sitesController.cpp" line="98"/>
|
||||||
<source>The JSON data is not an array in file: %1</source>
|
<source>The JSON data is not an array in file: %1</source>
|
||||||
<translation>JSON-данные не являются массивом в файле: %1</translation>
|
<translation>JSON-данные не являются массивом в файле: %1</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/controllers/sitesController.cpp" line="126"/>
|
<location filename="../ui/controllers/sitesController.cpp" line="129"/>
|
||||||
<source>Import completed</source>
|
<source>Import completed</source>
|
||||||
<translation>Импорт завершен</translation>
|
<translation>Импорт завершен</translation>
|
||||||
</message>
|
</message>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../ui/controllers/sitesController.cpp" line="145"/>
|
<location filename="../ui/controllers/sitesController.cpp" line="148"/>
|
||||||
<source>Export completed</source>
|
<source>Export completed</source>
|
||||||
<translation>Экспорт завершен</translation>
|
<translation>Экспорт завершен</translation>
|
||||||
</message>
|
</message>
|
||||||
@@ -5217,7 +5132,7 @@ This means that AmneziaWG keeps the fast performance of the original while addin
|
|||||||
<context>
|
<context>
|
||||||
<name>VpnConnection</name>
|
<name>VpnConnection</name>
|
||||||
<message>
|
<message>
|
||||||
<location filename="../vpnconnection.cpp" line="421"/>
|
<location filename="../vpnconnection.cpp" line="415"/>
|
||||||
<source>Mbps</source>
|
<source>Mbps</source>
|
||||||
<translation>Мбит/с</translation>
|
<translation>Мбит/с</translation>
|
||||||
</message>
|
</message>
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@@ -1,101 +0,0 @@
|
|||||||
#include "allowedDnsController.h"
|
|
||||||
|
|
||||||
#include <QFile>
|
|
||||||
#include <QStandardPaths>
|
|
||||||
#include <QJsonDocument>
|
|
||||||
#include <QJsonArray>
|
|
||||||
#include <QJsonObject>
|
|
||||||
|
|
||||||
#include "systemController.h"
|
|
||||||
#include "core/networkUtilities.h"
|
|
||||||
#include "core/defs.h"
|
|
||||||
|
|
||||||
AllowedDnsController::AllowedDnsController(const std::shared_ptr<Settings> &settings,
|
|
||||||
const QSharedPointer<AllowedDnsModel> &allowedDnsModel,
|
|
||||||
QObject *parent)
|
|
||||||
: QObject(parent), m_settings(settings), m_allowedDnsModel(allowedDnsModel)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
void AllowedDnsController::addDns(QString ip)
|
|
||||||
{
|
|
||||||
if (ip.isEmpty()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!NetworkUtilities::ipAddressRegExp().match(ip).hasMatch()) {
|
|
||||||
emit errorOccurred(tr("The address does not look like a valid IP address"));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (m_allowedDnsModel->addDns(ip)) {
|
|
||||||
emit finished(tr("New DNS server added: %1").arg(ip));
|
|
||||||
} else {
|
|
||||||
emit errorOccurred(tr("DNS server already exists: %1").arg(ip));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void AllowedDnsController::removeDns(int index)
|
|
||||||
{
|
|
||||||
auto modelIndex = m_allowedDnsModel->index(index);
|
|
||||||
auto ip = m_allowedDnsModel->data(modelIndex, AllowedDnsModel::Roles::IpRole).toString();
|
|
||||||
m_allowedDnsModel->removeDns(modelIndex);
|
|
||||||
|
|
||||||
emit finished(tr("DNS server removed: %1").arg(ip));
|
|
||||||
}
|
|
||||||
|
|
||||||
void AllowedDnsController::importDns(const QString &fileName, bool replaceExisting)
|
|
||||||
{
|
|
||||||
QByteArray jsonData;
|
|
||||||
if (!SystemController::readFile(fileName, jsonData)) {
|
|
||||||
emit errorOccurred(tr("Can't open file: %1").arg(fileName));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
QJsonDocument jsonDocument = QJsonDocument::fromJson(jsonData);
|
|
||||||
if (jsonDocument.isNull()) {
|
|
||||||
emit errorOccurred(tr("Failed to parse JSON data from file: %1").arg(fileName));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!jsonDocument.isArray()) {
|
|
||||||
emit errorOccurred(tr("The JSON data is not an array in file: %1").arg(fileName));
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
auto jsonArray = jsonDocument.array();
|
|
||||||
QStringList dnsServers;
|
|
||||||
|
|
||||||
for (auto jsonValue : jsonArray) {
|
|
||||||
auto ip = jsonValue.toString();
|
|
||||||
|
|
||||||
if (!NetworkUtilities::ipAddressRegExp().match(ip).hasMatch()) {
|
|
||||||
qDebug() << ip << " is not a valid IP address";
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
dnsServers.append(ip);
|
|
||||||
}
|
|
||||||
|
|
||||||
m_allowedDnsModel->addDnsList(dnsServers, replaceExisting);
|
|
||||||
|
|
||||||
emit finished(tr("Import completed"));
|
|
||||||
}
|
|
||||||
|
|
||||||
void AllowedDnsController::exportDns(const QString &fileName)
|
|
||||||
{
|
|
||||||
auto dnsServers = m_allowedDnsModel->getCurrentDnsServers();
|
|
||||||
|
|
||||||
QJsonArray jsonArray;
|
|
||||||
|
|
||||||
for (const auto &ip : dnsServers) {
|
|
||||||
jsonArray.append(ip);
|
|
||||||
}
|
|
||||||
|
|
||||||
QJsonDocument jsonDocument(jsonArray);
|
|
||||||
QByteArray jsonData = jsonDocument.toJson();
|
|
||||||
|
|
||||||
SystemController::saveFile(fileName, jsonData);
|
|
||||||
|
|
||||||
emit finished(tr("Export completed"));
|
|
||||||
}
|
|
||||||
@@ -1,35 +0,0 @@
|
|||||||
#ifndef ALLOWEDDNSCONTROLLER_H
|
|
||||||
#define ALLOWEDDNSCONTROLLER_H
|
|
||||||
|
|
||||||
#include <QObject>
|
|
||||||
|
|
||||||
#include "settings.h"
|
|
||||||
#include "ui/models/allowed_dns_model.h"
|
|
||||||
|
|
||||||
class AllowedDnsController : public QObject
|
|
||||||
{
|
|
||||||
Q_OBJECT
|
|
||||||
public:
|
|
||||||
explicit AllowedDnsController(const std::shared_ptr<Settings> &settings,
|
|
||||||
const QSharedPointer<AllowedDnsModel> &allowedDnsModel,
|
|
||||||
QObject *parent = nullptr);
|
|
||||||
|
|
||||||
public slots:
|
|
||||||
void addDns(QString ip);
|
|
||||||
void removeDns(int index);
|
|
||||||
|
|
||||||
void importDns(const QString &fileName, bool replaceExisting);
|
|
||||||
void exportDns(const QString &fileName);
|
|
||||||
|
|
||||||
signals:
|
|
||||||
void errorOccurred(const QString &errorMessage);
|
|
||||||
void finished(const QString &message);
|
|
||||||
|
|
||||||
void saveFile(const QString &fileName, const QString &data);
|
|
||||||
|
|
||||||
private:
|
|
||||||
std::shared_ptr<Settings> m_settings;
|
|
||||||
QSharedPointer<AllowedDnsModel> m_allowedDnsModel;
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif // ALLOWEDDNSCONTROLLER_H
|
|
||||||
@@ -19,7 +19,7 @@ namespace
|
|||||||
constexpr char cloak[] = "cloak";
|
constexpr char cloak[] = "cloak";
|
||||||
constexpr char awg[] = "awg";
|
constexpr char awg[] = "awg";
|
||||||
|
|
||||||
constexpr char apiEndpoint[] = "api_endpoint";
|
constexpr char apiEdnpoint[] = "api_endpoint";
|
||||||
constexpr char accessToken[] = "api_key";
|
constexpr char accessToken[] = "api_key";
|
||||||
constexpr char certificate[] = "certificate";
|
constexpr char certificate[] = "certificate";
|
||||||
constexpr char publicKey[] = "public_key";
|
constexpr char publicKey[] = "public_key";
|
||||||
@@ -251,6 +251,7 @@ bool ApiConfigsController::updateServiceFromGateway(const int serverIndex, const
|
|||||||
|
|
||||||
newServerConfig.insert(configKey::apiConfig, newApiConfig);
|
newServerConfig.insert(configKey::apiConfig, newApiConfig);
|
||||||
newServerConfig.insert(configKey::authData, authData);
|
newServerConfig.insert(configKey::authData, authData);
|
||||||
|
// newServerConfig.insert(
|
||||||
|
|
||||||
m_serversModel->editServer(newServerConfig, serverIndex);
|
m_serversModel->editServer(newServerConfig, serverIndex);
|
||||||
if (reloadServiceConfig) {
|
if (reloadServiceConfig) {
|
||||||
@@ -269,37 +270,54 @@ bool ApiConfigsController::updateServiceFromGateway(const int serverIndex, const
|
|||||||
|
|
||||||
bool ApiConfigsController::updateServiceFromTelegram(const int serverIndex)
|
bool ApiConfigsController::updateServiceFromTelegram(const int serverIndex)
|
||||||
{
|
{
|
||||||
|
auto serverConfig = m_serversModel->getServerConfig(serverIndex);
|
||||||
|
auto installationUuid = m_settings->getInstallationUuid(true);
|
||||||
|
|
||||||
#ifdef Q_OS_IOS
|
#ifdef Q_OS_IOS
|
||||||
IosController::Instance()->requestInetAccess();
|
IosController::Instance()->requestInetAccess();
|
||||||
QThread::msleep(10);
|
QThread::msleep(10);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
GatewayController gatewayController(m_settings->getGatewayEndpoint(), m_settings->isDevGatewayEnv(), apiDefs::requestTimeoutMsecs);
|
if (serverConfig.value(config_key::configVersion).toInt()) {
|
||||||
|
QNetworkRequest request;
|
||||||
|
request.setTransferTimeout(apiDefs::requestTimeoutMsecs);
|
||||||
|
request.setHeader(QNetworkRequest::ContentTypeHeader, "application/json");
|
||||||
|
request.setRawHeader("Authorization", "Api-Key " + serverConfig.value(configKey::accessToken).toString().toUtf8());
|
||||||
|
QString endpoint = serverConfig.value(configKey::apiEdnpoint).toString();
|
||||||
|
request.setUrl(endpoint);
|
||||||
|
|
||||||
auto serverConfig = m_serversModel->getServerConfig(serverIndex);
|
QString protocol = serverConfig.value(configKey::protocol).toString();
|
||||||
auto installationUuid = m_settings->getInstallationUuid(true);
|
|
||||||
|
|
||||||
QString serviceProtocol = serverConfig.value(configKey::protocol).toString();
|
ApiPayloadData apiPayloadData = generateApiPayloadData(protocol);
|
||||||
ApiPayloadData apiPayloadData = generateApiPayloadData(serviceProtocol);
|
|
||||||
|
|
||||||
QJsonObject apiPayload = fillApiPayload(serviceProtocol, apiPayloadData);
|
QJsonObject apiPayload = fillApiPayload(protocol, apiPayloadData);
|
||||||
apiPayload[configKey::uuid] = installationUuid;
|
apiPayload[configKey::uuid] = installationUuid;
|
||||||
apiPayload[configKey::accessToken] = serverConfig.value(configKey::accessToken).toString();
|
|
||||||
apiPayload[configKey::apiEndpoint] = serverConfig.value(configKey::apiEndpoint).toString();
|
|
||||||
|
|
||||||
QByteArray responseBody;
|
QByteArray requestBody = QJsonDocument(apiPayload).toJson();
|
||||||
ErrorCode errorCode = gatewayController.post(QString("%1v1/proxy_config"), apiPayload, responseBody);
|
|
||||||
|
|
||||||
if (errorCode == ErrorCode::NoError) {
|
QNetworkReply *reply = amnApp->networkManager()->post(request, requestBody);
|
||||||
fillServerConfig(serviceProtocol, apiPayloadData, responseBody, serverConfig);
|
|
||||||
|
|
||||||
|
QEventLoop wait;
|
||||||
|
connect(reply, &QNetworkReply::finished, &wait, &QEventLoop::quit);
|
||||||
|
|
||||||
|
QList<QSslError> sslErrors;
|
||||||
|
connect(reply, &QNetworkReply::sslErrors, [this, &sslErrors](const QList<QSslError> &errors) { sslErrors = errors; });
|
||||||
|
wait.exec();
|
||||||
|
|
||||||
|
auto errorCode = apiUtils::checkNetworkReplyErrors(sslErrors, reply);
|
||||||
|
if (errorCode != ErrorCode::NoError) {
|
||||||
|
reply->deleteLater();
|
||||||
|
emit errorOccurred(errorCode);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto apiResponseBody = reply->readAll();
|
||||||
|
reply->deleteLater();
|
||||||
|
fillServerConfig(protocol, apiPayloadData, apiResponseBody, serverConfig);
|
||||||
m_serversModel->editServer(serverConfig, serverIndex);
|
m_serversModel->editServer(serverConfig, serverIndex);
|
||||||
emit updateServerFromApiFinished();
|
emit updateServerFromApiFinished();
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
emit errorOccurred(errorCode);
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ApiConfigsController::deactivateDevice()
|
bool ApiConfigsController::deactivateDevice()
|
||||||
@@ -310,7 +328,7 @@ bool ApiConfigsController::deactivateDevice()
|
|||||||
auto serverConfigObject = m_serversModel->getServerConfig(serverIndex);
|
auto serverConfigObject = m_serversModel->getServerConfig(serverIndex);
|
||||||
auto apiConfigObject = serverConfigObject.value(configKey::apiConfig).toObject();
|
auto apiConfigObject = serverConfigObject.value(configKey::apiConfig).toObject();
|
||||||
|
|
||||||
if (!apiUtils::isPremiumServer(serverConfigObject)) {
|
if (apiUtils::getConfigType(serverConfigObject) != apiDefs::ConfigType::AmneziaPremiumV2) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -345,7 +363,7 @@ bool ApiConfigsController::deactivateExternalDevice(const QString &uuid, const Q
|
|||||||
auto serverConfigObject = m_serversModel->getServerConfig(serverIndex);
|
auto serverConfigObject = m_serversModel->getServerConfig(serverIndex);
|
||||||
auto apiConfigObject = serverConfigObject.value(configKey::apiConfig).toObject();
|
auto apiConfigObject = serverConfigObject.value(configKey::apiConfig).toObject();
|
||||||
|
|
||||||
if (!apiUtils::isPremiumServer(serverConfigObject)) {
|
if (apiUtils::getConfigType(serverConfigObject) != apiDefs::ConfigType::AmneziaPremiumV2) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -62,7 +62,7 @@ bool ApiSettingsController::getAccountInfo(bool reload)
|
|||||||
|
|
||||||
QByteArray responseBody;
|
QByteArray responseBody;
|
||||||
|
|
||||||
if (apiUtils::isPremiumServer(serverConfig)) {
|
if (apiUtils::getConfigType(serverConfig) == apiDefs::ConfigType::AmneziaPremiumV2) {
|
||||||
ErrorCode errorCode = gatewayController.post(QString("%1v1/account_info"), apiPayload, responseBody);
|
ErrorCode errorCode = gatewayController.post(QString("%1v1/account_info"), apiPayload, responseBody);
|
||||||
if (errorCode != ErrorCode::NoError) {
|
if (errorCode != ErrorCode::NoError) {
|
||||||
emit errorOccurred(errorCode);
|
emit errorOccurred(errorCode);
|
||||||
|
|||||||
@@ -56,7 +56,7 @@ namespace
|
|||||||
} else if ((config.contains(xrayConfigPatternInbound)) && (config.contains(xrayConfigPatternOutbound))) {
|
} else if ((config.contains(xrayConfigPatternInbound)) && (config.contains(xrayConfigPatternOutbound))) {
|
||||||
return ConfigTypes::Xray;
|
return ConfigTypes::Xray;
|
||||||
} else if (config.contains(openVpnConfigPatternCli)
|
} else if (config.contains(openVpnConfigPatternCli)
|
||||||
&& (config.contains(openVpnConfigPatternDriver1) || config.contains(openVpnConfigPatternDriver2))) {
|
&& (config.contains(openVpnConfigPatternDriver1) || config.contains(openVpnConfigPatternDriver2))) {
|
||||||
return ConfigTypes::OpenVpn;
|
return ConfigTypes::OpenVpn;
|
||||||
}
|
}
|
||||||
return ConfigTypes::Invalid;
|
return ConfigTypes::Invalid;
|
||||||
@@ -94,8 +94,6 @@ bool ImportController::extractConfigFromFile(const QString &fileName)
|
|||||||
|
|
||||||
bool ImportController::extractConfigFromData(QString data)
|
bool ImportController::extractConfigFromData(QString data)
|
||||||
{
|
{
|
||||||
m_maliciousWarningText.clear();
|
|
||||||
|
|
||||||
QString config = data;
|
QString config = data;
|
||||||
QString prefix;
|
QString prefix;
|
||||||
QString errormsg;
|
QString errormsg;
|
||||||
@@ -660,7 +658,6 @@ void ImportController::checkForMaliciousStrings(const QJsonObject &serverConfig)
|
|||||||
if ((containerName == ContainerProps::containerToString(DockerContainer::OpenVpn))
|
if ((containerName == ContainerProps::containerToString(DockerContainer::OpenVpn))
|
||||||
|| (containerName == ContainerProps::containerToString(DockerContainer::Cloak))
|
|| (containerName == ContainerProps::containerToString(DockerContainer::Cloak))
|
||||||
|| (containerName == ContainerProps::containerToString(DockerContainer::ShadowSocks))) {
|
|| (containerName == ContainerProps::containerToString(DockerContainer::ShadowSocks))) {
|
||||||
|
|
||||||
QString protocolConfig =
|
QString protocolConfig =
|
||||||
containerConfig[ProtocolProps::protoToString(Proto::OpenVpn)].toObject()[config_key::last_config].toString();
|
containerConfig[ProtocolProps::protoToString(Proto::OpenVpn)].toObject()[config_key::last_config].toString();
|
||||||
QString protocolConfigJson = QJsonDocument::fromJson(protocolConfig.toUtf8()).object()[config_key::config].toString();
|
QString protocolConfigJson = QJsonDocument::fromJson(protocolConfig.toUtf8()).object()[config_key::config].toString();
|
||||||
@@ -682,11 +679,8 @@ void ImportController::checkForMaliciousStrings(const QJsonObject &serverConfig)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
m_maliciousWarningText = tr("This configuration contains an OpenVPN setup. OpenVPN configurations can include malicious "
|
|
||||||
"scripts, so only add it if you fully trust the provider of this config. ");
|
|
||||||
|
|
||||||
if (maliciousStrings.size() >= dangerousTagsMaxCount) {
|
if (maliciousStrings.size() >= dangerousTagsMaxCount) {
|
||||||
m_maliciousWarningText.push_back(tr("<br>In the imported configuration, potentially dangerous lines were found:"));
|
m_maliciousWarningText = tr("In the imported configuration, potentially dangerous lines were found:");
|
||||||
for (const auto &string : maliciousStrings) {
|
for (const auto &string : maliciousStrings) {
|
||||||
m_maliciousWarningText.push_back(QString("<br><i>%1</i>").arg(string));
|
m_maliciousWarningText.push_back(QString("<br><i>%1</i>").arg(string));
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -31,14 +31,12 @@ namespace PageLoader
|
|||||||
PageSettingsLogging,
|
PageSettingsLogging,
|
||||||
PageSettingsSplitTunneling,
|
PageSettingsSplitTunneling,
|
||||||
PageSettingsAppSplitTunneling,
|
PageSettingsAppSplitTunneling,
|
||||||
PageSettingsKillSwitch,
|
|
||||||
PageSettingsApiServerInfo,
|
PageSettingsApiServerInfo,
|
||||||
PageSettingsApiAvailableCountries,
|
PageSettingsApiAvailableCountries,
|
||||||
PageSettingsApiSupport,
|
PageSettingsApiSupport,
|
||||||
PageSettingsApiInstructions,
|
PageSettingsApiInstructions,
|
||||||
PageSettingsApiNativeConfigs,
|
PageSettingsApiNativeConfigs,
|
||||||
PageSettingsApiDevices,
|
PageSettingsApiDevices,
|
||||||
PageSettingsKillSwitchExceptions,
|
|
||||||
|
|
||||||
PageServiceSftpSettings,
|
PageServiceSftpSettings,
|
||||||
PageServiceTorWebsiteSettings,
|
PageServiceTorWebsiteSettings,
|
||||||
|
|||||||
@@ -245,23 +245,6 @@ bool SettingsController::isKillSwitchEnabled()
|
|||||||
void SettingsController::toggleKillSwitch(bool enable)
|
void SettingsController::toggleKillSwitch(bool enable)
|
||||||
{
|
{
|
||||||
m_settings->setKillSwitchEnabled(enable);
|
m_settings->setKillSwitchEnabled(enable);
|
||||||
emit killSwitchEnabledChanged();
|
|
||||||
if (enable == false) {
|
|
||||||
emit strictKillSwitchEnabledChanged(false);
|
|
||||||
} else {
|
|
||||||
emit strictKillSwitchEnabledChanged(isStrictKillSwitchEnabled());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool SettingsController::isStrictKillSwitchEnabled()
|
|
||||||
{
|
|
||||||
return m_settings->isStrictKillSwitchEnabled();
|
|
||||||
}
|
|
||||||
|
|
||||||
void SettingsController::toggleStrictKillSwitch(bool enable)
|
|
||||||
{
|
|
||||||
m_settings->setStrictKillSwitchEnabled(enable);
|
|
||||||
emit strictKillSwitchEnabledChanged(enable);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool SettingsController::isNotificationPermissionGranted()
|
bool SettingsController::isNotificationPermissionGranted()
|
||||||
|
|||||||
@@ -24,8 +24,6 @@ public:
|
|||||||
Q_PROPERTY(QString secondaryDns READ getSecondaryDns WRITE setSecondaryDns NOTIFY secondaryDnsChanged)
|
Q_PROPERTY(QString secondaryDns READ getSecondaryDns WRITE setSecondaryDns NOTIFY secondaryDnsChanged)
|
||||||
Q_PROPERTY(bool isLoggingEnabled READ isLoggingEnabled WRITE toggleLogging NOTIFY loggingStateChanged)
|
Q_PROPERTY(bool isLoggingEnabled READ isLoggingEnabled WRITE toggleLogging NOTIFY loggingStateChanged)
|
||||||
Q_PROPERTY(bool isNotificationPermissionGranted READ isNotificationPermissionGranted NOTIFY onNotificationStateChanged)
|
Q_PROPERTY(bool isNotificationPermissionGranted READ isNotificationPermissionGranted NOTIFY onNotificationStateChanged)
|
||||||
Q_PROPERTY(bool isKillSwitchEnabled READ isKillSwitchEnabled WRITE toggleKillSwitch NOTIFY killSwitchEnabledChanged)
|
|
||||||
Q_PROPERTY(bool strictKillSwitchEnabled READ isStrictKillSwitchEnabled WRITE toggleStrictKillSwitch NOTIFY strictKillSwitchEnabledChanged)
|
|
||||||
|
|
||||||
Q_PROPERTY(bool isDevModeEnabled READ isDevModeEnabled NOTIFY devModeEnabled)
|
Q_PROPERTY(bool isDevModeEnabled READ isDevModeEnabled NOTIFY devModeEnabled)
|
||||||
Q_PROPERTY(QString gatewayEndpoint READ getGatewayEndpoint WRITE setGatewayEndpoint NOTIFY gatewayEndpointChanged)
|
Q_PROPERTY(QString gatewayEndpoint READ getGatewayEndpoint WRITE setGatewayEndpoint NOTIFY gatewayEndpointChanged)
|
||||||
@@ -77,9 +75,6 @@ public slots:
|
|||||||
bool isKillSwitchEnabled();
|
bool isKillSwitchEnabled();
|
||||||
void toggleKillSwitch(bool enable);
|
void toggleKillSwitch(bool enable);
|
||||||
|
|
||||||
bool isStrictKillSwitchEnabled();
|
|
||||||
void toggleStrictKillSwitch(bool enable);
|
|
||||||
|
|
||||||
bool isNotificationPermissionGranted();
|
bool isNotificationPermissionGranted();
|
||||||
void requestNotificationPermission();
|
void requestNotificationPermission();
|
||||||
|
|
||||||
@@ -103,8 +98,6 @@ signals:
|
|||||||
void primaryDnsChanged();
|
void primaryDnsChanged();
|
||||||
void secondaryDnsChanged();
|
void secondaryDnsChanged();
|
||||||
void loggingStateChanged();
|
void loggingStateChanged();
|
||||||
void killSwitchEnabledChanged();
|
|
||||||
void strictKillSwitchEnabledChanged(bool enabled);
|
|
||||||
|
|
||||||
void restoreBackupFinished();
|
void restoreBackupFinished();
|
||||||
void changeSettingsFinished(const QString &finishedMessage);
|
void changeSettingsFinished(const QString &finishedMessage);
|
||||||
|
|||||||
@@ -1,86 +0,0 @@
|
|||||||
#include "allowed_dns_model.h"
|
|
||||||
|
|
||||||
AllowedDnsModel::AllowedDnsModel(std::shared_ptr<Settings> settings, QObject *parent)
|
|
||||||
: QAbstractListModel(parent), m_settings(settings)
|
|
||||||
{
|
|
||||||
fillDnsServers();
|
|
||||||
}
|
|
||||||
|
|
||||||
int AllowedDnsModel::rowCount(const QModelIndex &parent) const
|
|
||||||
{
|
|
||||||
Q_UNUSED(parent)
|
|
||||||
return m_dnsServers.size();
|
|
||||||
}
|
|
||||||
|
|
||||||
QVariant AllowedDnsModel::data(const QModelIndex &index, int role) const
|
|
||||||
{
|
|
||||||
if (!index.isValid() || index.row() < 0 || index.row() >= static_cast<int>(rowCount()))
|
|
||||||
return QVariant();
|
|
||||||
|
|
||||||
switch (role) {
|
|
||||||
case IpRole:
|
|
||||||
return m_dnsServers.at(index.row());
|
|
||||||
default:
|
|
||||||
return QVariant();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool AllowedDnsModel::addDns(const QString &ip)
|
|
||||||
{
|
|
||||||
if (m_dnsServers.contains(ip)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
beginInsertRows(QModelIndex(), rowCount(), rowCount());
|
|
||||||
m_dnsServers.append(ip);
|
|
||||||
m_settings->setAllowedDnsServers(m_dnsServers);
|
|
||||||
endInsertRows();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void AllowedDnsModel::addDnsList(const QStringList &dnsServers, bool replaceExisting)
|
|
||||||
{
|
|
||||||
beginResetModel();
|
|
||||||
|
|
||||||
if (replaceExisting) {
|
|
||||||
m_dnsServers.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
for (const QString &ip : dnsServers) {
|
|
||||||
if (!m_dnsServers.contains(ip)) {
|
|
||||||
m_dnsServers.append(ip);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
m_settings->setAllowedDnsServers(m_dnsServers);
|
|
||||||
endResetModel();
|
|
||||||
}
|
|
||||||
|
|
||||||
void AllowedDnsModel::removeDns(QModelIndex index)
|
|
||||||
{
|
|
||||||
if (!index.isValid() || index.row() >= m_dnsServers.size()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
beginRemoveRows(QModelIndex(), index.row(), index.row());
|
|
||||||
m_dnsServers.removeAt(index.row());
|
|
||||||
m_settings->setAllowedDnsServers(m_dnsServers);
|
|
||||||
endRemoveRows();
|
|
||||||
}
|
|
||||||
|
|
||||||
QStringList AllowedDnsModel::getCurrentDnsServers()
|
|
||||||
{
|
|
||||||
return m_dnsServers;
|
|
||||||
}
|
|
||||||
|
|
||||||
QHash<int, QByteArray> AllowedDnsModel::roleNames() const
|
|
||||||
{
|
|
||||||
QHash<int, QByteArray> roles;
|
|
||||||
roles[IpRole] = "ip";
|
|
||||||
return roles;
|
|
||||||
}
|
|
||||||
|
|
||||||
void AllowedDnsModel::fillDnsServers()
|
|
||||||
{
|
|
||||||
m_dnsServers = m_settings->allowedDnsServers();
|
|
||||||
}
|
|
||||||
@@ -1,37 +0,0 @@
|
|||||||
#ifndef ALLOWEDDNSMODEL_H
|
|
||||||
#define ALLOWEDDNSMODEL_H
|
|
||||||
|
|
||||||
#include <QAbstractListModel>
|
|
||||||
#include "settings.h"
|
|
||||||
|
|
||||||
class AllowedDnsModel : public QAbstractListModel
|
|
||||||
{
|
|
||||||
Q_OBJECT
|
|
||||||
|
|
||||||
public:
|
|
||||||
enum Roles {
|
|
||||||
IpRole = Qt::UserRole + 1
|
|
||||||
};
|
|
||||||
|
|
||||||
explicit AllowedDnsModel(std::shared_ptr<Settings> settings, QObject *parent = nullptr);
|
|
||||||
|
|
||||||
int rowCount(const QModelIndex &parent = QModelIndex()) const override;
|
|
||||||
QVariant data(const QModelIndex &index, int role = Qt::DisplayRole) const override;
|
|
||||||
|
|
||||||
public slots:
|
|
||||||
bool addDns(const QString &ip);
|
|
||||||
void addDnsList(const QStringList &dnsServers, bool replaceExisting);
|
|
||||||
void removeDns(QModelIndex index);
|
|
||||||
QStringList getCurrentDnsServers();
|
|
||||||
|
|
||||||
protected:
|
|
||||||
QHash<int, QByteArray> roleNames() const override;
|
|
||||||
|
|
||||||
private:
|
|
||||||
void fillDnsServers();
|
|
||||||
|
|
||||||
std::shared_ptr<Settings> m_settings;
|
|
||||||
QStringList m_dnsServers;
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif // ALLOWEDDNSMODEL_H
|
|
||||||
@@ -48,19 +48,15 @@ QVariant ApiAccountInfoModel::data(const QModelIndex &index, int role) const
|
|||||||
}
|
}
|
||||||
case ServiceDescriptionRole: {
|
case ServiceDescriptionRole: {
|
||||||
if (m_accountInfoData.configType == apiDefs::ConfigType::AmneziaPremiumV2) {
|
if (m_accountInfoData.configType == apiDefs::ConfigType::AmneziaPremiumV2) {
|
||||||
return tr("Classic VPN for seamless work, downloading large files, and watching videos. Access all websites and online "
|
return tr("Classic VPN for seamless work, downloading large files, and watching videos. Access all websites and online resources. "
|
||||||
"resources. "
|
|
||||||
"Speeds up to 200 Mbps");
|
"Speeds up to 200 Mbps");
|
||||||
} else if (m_accountInfoData.configType == apiDefs::ConfigType::AmneziaFreeV3) {
|
} else if (m_accountInfoData.configType == apiDefs::ConfigType::AmneziaFreeV3) {
|
||||||
return tr("Free unlimited access to a basic set of websites such as Facebook, Instagram, Twitter (X), Discord, Telegram and "
|
return tr("Free unlimited access to a basic set of websites such as Facebook, Instagram, Twitter (X), Discord, Telegram and "
|
||||||
"more. YouTube is not included in the free plan.");
|
"more. YouTube is not included in the free plan.");
|
||||||
} else {
|
|
||||||
return "";
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
case IsComponentVisibleRole: {
|
case IsComponentVisibleRole: {
|
||||||
return m_accountInfoData.configType == apiDefs::ConfigType::AmneziaPremiumV2
|
return m_accountInfoData.configType == apiDefs::ConfigType::AmneziaPremiumV2;
|
||||||
|| m_accountInfoData.configType == apiDefs::ConfigType::ExternalPremium;
|
|
||||||
}
|
}
|
||||||
case HasExpiredWorkerRole: {
|
case HasExpiredWorkerRole: {
|
||||||
for (int i = 0; i < m_issuedConfigsInfo.size(); i++) {
|
for (int i = 0; i < m_issuedConfigsInfo.size(); i++) {
|
||||||
@@ -97,8 +93,6 @@ void ApiAccountInfoModel::updateModel(const QJsonObject &accountInfoObject, cons
|
|||||||
|
|
||||||
m_accountInfoData = accountInfoData;
|
m_accountInfoData = accountInfoData;
|
||||||
|
|
||||||
m_supportInfo = accountInfoObject.value(apiDefs::key::supportInfo).toObject();
|
|
||||||
|
|
||||||
endResetModel();
|
endResetModel();
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -127,27 +121,12 @@ QJsonArray ApiAccountInfoModel::getIssuedConfigsInfo()
|
|||||||
|
|
||||||
QString ApiAccountInfoModel::getTelegramBotLink()
|
QString ApiAccountInfoModel::getTelegramBotLink()
|
||||||
{
|
{
|
||||||
return m_supportInfo.value(apiDefs::key::telegram).toString();
|
if (m_accountInfoData.configType == apiDefs::ConfigType::AmneziaFreeV3) {
|
||||||
}
|
return tr("amnezia_free_support_bot");
|
||||||
|
} else if (m_accountInfoData.configType == apiDefs::ConfigType::AmneziaPremiumV2) {
|
||||||
QString ApiAccountInfoModel::getEmailLink()
|
return tr("amnezia_premium_support_bot");
|
||||||
{
|
}
|
||||||
return m_supportInfo.value(apiDefs::key::email).toString();
|
return "";
|
||||||
}
|
|
||||||
|
|
||||||
QString ApiAccountInfoModel::getBillingEmailLink()
|
|
||||||
{
|
|
||||||
return m_supportInfo.value(apiDefs::key::billingEmail).toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
QString ApiAccountInfoModel::getSiteLink()
|
|
||||||
{
|
|
||||||
return m_supportInfo.value(apiDefs::key::websiteName).toString();
|
|
||||||
}
|
|
||||||
|
|
||||||
QString ApiAccountInfoModel::getFullSiteLink()
|
|
||||||
{
|
|
||||||
return m_supportInfo.value(apiDefs::key::website).toString();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
QHash<int, QByteArray> ApiAccountInfoModel::roleNames() const
|
QHash<int, QByteArray> ApiAccountInfoModel::roleNames() const
|
||||||
|
|||||||
@@ -33,12 +33,7 @@ public slots:
|
|||||||
|
|
||||||
QJsonArray getAvailableCountries();
|
QJsonArray getAvailableCountries();
|
||||||
QJsonArray getIssuedConfigsInfo();
|
QJsonArray getIssuedConfigsInfo();
|
||||||
|
|
||||||
QString getTelegramBotLink();
|
QString getTelegramBotLink();
|
||||||
QString getEmailLink();
|
|
||||||
QString getBillingEmailLink();
|
|
||||||
QString getSiteLink();
|
|
||||||
QString getFullSiteLink();
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
QHash<int, QByteArray> roleNames() const override;
|
QHash<int, QByteArray> roleNames() const override;
|
||||||
@@ -56,7 +51,6 @@ private:
|
|||||||
AccountInfoData m_accountInfoData;
|
AccountInfoData m_accountInfoData;
|
||||||
QJsonArray m_availableCountries;
|
QJsonArray m_availableCountries;
|
||||||
QJsonArray m_issuedConfigsInfo;
|
QJsonArray m_issuedConfigsInfo;
|
||||||
QJsonObject m_supportInfo;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // APIACCOUNTINFOMODEL_H
|
#endif // APIACCOUNTINFOMODEL_H
|
||||||
|
|||||||
@@ -1,11 +1,13 @@
|
|||||||
#include "languageModel.h"
|
#include "languageModel.h"
|
||||||
|
|
||||||
LanguageModel::LanguageModel(std::shared_ptr<Settings> settings, QObject *parent) : m_settings(settings), QAbstractListModel(parent)
|
LanguageModel::LanguageModel(std::shared_ptr<Settings> settings, QObject *parent)
|
||||||
|
: m_settings(settings), QAbstractListModel(parent)
|
||||||
{
|
{
|
||||||
QMetaEnum metaEnum = QMetaEnum::fromType<LanguageSettings::AvailableLanguageEnum>();
|
QMetaEnum metaEnum = QMetaEnum::fromType<LanguageSettings::AvailableLanguageEnum>();
|
||||||
for (int i = 0; i < metaEnum.keyCount(); i++) {
|
for (int i = 0; i < metaEnum.keyCount(); i++) {
|
||||||
m_availableLanguages.push_back(LanguageModelData { getLocalLanguageName(static_cast<LanguageSettings::AvailableLanguageEnum>(i)),
|
m_availableLanguages.push_back(
|
||||||
static_cast<LanguageSettings::AvailableLanguageEnum>(i) });
|
LanguageModelData {getLocalLanguageName(static_cast<LanguageSettings::AvailableLanguageEnum>(i)),
|
||||||
|
static_cast<LanguageSettings::AvailableLanguageEnum>(i) });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -48,7 +50,8 @@ QString LanguageModel::getLocalLanguageName(const LanguageSettings::AvailableLan
|
|||||||
case LanguageSettings::AvailableLanguageEnum::Burmese: strLanguage = "မြန်မာဘာသာ"; break;
|
case LanguageSettings::AvailableLanguageEnum::Burmese: strLanguage = "မြန်မာဘာသာ"; break;
|
||||||
case LanguageSettings::AvailableLanguageEnum::Urdu: strLanguage = "اُرْدُوْ"; break;
|
case LanguageSettings::AvailableLanguageEnum::Urdu: strLanguage = "اُرْدُوْ"; break;
|
||||||
case LanguageSettings::AvailableLanguageEnum::Hindi: strLanguage = "हिन्दी"; break;
|
case LanguageSettings::AvailableLanguageEnum::Hindi: strLanguage = "हिन्दी"; break;
|
||||||
default: break;
|
default:
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return strLanguage;
|
return strLanguage;
|
||||||
@@ -101,12 +104,11 @@ QString LanguageModel::getCurrentLanguageName()
|
|||||||
return m_availableLanguages[getCurrentLanguageIndex()].name;
|
return m_availableLanguages[getCurrentLanguageIndex()].name;
|
||||||
}
|
}
|
||||||
|
|
||||||
QString LanguageModel::getCurrentSiteUrl(const QString &path)
|
QString LanguageModel::getCurrentSiteUrl()
|
||||||
{
|
{
|
||||||
auto language = static_cast<LanguageSettings::AvailableLanguageEnum>(getCurrentLanguageIndex());
|
auto language = static_cast<LanguageSettings::AvailableLanguageEnum>(getCurrentLanguageIndex());
|
||||||
switch (language) {
|
switch (language) {
|
||||||
case LanguageSettings::AvailableLanguageEnum::Russian:
|
case LanguageSettings::AvailableLanguageEnum::Russian: return "https://storage.googleapis.com/amnezia/amnezia.org";
|
||||||
return "https://storage.googleapis.com/amnezia/amnezia.org" + (path.isEmpty() ? "" : (QString("?m-path=/%1").arg(path)));
|
default: return "https://amnezia.org";
|
||||||
default: return QString("https://amnezia.org") + (path.isEmpty() ? "" : (QString("/%1").arg(path)));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -59,7 +59,7 @@ public slots:
|
|||||||
int getCurrentLanguageIndex();
|
int getCurrentLanguageIndex();
|
||||||
int getLineHeightAppend();
|
int getLineHeightAppend();
|
||||||
QString getCurrentLanguageName();
|
QString getCurrentLanguageName();
|
||||||
QString getCurrentSiteUrl(const QString &path = "");
|
QString getCurrentSiteUrl();
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void updateTranslations(const QLocale &locale);
|
void updateTranslations(const QLocale &locale);
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ Rectangle {
|
|||||||
cursorShape: Qt.PointingHandCursor
|
cursorShape: Qt.PointingHandCursor
|
||||||
|
|
||||||
onClicked: function() {
|
onClicked: function() {
|
||||||
Qt.openUrlExternally(LanguageModel.getCurrentSiteUrl("premium"))
|
Qt.openUrlExternally(LanguageModel.getCurrentSiteUrl() + "/premium")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,73 +0,0 @@
|
|||||||
import QtQuick
|
|
||||||
import QtQuick.Controls
|
|
||||||
import QtQuick.Layouts
|
|
||||||
|
|
||||||
import Style 1.0
|
|
||||||
import "../Controls2"
|
|
||||||
import "../Controls2/TextTypes"
|
|
||||||
|
|
||||||
Item {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
property bool enabled: true
|
|
||||||
property string placeholderText: ""
|
|
||||||
property alias textField: searchField.textField
|
|
||||||
|
|
||||||
signal addClicked(string text)
|
|
||||||
signal moreClicked()
|
|
||||||
|
|
||||||
implicitWidth: 360
|
|
||||||
implicitHeight: 96
|
|
||||||
|
|
||||||
Rectangle {
|
|
||||||
id: background
|
|
||||||
anchors.fill: parent
|
|
||||||
color: "#0E0F12"
|
|
||||||
opacity: 0.85
|
|
||||||
z: -1
|
|
||||||
}
|
|
||||||
|
|
||||||
RowLayout {
|
|
||||||
id: addSiteButton
|
|
||||||
|
|
||||||
enabled: root.enabled
|
|
||||||
spacing: 2
|
|
||||||
|
|
||||||
anchors {
|
|
||||||
fill: parent
|
|
||||||
topMargin: 16
|
|
||||||
leftMargin: 16
|
|
||||||
rightMargin: 16
|
|
||||||
bottomMargin: 24
|
|
||||||
}
|
|
||||||
|
|
||||||
TextFieldWithHeaderType {
|
|
||||||
id: searchField
|
|
||||||
|
|
||||||
Layout.fillWidth: true
|
|
||||||
rightButtonClickedOnEnter: true
|
|
||||||
|
|
||||||
textField.placeholderText: root.placeholderText
|
|
||||||
buttonImageSource: "qrc:/images/controls/plus.svg"
|
|
||||||
|
|
||||||
clickedFunc: function() {
|
|
||||||
root.addClicked(textField.text)
|
|
||||||
textField.text = ""
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ImageButtonType {
|
|
||||||
id: addSiteButtonImage
|
|
||||||
implicitWidth: 56
|
|
||||||
implicitHeight: 56
|
|
||||||
|
|
||||||
image: "qrc:/images/controls/more-vertical.svg"
|
|
||||||
imageColor: AmneziaStyle.color.paleGray
|
|
||||||
|
|
||||||
onClicked: root.moreClicked()
|
|
||||||
|
|
||||||
Keys.onReturnPressed: addSiteButtonImage.clicked()
|
|
||||||
Keys.onEnterPressed: addSiteButtonImage.clicked()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -18,8 +18,7 @@ ListView {
|
|||||||
property var selectedText
|
property var selectedText
|
||||||
|
|
||||||
width: rootWidth
|
width: rootWidth
|
||||||
anchors.top: parent.top
|
height: contentItem.height
|
||||||
anchors.bottom: parent.bottom
|
|
||||||
|
|
||||||
clip: true
|
clip: true
|
||||||
snapMode: ListView.SnapToItem
|
snapMode: ListView.SnapToItem
|
||||||
|
|||||||
@@ -1,45 +0,0 @@
|
|||||||
import QtQuick
|
|
||||||
import QtQuick.Layouts
|
|
||||||
|
|
||||||
import Style 1.0
|
|
||||||
|
|
||||||
import "TextTypes"
|
|
||||||
|
|
||||||
Item {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
property string headerText
|
|
||||||
property int headerTextMaximumLineCount: 2
|
|
||||||
property int headerTextElide: Qt.ElideRight
|
|
||||||
property string descriptionText
|
|
||||||
property alias headerRow: headerRow
|
|
||||||
|
|
||||||
implicitWidth: content.implicitWidth
|
|
||||||
implicitHeight: content.implicitHeight
|
|
||||||
|
|
||||||
ColumnLayout {
|
|
||||||
id: content
|
|
||||||
anchors.fill: parent
|
|
||||||
|
|
||||||
RowLayout {
|
|
||||||
id: headerRow
|
|
||||||
|
|
||||||
Header1TextType {
|
|
||||||
id: header
|
|
||||||
Layout.fillWidth: true
|
|
||||||
text: root.headerText
|
|
||||||
maximumLineCount: root.headerTextMaximumLineCount
|
|
||||||
elide: root.headerTextElide
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ParagraphTextType {
|
|
||||||
id: description
|
|
||||||
Layout.topMargin: 16
|
|
||||||
Layout.fillWidth: true
|
|
||||||
text: root.descriptionText
|
|
||||||
color: AmneziaStyle.color.mutedGray
|
|
||||||
visible: root.descriptionText !== ""
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -239,7 +239,6 @@ Item {
|
|||||||
sourceComponent: root.listView
|
sourceComponent: root.listView
|
||||||
|
|
||||||
Layout.fillHeight: true
|
Layout.fillHeight: true
|
||||||
Layout.fillWidth: true
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,86 @@
|
|||||||
|
import QtQuick
|
||||||
|
import QtQuick.Layouts
|
||||||
|
|
||||||
|
import Style 1.0
|
||||||
|
|
||||||
|
import "TextTypes"
|
||||||
|
|
||||||
|
Item {
|
||||||
|
id: root
|
||||||
|
|
||||||
|
property string actionButtonImage
|
||||||
|
property var actionButtonFunction
|
||||||
|
|
||||||
|
property alias actionButton: headerActionButton
|
||||||
|
|
||||||
|
property string headerText
|
||||||
|
property int headerTextMaximumLineCount: 2
|
||||||
|
property int headerTextElide: Qt.ElideRight
|
||||||
|
|
||||||
|
property string descriptionText
|
||||||
|
|
||||||
|
implicitWidth: content.implicitWidth
|
||||||
|
implicitHeight: content.implicitHeight
|
||||||
|
|
||||||
|
ColumnLayout {
|
||||||
|
id: content
|
||||||
|
anchors.fill: parent
|
||||||
|
|
||||||
|
RowLayout {
|
||||||
|
Header1TextType {
|
||||||
|
id: header
|
||||||
|
|
||||||
|
Layout.fillWidth: true
|
||||||
|
|
||||||
|
text: root.headerText
|
||||||
|
maximumLineCount: root.headerTextMaximumLineCount
|
||||||
|
elide: root.headerTextElide
|
||||||
|
}
|
||||||
|
|
||||||
|
ImageButtonType {
|
||||||
|
id: headerActionButton
|
||||||
|
|
||||||
|
implicitWidth: 40
|
||||||
|
implicitHeight: 40
|
||||||
|
|
||||||
|
Layout.alignment: Qt.AlignRight
|
||||||
|
|
||||||
|
image: root.actionButtonImage
|
||||||
|
imageColor: AmneziaStyle.color.paleGray
|
||||||
|
|
||||||
|
visible: image ? true : false
|
||||||
|
|
||||||
|
onClicked: {
|
||||||
|
if (actionButtonFunction && typeof actionButtonFunction === "function") {
|
||||||
|
actionButtonFunction()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ParagraphTextType {
|
||||||
|
id: description
|
||||||
|
|
||||||
|
Layout.topMargin: 16
|
||||||
|
Layout.fillWidth: true
|
||||||
|
|
||||||
|
text: root.descriptionText
|
||||||
|
|
||||||
|
color: AmneziaStyle.color.mutedGray
|
||||||
|
|
||||||
|
visible: root.descriptionText !== ""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Keys.onEnterPressed: {
|
||||||
|
if (actionButtonFunction && typeof actionButtonFunction === "function") {
|
||||||
|
actionButtonFunction()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Keys.onReturnPressed: {
|
||||||
|
if (actionButtonFunction && typeof actionButtonFunction === "function") {
|
||||||
|
actionButtonFunction()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
import QtQuick
|
|
||||||
import QtQuick.Layouts
|
|
||||||
|
|
||||||
import Style 1.0
|
|
||||||
|
|
||||||
BaseHeaderType {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
property string actionButtonImage
|
|
||||||
property var actionButtonFunction
|
|
||||||
property alias actionButton: headerActionButton
|
|
||||||
|
|
||||||
Component.onCompleted: {
|
|
||||||
headerRow.children.push(headerActionButton)
|
|
||||||
}
|
|
||||||
|
|
||||||
ImageButtonType {
|
|
||||||
id: headerActionButton
|
|
||||||
implicitWidth: 40
|
|
||||||
implicitHeight: 40
|
|
||||||
Layout.alignment: Qt.AlignRight
|
|
||||||
image: root.actionButtonImage
|
|
||||||
imageColor: AmneziaStyle.color.paleGray
|
|
||||||
visible: image ? true : false
|
|
||||||
|
|
||||||
onClicked: {
|
|
||||||
if (actionButtonFunction && typeof actionButtonFunction === "function") {
|
|
||||||
actionButtonFunction()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Keys.onEnterPressed: {
|
|
||||||
if (actionButtonFunction && typeof actionButtonFunction === "function") {
|
|
||||||
actionButtonFunction()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Keys.onReturnPressed: {
|
|
||||||
if (actionButtonFunction && typeof actionButtonFunction === "function") {
|
|
||||||
actionButtonFunction()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,28 +0,0 @@
|
|||||||
import QtQuick
|
|
||||||
import QtQuick.Layouts
|
|
||||||
|
|
||||||
import Style 1.0
|
|
||||||
|
|
||||||
BaseHeaderType {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
property var switcherFunction
|
|
||||||
property bool showSwitcher: false
|
|
||||||
property alias switcher: headerSwitcher
|
|
||||||
|
|
||||||
Component.onCompleted: {
|
|
||||||
headerRow.children.push(headerSwitcher)
|
|
||||||
}
|
|
||||||
|
|
||||||
SwitcherType {
|
|
||||||
id: headerSwitcher
|
|
||||||
Layout.alignment: Qt.AlignRight
|
|
||||||
visible: root.showSwitcher
|
|
||||||
|
|
||||||
onToggled: {
|
|
||||||
if (switcherFunction && typeof switcherFunction === "function") {
|
|
||||||
switcherFunction(checked)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -20,11 +20,7 @@ RadioButton {
|
|||||||
property string selectedColor: AmneziaStyle.color.transparent
|
property string selectedColor: AmneziaStyle.color.transparent
|
||||||
|
|
||||||
property string textColor: AmneziaStyle.color.paleGray
|
property string textColor: AmneziaStyle.color.paleGray
|
||||||
property string textDisabledColor: AmneziaStyle.color.mutedGray
|
|
||||||
property string selectedTextColor: AmneziaStyle.color.goldenApricot
|
property string selectedTextColor: AmneziaStyle.color.goldenApricot
|
||||||
property string selectedTextDisabledColor: AmneziaStyle.color.burntOrange
|
|
||||||
property string descriptionColor: AmneziaStyle.color.mutedGray
|
|
||||||
property string descriptionDisabledColor: AmneziaStyle.color.charcoalGray
|
|
||||||
|
|
||||||
property string borderFocusedColor: AmneziaStyle.color.paleGray
|
property string borderFocusedColor: AmneziaStyle.color.paleGray
|
||||||
property int borderFocusedWidth: 1
|
property int borderFocusedWidth: 1
|
||||||
@@ -34,12 +30,6 @@ RadioButton {
|
|||||||
|
|
||||||
property bool isFocusable: true
|
property bool isFocusable: true
|
||||||
|
|
||||||
|
|
||||||
property string radioButtonInnerCirclePressedSource: "qrc:/images/controls/radio-button-inner-circle-pressed.png"
|
|
||||||
property string radioButtonInnerCircleSource: "qrc:/images/controls/radio-button-inner-circle.png"
|
|
||||||
property string radioButtonPressedSource: "qrc:/images/controls/radio-button-pressed.svg"
|
|
||||||
property string radioButtonDefaultSource: "qrc:/images/controls/radio-button.svg"
|
|
||||||
|
|
||||||
Keys.onTabPressed: {
|
Keys.onTabPressed: {
|
||||||
FocusController.nextKeyTabItem()
|
FocusController.nextKeyTabItem()
|
||||||
}
|
}
|
||||||
@@ -104,15 +94,14 @@ RadioButton {
|
|||||||
if (showImage) {
|
if (showImage) {
|
||||||
return imageSource
|
return imageSource
|
||||||
} else if (root.pressed) {
|
} else if (root.pressed) {
|
||||||
return root.radioButtonInnerCirclePressedSource
|
return "qrc:/images/controls/radio-button-inner-circle-pressed.png"
|
||||||
} else if (root.checked) {
|
} else if (root.checked) {
|
||||||
return root.radioButtonInnerCircleSource
|
return "qrc:/images/controls/radio-button-inner-circle.png"
|
||||||
}
|
}
|
||||||
|
|
||||||
return ""
|
return ""
|
||||||
}
|
}
|
||||||
|
|
||||||
opacity: root.enabled ? 1.0 : 0.3
|
|
||||||
anchors.centerIn: parent
|
anchors.centerIn: parent
|
||||||
|
|
||||||
width: 24
|
width: 24
|
||||||
@@ -124,13 +113,12 @@ RadioButton {
|
|||||||
if (showImage) {
|
if (showImage) {
|
||||||
return ""
|
return ""
|
||||||
} else if (root.pressed || root.checked) {
|
} else if (root.pressed || root.checked) {
|
||||||
return root.radioButtonPressedSource
|
return "qrc:/images/controls/radio-button-pressed.svg"
|
||||||
} else {
|
} else {
|
||||||
return root.radioButtonDefaultSource
|
return "qrc:/images/controls/radio-button.svg"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
opacity: root.enabled ? 1.0 : 0.3
|
|
||||||
anchors.centerIn: parent
|
anchors.centerIn: parent
|
||||||
|
|
||||||
width: 24
|
width: 24
|
||||||
@@ -160,11 +148,10 @@ RadioButton {
|
|||||||
elide: root.textElide
|
elide: root.textElide
|
||||||
|
|
||||||
color: {
|
color: {
|
||||||
if (root.enabled) {
|
if (root.checked) {
|
||||||
return root.checked ? selectedTextColor : textColor
|
return selectedTextColor
|
||||||
} else {
|
|
||||||
return root.checked ? selectedTextDisabledColor : textDisabledColor
|
|
||||||
}
|
}
|
||||||
|
return textColor
|
||||||
}
|
}
|
||||||
|
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
@@ -177,7 +164,7 @@ RadioButton {
|
|||||||
CaptionTextType {
|
CaptionTextType {
|
||||||
id: description
|
id: description
|
||||||
|
|
||||||
color: root.enabled ? root.descriptionColor : root.descriptionDisabledColor
|
color: AmneziaStyle.color.mutedGray
|
||||||
text: root.descriptionText
|
text: root.descriptionText
|
||||||
|
|
||||||
visible: root.descriptionText !== ""
|
visible: root.descriptionText !== ""
|
||||||
@@ -190,7 +177,6 @@ RadioButton {
|
|||||||
MouseArea {
|
MouseArea {
|
||||||
anchors.fill: root
|
anchors.fill: root
|
||||||
cursorShape: Qt.PointingHandCursor
|
cursorShape: Qt.PointingHandCursor
|
||||||
preventStealing: false
|
|
||||||
enabled: false
|
enabled: false
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -56,7 +56,7 @@ PageType {
|
|||||||
anchors.rightMargin: 16
|
anchors.rightMargin: 16
|
||||||
anchors.leftMargin: 16
|
anchors.leftMargin: 16
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.topMargin: 20
|
Layout.topMargin: 20
|
||||||
|
|
||||||
|
|||||||
@@ -39,7 +39,7 @@ PageType {
|
|||||||
header: ColumnLayout {
|
header: ColumnLayout {
|
||||||
width: listView.width
|
width: listView.width
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
id: header
|
id: header
|
||||||
|
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|||||||
@@ -91,7 +91,7 @@ PageType {
|
|||||||
|
|
||||||
spacing: 0
|
spacing: 0
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|
||||||
headerText: qsTr("AmneziaWG settings")
|
headerText: qsTr("AmneziaWG settings")
|
||||||
|
|||||||
@@ -91,7 +91,7 @@ PageType {
|
|||||||
|
|
||||||
spacing: 0
|
spacing: 0
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|
||||||
headerText: qsTr("AmneziaWG settings")
|
headerText: qsTr("AmneziaWG settings")
|
||||||
|
|||||||
@@ -76,7 +76,7 @@ PageType {
|
|||||||
|
|
||||||
spacing: 0
|
spacing: 0
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|
||||||
headerText: qsTr("Cloak settings")
|
headerText: qsTr("Cloak settings")
|
||||||
|
|||||||
@@ -75,7 +75,7 @@ PageType {
|
|||||||
|
|
||||||
spacing: 0
|
spacing: 0
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|
||||||
headerText: qsTr("OpenVPN settings")
|
headerText: qsTr("OpenVPN settings")
|
||||||
|
|||||||
@@ -32,7 +32,7 @@ PageType {
|
|||||||
id: backButton
|
id: backButton
|
||||||
}
|
}
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.leftMargin: 16
|
Layout.leftMargin: 16
|
||||||
Layout.rightMargin: 16
|
Layout.rightMargin: 16
|
||||||
|
|||||||
@@ -78,7 +78,7 @@ PageType {
|
|||||||
|
|
||||||
spacing: 0
|
spacing: 0
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|
||||||
headerText: qsTr("Shadowsocks settings")
|
headerText: qsTr("Shadowsocks settings")
|
||||||
|
|||||||
@@ -85,7 +85,7 @@ PageType {
|
|||||||
|
|
||||||
spacing: 0
|
spacing: 0
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|
||||||
headerText: qsTr("WG settings")
|
headerText: qsTr("WG settings")
|
||||||
|
|||||||
@@ -77,7 +77,7 @@ PageType {
|
|||||||
|
|
||||||
spacing: 0
|
spacing: 0
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
headerText: qsTr("WG settings")
|
headerText: qsTr("WG settings")
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -75,7 +75,7 @@ PageType {
|
|||||||
|
|
||||||
spacing: 0
|
spacing: 0
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
headerText: qsTr("XRay settings")
|
headerText: qsTr("XRay settings")
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -43,7 +43,7 @@ PageType {
|
|||||||
anchors.left: parent.left
|
anchors.left: parent.left
|
||||||
anchors.right: parent.right
|
anchors.right: parent.right
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
id: header
|
id: header
|
||||||
|
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|||||||
@@ -85,7 +85,7 @@ PageType {
|
|||||||
|
|
||||||
spacing: 0
|
spacing: 0
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.leftMargin: 16
|
Layout.leftMargin: 16
|
||||||
Layout.rightMargin: 16
|
Layout.rightMargin: 16
|
||||||
|
|||||||
@@ -77,7 +77,7 @@ PageType {
|
|||||||
|
|
||||||
spacing: 0
|
spacing: 0
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.leftMargin: 16
|
Layout.leftMargin: 16
|
||||||
Layout.rightMargin: 16
|
Layout.rightMargin: 16
|
||||||
@@ -217,7 +217,7 @@ PageType {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|
||||||
headerText: qsTr("SOCKS5 settings")
|
headerText: qsTr("SOCKS5 settings")
|
||||||
|
|||||||
@@ -54,7 +54,7 @@ PageType {
|
|||||||
|
|
||||||
spacing: 0
|
spacing: 0
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.leftMargin: 16
|
Layout.leftMargin: 16
|
||||||
Layout.rightMargin: 16
|
Layout.rightMargin: 16
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ PageType {
|
|||||||
|
|
||||||
spacing: 0
|
spacing: 0
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
id: header
|
id: header
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.topMargin: 24
|
Layout.topMargin: 24
|
||||||
|
|||||||
@@ -252,7 +252,7 @@ PageType {
|
|||||||
text: qsTr("Privacy Policy")
|
text: qsTr("Privacy Policy")
|
||||||
|
|
||||||
clickedFunc: function() {
|
clickedFunc: function() {
|
||||||
Qt.openUrlExternally(LanguageModel.getCurrentSiteUrl("policy"))
|
Qt.openUrlExternally(LanguageModel.getCurrentSiteUrl() + "/policy")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -69,7 +69,7 @@ PageType {
|
|||||||
Layout.topMargin: 20
|
Layout.topMargin: 20
|
||||||
}
|
}
|
||||||
|
|
||||||
HeaderTypeWithButton {
|
HeaderType {
|
||||||
id: headerContent
|
id: headerContent
|
||||||
objectName: "headerContent"
|
objectName: "headerContent"
|
||||||
|
|
||||||
@@ -135,6 +135,12 @@ PageType {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
MouseArea {
|
||||||
|
anchors.fill: containerRadioButton
|
||||||
|
cursorShape: Qt.PointingHandCursor
|
||||||
|
enabled: false
|
||||||
|
}
|
||||||
|
|
||||||
Keys.onEnterPressed: {
|
Keys.onEnterPressed: {
|
||||||
if (checkable) {
|
if (checkable) {
|
||||||
checked = true
|
checked = true
|
||||||
|
|||||||
@@ -35,7 +35,7 @@ PageType {
|
|||||||
id: backButton
|
id: backButton
|
||||||
}
|
}
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
id: header
|
id: header
|
||||||
|
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|||||||
@@ -91,7 +91,7 @@ PageType {
|
|||||||
id: backButton
|
id: backButton
|
||||||
}
|
}
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
id: header
|
id: header
|
||||||
|
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|||||||
@@ -38,7 +38,7 @@ PageType {
|
|||||||
id: backButton
|
id: backButton
|
||||||
}
|
}
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
id: header
|
id: header
|
||||||
|
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|||||||
@@ -93,7 +93,7 @@ PageType {
|
|||||||
Layout.topMargin: 20
|
Layout.topMargin: 20
|
||||||
}
|
}
|
||||||
|
|
||||||
HeaderTypeWithButton {
|
HeaderType {
|
||||||
id: headerContent
|
id: headerContent
|
||||||
objectName: "headerContent"
|
objectName: "headerContent"
|
||||||
|
|
||||||
|
|||||||
@@ -28,24 +28,24 @@ PageType {
|
|||||||
id: techSupport
|
id: techSupport
|
||||||
|
|
||||||
readonly property string title: qsTr("Email")
|
readonly property string title: qsTr("Email")
|
||||||
readonly property string description: ApiAccountInfoModel.getEmailLink()
|
readonly property string description: qsTr("support@amnezia.org")
|
||||||
readonly property string link: "mailto:" + ApiAccountInfoModel.getEmailLink()
|
readonly property string link: "mailto:support@amnezia.org"
|
||||||
}
|
}
|
||||||
|
|
||||||
QtObject {
|
QtObject {
|
||||||
id: paymentSupport
|
id: paymentSupport
|
||||||
|
|
||||||
readonly property string title: qsTr("Email Billing & Orders")
|
readonly property string title: qsTr("Email Billing & Orders")
|
||||||
readonly property string description: ApiAccountInfoModel.getBillingEmailLink()
|
readonly property string description: qsTr("help@vpnpay.io")
|
||||||
readonly property string link: "mailto:" + ApiAccountInfoModel.getBillingEmailLink()
|
readonly property string link: "mailto:help@vpnpay.io"
|
||||||
}
|
}
|
||||||
|
|
||||||
QtObject {
|
QtObject {
|
||||||
id: site
|
id: site
|
||||||
|
|
||||||
readonly property string title: qsTr("Website")
|
readonly property string title: qsTr("Website")
|
||||||
readonly property string description: ApiAccountInfoModel.getSiteLink()
|
readonly property string description: qsTr("amnezia.org")
|
||||||
readonly property string link: ApiAccountInfoModel.getFullSiteLink()
|
readonly property string link: LanguageModel.getCurrentSiteUrl()
|
||||||
}
|
}
|
||||||
|
|
||||||
property list<QtObject> supportModel: [
|
property list<QtObject> supportModel: [
|
||||||
@@ -71,7 +71,7 @@ PageType {
|
|||||||
id: backButton
|
id: backButton
|
||||||
}
|
}
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
id: header
|
id: header
|
||||||
|
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|||||||
@@ -79,22 +79,29 @@ PageType {
|
|||||||
id: backButton
|
id: backButton
|
||||||
}
|
}
|
||||||
|
|
||||||
HeaderTypeWithSwitcher {
|
RowLayout {
|
||||||
Layout.fillWidth: true
|
HeaderType {
|
||||||
Layout.leftMargin: 16
|
Layout.fillWidth: true
|
||||||
Layout.rightMargin: 16
|
Layout.leftMargin: 16
|
||||||
|
|
||||||
headerText: qsTr("App split tunneling")
|
headerText: qsTr("App split tunneling")
|
||||||
|
|
||||||
enabled: root.pageEnabled
|
|
||||||
showSwitcher: true
|
|
||||||
switcher {
|
|
||||||
checked: AppSplitTunnelingModel.isTunnelingEnabled
|
|
||||||
enabled: root.pageEnabled
|
enabled: root.pageEnabled
|
||||||
}
|
}
|
||||||
switcherFunction: function(checked) {
|
|
||||||
AppSplitTunnelingModel.toggleSplitTunneling(checked)
|
SwitcherType {
|
||||||
selector.text = root.routeModesModel[getRouteModesModelIndex()].name
|
id: switcher
|
||||||
|
|
||||||
|
Layout.fillWidth: true
|
||||||
|
Layout.rightMargin: 16
|
||||||
|
|
||||||
|
enabled: root.pageEnabled
|
||||||
|
|
||||||
|
checked: AppSplitTunnelingModel.isTunnelingEnabled
|
||||||
|
onToggled: {
|
||||||
|
AppSplitTunnelingModel.toggleSplitTunneling(checked)
|
||||||
|
selector.text = root.routeModesModel[getRouteModesModelIndex()].name
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -38,7 +38,7 @@ PageType {
|
|||||||
|
|
||||||
spacing: 0
|
spacing: 0
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.leftMargin: 16
|
Layout.leftMargin: 16
|
||||||
Layout.rightMargin: 16
|
Layout.rightMargin: 16
|
||||||
|
|||||||
@@ -60,7 +60,7 @@ PageType {
|
|||||||
|
|
||||||
spacing: 16
|
spacing: 16
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|
||||||
headerText: qsTr("Back up your configuration")
|
headerText: qsTr("Back up your configuration")
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ PageType {
|
|||||||
anchors.left: parent.left
|
anchors.left: parent.left
|
||||||
anchors.right: parent.right
|
anchors.right: parent.right
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
Layout.leftMargin: 16
|
Layout.leftMargin: 16
|
||||||
Layout.rightMargin: 16
|
Layout.rightMargin: 16
|
||||||
@@ -94,7 +94,9 @@ PageType {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
DividerType {}
|
DividerType {
|
||||||
|
visible: root.isAppSplitTinnelingEnabled
|
||||||
|
}
|
||||||
|
|
||||||
LabelWithButtonType {
|
LabelWithButtonType {
|
||||||
id: splitTunnelingButton2
|
id: splitTunnelingButton2
|
||||||
@@ -117,20 +119,29 @@ PageType {
|
|||||||
visible: root.isAppSplitTinnelingEnabled
|
visible: root.isAppSplitTinnelingEnabled
|
||||||
}
|
}
|
||||||
|
|
||||||
LabelWithButtonType {
|
SwitcherType {
|
||||||
id: killSwitchButton
|
id: killSwitchSwitcher
|
||||||
visible: !GC.isMobile()
|
visible: !GC.isMobile()
|
||||||
|
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
Layout.margins: 16
|
||||||
|
|
||||||
text: qsTr("KillSwitch")
|
text: qsTr("KillSwitch")
|
||||||
descriptionText: qsTr("Blocks network connections without VPN")
|
descriptionText: qsTr("Disables your internet if your encrypted VPN connection drops out for any reason.")
|
||||||
rightImageSource: "qrc:/images/controls/chevron-right.svg"
|
|
||||||
|
|
||||||
parentFlickable: fl
|
parentFlickable: fl
|
||||||
|
|
||||||
clickedFunction: function() {
|
checked: SettingsController.isKillSwitchEnabled()
|
||||||
PageController.goToPage(PageEnum.PageSettingsKillSwitch)
|
checkable: !ConnectionController.isConnected
|
||||||
|
onCheckedChanged: {
|
||||||
|
if (checked !== SettingsController.isKillSwitchEnabled()) {
|
||||||
|
SettingsController.toggleKillSwitch(checked)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
onClicked: {
|
||||||
|
if (!checkable) {
|
||||||
|
PageController.showNotificationMessage(qsTr("Cannot change KillSwitch settings during active connection"))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -50,7 +50,7 @@ PageType {
|
|||||||
|
|
||||||
spacing: 16
|
spacing: 16
|
||||||
|
|
||||||
BaseHeaderType {
|
HeaderType {
|
||||||
Layout.fillWidth: true
|
Layout.fillWidth: true
|
||||||
|
|
||||||
headerText: qsTr("DNS servers")
|
headerText: qsTr("DNS servers")
|
||||||
|
|||||||
@@ -1,123 +0,0 @@
|
|||||||
import QtQuick
|
|
||||||
import QtQuick.Controls
|
|
||||||
import QtQuick.Layouts
|
|
||||||
|
|
||||||
import PageEnum 1.0
|
|
||||||
import Style 1.0
|
|
||||||
|
|
||||||
import "./"
|
|
||||||
import "../Controls2"
|
|
||||||
import "../Config"
|
|
||||||
|
|
||||||
PageType {
|
|
||||||
id: root
|
|
||||||
|
|
||||||
BackButtonType {
|
|
||||||
id: backButton
|
|
||||||
anchors.top: parent.top
|
|
||||||
anchors.left: parent.left
|
|
||||||
anchors.right: parent.right
|
|
||||||
anchors.topMargin: 20
|
|
||||||
}
|
|
||||||
|
|
||||||
FlickableType {
|
|
||||||
id: fl
|
|
||||||
anchors.top: backButton.bottom
|
|
||||||
anchors.bottom: parent.bottom
|
|
||||||
contentHeight: content.height
|
|
||||||
|
|
||||||
ColumnLayout {
|
|
||||||
id: content
|
|
||||||
anchors.top: parent.top
|
|
||||||
anchors.left: parent.left
|
|
||||||
anchors.right: parent.right
|
|
||||||
|
|
||||||
HeaderTypeWithSwitcher {
|
|
||||||
Layout.fillWidth: true
|
|
||||||
Layout.leftMargin: 16
|
|
||||||
Layout.rightMargin: 16
|
|
||||||
|
|
||||||
headerText: qsTr("KillSwitch")
|
|
||||||
descriptionText: qsTr("Enable to ensure network traffic goes through a secure VPN tunnel, preventing accidental exposure of your IP and DNS queries if the connection drops")
|
|
||||||
|
|
||||||
showSwitcher: true
|
|
||||||
switcher {
|
|
||||||
checked: SettingsController.isKillSwitchEnabled
|
|
||||||
enabled: !ConnectionController.isConnected
|
|
||||||
}
|
|
||||||
switcherFunction: function(checked) {
|
|
||||||
if (!ConnectionController.isConnected) {
|
|
||||||
SettingsController.isKillSwitchEnabled = checked
|
|
||||||
} else {
|
|
||||||
PageController.showNotificationMessage(qsTr("Cannot change killSwitch settings during active connection"))
|
|
||||||
switcher.checked = SettingsController.isKillSwitchEnabled
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
VerticalRadioButton {
|
|
||||||
id: softKillSwitch
|
|
||||||
Layout.fillWidth: true
|
|
||||||
Layout.topMargin: 32
|
|
||||||
Layout.leftMargin: 16
|
|
||||||
Layout.rightMargin: 16
|
|
||||||
|
|
||||||
enabled: SettingsController.isKillSwitchEnabled && !ConnectionController.isConnected
|
|
||||||
checked: !SettingsController.strictKillSwitchEnabled
|
|
||||||
|
|
||||||
text: qsTr("Soft KillSwitch")
|
|
||||||
descriptionText: qsTr("Internet connection is blocked if VPN connection drops accidentally")
|
|
||||||
|
|
||||||
onClicked: function() {
|
|
||||||
SettingsController.strictKillSwitchEnabled = false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
DividerType {}
|
|
||||||
|
|
||||||
VerticalRadioButton {
|
|
||||||
id: strictKillSwitch
|
|
||||||
Layout.fillWidth: true
|
|
||||||
Layout.leftMargin: 16
|
|
||||||
Layout.rightMargin: 16
|
|
||||||
|
|
||||||
enabled: SettingsController.isKillSwitchEnabled && !ConnectionController.isConnected
|
|
||||||
checked: SettingsController.strictKillSwitchEnabled
|
|
||||||
|
|
||||||
text: qsTr("Strict KillSwitch")
|
|
||||||
descriptionText: qsTr("Internet connection is blocked even if VPN was turned off manually or not started")
|
|
||||||
|
|
||||||
onClicked: function() {
|
|
||||||
var headerText = qsTr("Just a little heads-up")
|
|
||||||
var descriptionText = qsTr("If you disconnect from VPN or the VPN connection drops while the Strict Kill Switch is turned on, your internet access will be disabled. To restore it, connect to VPN, change the Kill Switch mode or turn the Kill Switch off.")
|
|
||||||
var yesButtonText = qsTr("Continue")
|
|
||||||
var noButtonText = qsTr("Cancel")
|
|
||||||
|
|
||||||
var yesButtonFunction = function() {
|
|
||||||
SettingsController.strictKillSwitchEnabled = true
|
|
||||||
}
|
|
||||||
var noButtonFunction = function() {
|
|
||||||
}
|
|
||||||
|
|
||||||
showQuestionDrawer(headerText, descriptionText, yesButtonText, noButtonText, yesButtonFunction, noButtonFunction)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
DividerType {}
|
|
||||||
|
|
||||||
LabelWithButtonType {
|
|
||||||
Layout.topMargin: 32
|
|
||||||
Layout.fillWidth: true
|
|
||||||
|
|
||||||
enabled: true
|
|
||||||
text: qsTr("DNS Exceptions")
|
|
||||||
descriptionText: qsTr("DNS servers from the list will remain accessible when Kill Switch is triggered")
|
|
||||||
rightImageSource: "qrc:/images/controls/chevron-right.svg"
|
|
||||||
|
|
||||||
clickedFunction: function() {
|
|
||||||
PageController.goToPage(PageEnum.PageSettingsKillSwitchExceptions)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user